From 2a87bd89951204a5dd15b60ba86cbba72fc3b088 Mon Sep 17 00:00:00 2001 From: Campbell Barton Date: Fri, 3 Nov 2017 20:26:35 +1100 Subject: Cleanup: split interface_regions.c Each region type is quite separate and file was getting too big. --- .../blender/editors/interface/interface_regions.c | 3378 +------------------- 1 file changed, 9 insertions(+), 3369 deletions(-) (limited to 'source/blender/editors/interface/interface_regions.c') diff --git a/source/blender/editors/interface/interface_regions.c b/source/blender/editors/interface/interface_regions.c index 80cf19bdd0b..dd1d38bbe2e 100644 --- a/source/blender/editors/interface/interface_regions.c +++ b/source/blender/editors/interface/interface_regions.c @@ -25,83 +25,27 @@ /** \file blender/editors/interface/interface_regions.c * \ingroup edinterface + * + * General Interface Region Code + * + * \note Most logic is now in 'interface_region_*.c' */ -#include -#include -#include -#include - -#include "MEM_guardedalloc.h" - -#include "DNA_userdef_types.h" - -#include "BLI_math.h" -#include "BLI_blenlib.h" #include "BLI_utildefines.h" -#include "BLI_ghash.h" - -#include "PIL_time.h" +#include "MEM_guardedalloc.h" +#include "BLI_listbase.h" #include "BKE_context.h" #include "BKE_screen.h" -#include "BKE_report.h" -#include "BKE_global.h" #include "WM_api.h" -#include "WM_types.h" #include "wm_draw.h" -#include "wm_subwindow.h" - -#include "RNA_access.h" - -#include "BIF_gl.h" - -#include "UI_interface.h" -#include "UI_interface_icons.h" -#include "UI_view2d.h" - -#include "BLF_api.h" -#include "BLT_translation.h" #include "ED_screen.h" -#include "IMB_colormanagement.h" - -#include "interface_intern.h" - -#define MENU_PADDING (int)(0.2f * UI_UNIT_Y) -#define MENU_BORDER (int)(0.3f * U.widget_unit) - - -bool ui_but_menu_step_poll(const uiBut *but) -{ - BLI_assert(but->type == UI_BTYPE_MENU); - - /* currently only RNA buttons */ - return ((but->menu_step_func != NULL) || - (but->rnaprop && RNA_property_type(but->rnaprop) == PROP_ENUM)); -} - -int ui_but_menu_step(uiBut *but, int direction) -{ - if (ui_but_menu_step_poll(but)) { - if (but->menu_step_func) { - return but->menu_step_func(but->block->evil_C, direction, but->poin); - } - else { - const int curval = RNA_property_enum_get(&but->rnapoin, but->rnaprop); - return RNA_property_enum_step(but->block->evil_C, &but->rnapoin, but->rnaprop, curval, direction); - } - } - - printf("%s: cannot cycle button '%s'\n", __func__, but->str); - return 0; -} - -/******************** Creating Temporary regions ******************/ +#include "interface_regions_intern.h" -static ARegion *ui_region_temp_add(bScreen *sc) +ARegion *ui_region_temp_add(bScreen *sc) { ARegion *ar; @@ -114,7 +58,7 @@ static ARegion *ui_region_temp_add(bScreen *sc) return ar; } -static void ui_region_temp_remove(bContext *C, bScreen *sc, ARegion *ar) +void ui_region_temp_remove(bContext *C, bScreen *sc, ARegion *ar) { wmWindow *win = CTX_wm_window(C); if (win) @@ -124,3307 +68,3 @@ static void ui_region_temp_remove(bContext *C, bScreen *sc, ARegion *ar) BKE_area_region_free(NULL, ar); /* NULL: no spacetype */ BLI_freelinkN(&sc->regionbase, ar); } - -/************************* Creating Tooltips **********************/ - -#define UI_TIP_PAD_FAC 1.3f -#define UI_TIP_PADDING (int)(UI_TIP_PAD_FAC * UI_UNIT_Y) -#define UI_TIP_MAXWIDTH 600 - -#define MAX_TOOLTIP_LINES 8 -typedef struct uiTooltipData { - rcti bbox; - uiFontStyle fstyle; - char lines[MAX_TOOLTIP_LINES][2048]; - char header[2048], active_info[2048]; - struct { - enum { - UI_TIP_STYLE_NORMAL = 0, - UI_TIP_STYLE_HEADER, - UI_TIP_STYLE_MONO, - } style : 3; - enum { - UI_TIP_LC_MAIN = 0, /* primary text */ - UI_TIP_LC_VALUE, /* the value of buttons (also shortcuts) */ - UI_TIP_LC_ACTIVE, /* titles of active enum values */ - UI_TIP_LC_NORMAL, /* regular text */ - UI_TIP_LC_PYTHON, /* Python snippet */ - UI_TIP_LC_ALERT, /* description of why operator can't run */ - } color_id : 4; - int is_pad : 1; - } format[MAX_TOOLTIP_LINES]; - - struct { - unsigned int x_pos; /* x cursor position at the end of the last line */ - unsigned int lines; /* number of lines, 1 or more with word-wrap */ - } line_geom[MAX_TOOLTIP_LINES]; - - int wrap_width; - - int totline; - int toth, lineh; -} uiTooltipData; - -#define UI_TIP_LC_MAX 6 - -BLI_STATIC_ASSERT(UI_TIP_LC_MAX == UI_TIP_LC_ALERT + 1, "invalid lc-max"); -BLI_STATIC_ASSERT(sizeof(((uiTooltipData *)NULL)->format[0]) <= sizeof(int), "oversize"); - -static void rgb_tint( - float col[3], - float h, float h_strength, - float v, float v_strength) -{ - float col_hsv_from[3]; - float col_hsv_to[3]; - - rgb_to_hsv_v(col, col_hsv_from); - - col_hsv_to[0] = h; - col_hsv_to[1] = h_strength; - col_hsv_to[2] = (col_hsv_from[2] * (1.0f - v_strength)) + (v * v_strength); - - hsv_to_rgb_v(col_hsv_to, col); -} - -static void ui_tooltip_region_draw_cb(const bContext *UNUSED(C), ARegion *ar) -{ - const float pad_px = UI_TIP_PADDING; - uiTooltipData *data = ar->regiondata; - uiWidgetColors *theme = ui_tooltip_get_theme(); - rcti bbox = data->bbox; - float tip_colors[UI_TIP_LC_MAX][3]; - - float *main_color = tip_colors[UI_TIP_LC_MAIN]; /* the color from the theme */ - float *value_color = tip_colors[UI_TIP_LC_VALUE]; - float *active_color = tip_colors[UI_TIP_LC_ACTIVE]; - float *normal_color = tip_colors[UI_TIP_LC_NORMAL]; - float *python_color = tip_colors[UI_TIP_LC_PYTHON]; - float *alert_color = tip_colors[UI_TIP_LC_ALERT]; - - float background_color[3]; - float tone_bg; - int i, multisample_enabled; - - /* disable AA, makes widgets too blurry */ - multisample_enabled = glIsEnabled(GL_MULTISAMPLE); - if (multisample_enabled) - glDisable(GL_MULTISAMPLE); - - wmOrtho2_region_pixelspace(ar); - - /* draw background */ - ui_draw_tooltip_background(UI_style_get(), NULL, &bbox); - - /* set background_color */ - rgb_uchar_to_float(background_color, (const unsigned char *)theme->inner); - - /* calculate normal_color */ - rgb_uchar_to_float(main_color, (const unsigned char *)theme->text); - copy_v3_v3(active_color, main_color); - copy_v3_v3(normal_color, main_color); - copy_v3_v3(python_color, main_color); - copy_v3_v3(alert_color, main_color); - copy_v3_v3(value_color, main_color); - - /* find the brightness difference between background and text colors */ - - tone_bg = rgb_to_grayscale(background_color); - /* tone_fg = rgb_to_grayscale(main_color); */ - - /* mix the colors */ - rgb_tint(value_color, 0.0f, 0.0f, tone_bg, 0.2f); /* light gray */ - rgb_tint(active_color, 0.6f, 0.2f, tone_bg, 0.2f); /* light blue */ - rgb_tint(normal_color, 0.0f, 0.0f, tone_bg, 0.4f); /* gray */ - rgb_tint(python_color, 0.0f, 0.0f, tone_bg, 0.5f); /* dark gray */ - rgb_tint(alert_color, 0.0f, 0.8f, tone_bg, 0.1f); /* red */ - - /* draw text */ - BLF_wordwrap(data->fstyle.uifont_id, data->wrap_width); - BLF_wordwrap(blf_mono_font, data->wrap_width); - - bbox.xmin += 0.5f * pad_px; /* add padding to the text */ - bbox.ymax -= 0.25f * pad_px; - - for (i = 0; i < data->totline; i++) { - bbox.ymin = bbox.ymax - (data->lineh * data->line_geom[i].lines); - if (data->format[i].style == UI_TIP_STYLE_HEADER) { - /* draw header and active data (is done here to be able to change color) */ - uiFontStyle fstyle_header = data->fstyle; - float xofs, yofs; - - /* override text-style */ - fstyle_header.shadow = 1; - fstyle_header.shadowcolor = rgb_to_grayscale(tip_colors[UI_TIP_LC_MAIN]); - fstyle_header.shadx = fstyle_header.shady = 0; - fstyle_header.shadowalpha = 1.0f; - fstyle_header.word_wrap = true; - - UI_fontstyle_set(&fstyle_header); - glColor3fv(tip_colors[UI_TIP_LC_MAIN]); - UI_fontstyle_draw(&fstyle_header, &bbox, data->header); - - /* offset to the end of the last line */ - xofs = data->line_geom[i].x_pos; - yofs = data->lineh * (data->line_geom[i].lines - 1); - bbox.xmin += xofs; - bbox.ymax -= yofs; - - glColor3fv(tip_colors[UI_TIP_LC_ACTIVE]); - fstyle_header.shadow = 0; - UI_fontstyle_draw(&fstyle_header, &bbox, data->active_info); - - /* undo offset */ - bbox.xmin -= xofs; - bbox.ymax += yofs; - } - else if (data->format[i].style == UI_TIP_STYLE_MONO) { - uiFontStyle fstyle_mono = data->fstyle; - fstyle_mono.uifont_id = blf_mono_font; - fstyle_mono.word_wrap = true; - - UI_fontstyle_set(&fstyle_mono); - /* XXX, needed because we dont have mono in 'U.uifonts' */ - BLF_size(fstyle_mono.uifont_id, fstyle_mono.points * U.pixelsize, U.dpi); - glColor3fv(tip_colors[data->format[i].color_id]); - UI_fontstyle_draw(&fstyle_mono, &bbox, data->lines[i]); - } - else { - uiFontStyle fstyle_normal = data->fstyle; - BLI_assert(data->format[i].style == UI_TIP_STYLE_NORMAL); - fstyle_normal.word_wrap = true; - - /* draw remaining data */ - UI_fontstyle_set(&fstyle_normal); - glColor3fv(tip_colors[data->format[i].color_id]); - UI_fontstyle_draw(&fstyle_normal, &bbox, data->lines[i]); - } - - bbox.ymax -= data->lineh * data->line_geom[i].lines; - - if ((i + 1 != data->totline) && data->format[i + 1].is_pad) { - bbox.ymax -= data->lineh * (UI_TIP_PAD_FAC - 1); - } - } - - BLF_disable(data->fstyle.uifont_id, BLF_WORD_WRAP); - BLF_disable(blf_mono_font, BLF_WORD_WRAP); - - if (multisample_enabled) - glEnable(GL_MULTISAMPLE); -} - -static void ui_tooltip_region_free_cb(ARegion *ar) -{ - uiTooltipData *data; - - data = ar->regiondata; - MEM_freeN(data); - ar->regiondata = NULL; -} - -static uiTooltipData *ui_tooltip_data_from_button(bContext *C, uiBut *but) -{ - uiStringInfo but_tip = {BUT_GET_TIP, NULL}; - uiStringInfo enum_label = {BUT_GET_RNAENUM_LABEL, NULL}; - uiStringInfo enum_tip = {BUT_GET_RNAENUM_TIP, NULL}; - uiStringInfo op_keymap = {BUT_GET_OP_KEYMAP, NULL}; - uiStringInfo prop_keymap = {BUT_GET_PROP_KEYMAP, NULL}; - uiStringInfo rna_struct = {BUT_GET_RNASTRUCT_IDENTIFIER, NULL}; - uiStringInfo rna_prop = {BUT_GET_RNAPROP_IDENTIFIER, NULL}; - - char buf[512]; - - /* create tooltip data */ - uiTooltipData *data = MEM_callocN(sizeof(uiTooltipData), "uiTooltipData"); - - UI_but_string_info_get(C, but, &but_tip, &enum_label, &enum_tip, &op_keymap, &prop_keymap, &rna_struct, &rna_prop, NULL); - - /* Tip */ - if (but_tip.strinfo) { - if (enum_label.strinfo) { - BLI_snprintf(data->header, sizeof(data->header), "%s: ", but_tip.strinfo); - BLI_strncpy(data->active_info, enum_label.strinfo, sizeof(data->lines[0])); - } - else { - BLI_snprintf(data->header, sizeof(data->header), "%s.", but_tip.strinfo); - } - data->format[data->totline].style = UI_TIP_STYLE_HEADER; - data->totline++; - - /* special case enum rna buttons */ - if ((but->type & UI_BTYPE_ROW) && but->rnaprop && RNA_property_flag(but->rnaprop) & PROP_ENUM_FLAG) { - BLI_strncpy(data->lines[data->totline], IFACE_("(Shift-Click/Drag to select multiple)"), - sizeof(data->lines[0])); - - data->format[data->totline].color_id = UI_TIP_LC_NORMAL; - data->totline++; - } - - } - /* Enum item label & tip */ - if (enum_tip.strinfo) { - BLI_strncpy(data->lines[data->totline], enum_tip.strinfo, sizeof(data->lines[0])); - data->format[data->totline].is_pad = true; - data->format[data->totline].color_id = UI_TIP_LC_VALUE; - data->totline++; - } - - /* Op shortcut */ - if (op_keymap.strinfo) { - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Shortcut: %s"), op_keymap.strinfo); - data->format[data->totline].is_pad = true; - data->format[data->totline].color_id = UI_TIP_LC_VALUE; - data->totline++; - } - - /* Property context-toggle shortcut */ - if (prop_keymap.strinfo) { - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Shortcut: %s"), prop_keymap.strinfo); - data->format[data->totline].is_pad = true; - data->format[data->totline].color_id = UI_TIP_LC_VALUE; - data->totline++; - } - - if (ELEM(but->type, UI_BTYPE_TEXT, UI_BTYPE_SEARCH_MENU)) { - /* better not show the value of a password */ - if ((but->rnaprop && (RNA_property_subtype(but->rnaprop) == PROP_PASSWORD)) == 0) { - /* full string */ - ui_but_string_get(but, buf, sizeof(buf)); - if (buf[0]) { - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Value: %s"), buf); - data->format[data->totline].is_pad = true; - data->format[data->totline].color_id = UI_TIP_LC_VALUE; - data->totline++; - } - } - } - - if (but->rnaprop) { - int unit_type = UI_but_unit_type_get(but); - - if (unit_type == PROP_UNIT_ROTATION) { - if (RNA_property_type(but->rnaprop) == PROP_FLOAT) { - float value = RNA_property_array_check(but->rnaprop) ? - RNA_property_float_get_index(&but->rnapoin, but->rnaprop, but->rnaindex) : - RNA_property_float_get(&but->rnapoin, but->rnaprop); - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Radians: %f"), value); - data->format[data->totline].color_id = UI_TIP_LC_NORMAL; - data->totline++; - } - } - - if (but->flag & UI_BUT_DRIVEN) { - if (ui_but_anim_expression_get(but, buf, sizeof(buf))) { - /* expression */ - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Expression: %s"), buf); - data->format[data->totline].color_id = UI_TIP_LC_NORMAL; - data->totline++; - } - } - - if (but->rnapoin.id.data) { - ID *id = but->rnapoin.id.data; - if (ID_IS_LINKED_DATABLOCK(id)) { - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Library: %s"), id->lib->name); - data->format[data->totline].color_id = UI_TIP_LC_NORMAL; - data->totline++; - } - } - } - else if (but->optype) { - PointerRNA *opptr; - char *str; - opptr = UI_but_operator_ptr_get(but); /* allocated when needed, the button owns it */ - - /* so the context is passed to itemf functions (some py itemf functions use it) */ - WM_operator_properties_sanitize(opptr, false); - - str = WM_operator_pystring_ex(C, NULL, false, false, but->optype, opptr); - - /* avoid overly verbose tips (eg, arrays of 20 layers), exact limit is arbitrary */ - WM_operator_pystring_abbreviate(str, 32); - - /* operator info */ - if ((U.flag & USER_TOOLTIPS_PYTHON) == 0) { - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Python: %s"), str); - data->format[data->totline].style = UI_TIP_STYLE_MONO; - data->format[data->totline].is_pad = true; - data->format[data->totline].color_id = UI_TIP_LC_PYTHON; - data->totline++; - } - - MEM_freeN(str); - } - - /* button is disabled, we may be able to tell user why */ - if (but->flag & UI_BUT_DISABLED) { - const char *disabled_msg = NULL; - - /* if operator poll check failed, it can give pretty precise info why */ - if (but->optype) { - CTX_wm_operator_poll_msg_set(C, NULL); - WM_operator_poll_context(C, but->optype, but->opcontext); - disabled_msg = CTX_wm_operator_poll_msg_get(C); - } - /* alternatively, buttons can store some reasoning too */ - else if (but->disabled_info) { - disabled_msg = TIP_(but->disabled_info); - } - - if (disabled_msg && disabled_msg[0]) { - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), TIP_("Disabled: %s"), disabled_msg); - data->format[data->totline].color_id = UI_TIP_LC_ALERT; - data->totline++; - } - } - - if ((U.flag & USER_TOOLTIPS_PYTHON) == 0 && !but->optype && rna_struct.strinfo) { - if (rna_prop.strinfo) { - /* Struct and prop */ - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), - TIP_("Python: %s.%s"), - rna_struct.strinfo, rna_prop.strinfo); - } - else { - /* Only struct (e.g. menus) */ - BLI_snprintf(data->lines[data->totline], sizeof(data->lines[0]), - TIP_("Python: %s"), rna_struct.strinfo); - } - data->format[data->totline].style = UI_TIP_STYLE_MONO; - data->format[data->totline].is_pad = true; - data->format[data->totline].color_id = UI_TIP_LC_PYTHON; - data->totline++; - - if (but->rnapoin.id.data) { - /* this could get its own 'BUT_GET_...' type */ - - /* never fails */ - char *id_path; - - if (but->rnaprop) { - id_path = RNA_path_full_property_py_ex(&but->rnapoin, but->rnaprop, but->rnaindex, true); - } - else { - id_path = RNA_path_full_struct_py(&but->rnapoin); - } - - BLI_strncat_utf8(data->lines[data->totline], id_path, sizeof(data->lines[0])); - MEM_freeN(id_path); - - data->format[data->totline].style = UI_TIP_STYLE_MONO; - data->format[data->totline].color_id = UI_TIP_LC_PYTHON; - data->totline++; - } - } - - /* Free strinfo's... */ - if (but_tip.strinfo) - MEM_freeN(but_tip.strinfo); - if (enum_label.strinfo) - MEM_freeN(enum_label.strinfo); - if (enum_tip.strinfo) - MEM_freeN(enum_tip.strinfo); - if (op_keymap.strinfo) - MEM_freeN(op_keymap.strinfo); - if (prop_keymap.strinfo) - MEM_freeN(prop_keymap.strinfo); - if (rna_struct.strinfo) - MEM_freeN(rna_struct.strinfo); - if (rna_prop.strinfo) - MEM_freeN(rna_prop.strinfo); - - BLI_assert(data->totline < MAX_TOOLTIP_LINES); - - if (data->totline == 0) { - MEM_freeN(data); - return NULL; - } - else { - return data; - } -} - -ARegion *ui_tooltip_create(bContext *C, ARegion *butregion, uiBut *but) -{ - const float pad_px = UI_TIP_PADDING; - wmWindow *win = CTX_wm_window(C); - const int winx = WM_window_pixels_x(win); - uiStyle *style = UI_style_get(); - static ARegionType type; - ARegion *ar; -/* IDProperty *prop;*/ - /* aspect values that shrink text are likely unreadable */ - const float aspect = min_ff(1.0f, but->block->aspect); - int fonth, fontw; - int ofsx, ofsy, h, i; - rctf rect_fl; - rcti rect_i; - int font_flag = 0; - - if (but->drawflag & UI_BUT_NO_TOOLTIP) { - return NULL; - } - - uiTooltipData *data = ui_tooltip_data_from_button(C, but); - if (data == NULL) { - return NULL; - } - - /* create area region */ - ar = ui_region_temp_add(CTX_wm_screen(C)); - - memset(&type, 0, sizeof(ARegionType)); - type.draw = ui_tooltip_region_draw_cb; - type.free = ui_tooltip_region_free_cb; - type.regionid = RGN_TYPE_TEMPORARY; - ar->type = &type; - - /* set font, get bb */ - data->fstyle = style->widget; /* copy struct */ - ui_fontscale(&data->fstyle.points, aspect); - - UI_fontstyle_set(&data->fstyle); - - data->wrap_width = min_ii(UI_TIP_MAXWIDTH * U.pixelsize / aspect, winx - (UI_TIP_PADDING * 2)); - - font_flag |= BLF_WORD_WRAP; - if (data->fstyle.kerning == 1) { - font_flag |= BLF_KERNING_DEFAULT; - } - BLF_enable(data->fstyle.uifont_id, font_flag); - BLF_enable(blf_mono_font, font_flag); - BLF_wordwrap(data->fstyle.uifont_id, data->wrap_width); - BLF_wordwrap(blf_mono_font, data->wrap_width); - - /* these defines tweaked depending on font */ -#define TIP_BORDER_X (16.0f / aspect) -#define TIP_BORDER_Y (6.0f / aspect) - - h = BLF_height_max(data->fstyle.uifont_id); - - for (i = 0, fontw = 0, fonth = 0; i < data->totline; i++) { - struct ResultBLF info; - int w, x_pos = 0; - - if (data->format[i].style == UI_TIP_STYLE_HEADER) { - w = BLF_width_ex(data->fstyle.uifont_id, data->header, sizeof(data->header), &info); - /* check for enum label */ - if (data->active_info[0]) { - x_pos = info.width; - w = max_ii(w, x_pos + BLF_width(data->fstyle.uifont_id, data->active_info, sizeof(data->active_info))); - } - } - else if (data->format[i].style == UI_TIP_STYLE_MONO) { - BLF_size(blf_mono_font, data->fstyle.points * U.pixelsize, U.dpi); - - w = BLF_width_ex(blf_mono_font, data->lines[i], sizeof(data->lines[i]), &info); - } - else { - BLI_assert(data->format[i].style == UI_TIP_STYLE_NORMAL); - - w = BLF_width_ex(data->fstyle.uifont_id, data->lines[i], sizeof(data->lines[i]), &info); - } - - fontw = max_ii(fontw, w); - - fonth += h * info.lines; - if ((i + 1 != data->totline) && data->format[i + 1].is_pad) { - fonth += h * (UI_TIP_PAD_FAC - 1); - } - - data->line_geom[i].lines = info.lines; - data->line_geom[i].x_pos = x_pos; - } - - //fontw *= aspect; - - BLF_disable(data->fstyle.uifont_id, font_flag); - BLF_disable(blf_mono_font, font_flag); - - ar->regiondata = data; - - data->toth = fonth; - data->lineh = h; - - /* compute position */ - ofsx = 0; //(but->block->panel) ? but->block->panel->ofsx : 0; - ofsy = 0; //(but->block->panel) ? but->block->panel->ofsy : 0; - - rect_fl.xmin = BLI_rctf_cent_x(&but->rect) + ofsx - TIP_BORDER_X; - rect_fl.xmax = rect_fl.xmin + fontw + pad_px; - rect_fl.ymax = but->rect.ymin + ofsy - TIP_BORDER_Y; - rect_fl.ymin = rect_fl.ymax - fonth - TIP_BORDER_Y; - - /* since the text has beens caled already, the size of tooltips is defined now */ - /* here we try to figure out the right location */ - if (butregion) { - float mx, my; - float ofsx_fl = rect_fl.xmin, ofsy_fl = rect_fl.ymax; - ui_block_to_window_fl(butregion, but->block, &ofsx_fl, &ofsy_fl); - -#if 1 - /* use X mouse location */ - mx = (win->eventstate->x + (TIP_BORDER_X * 2)) - BLI_rctf_cent_x(&but->rect); -#else - mx = ofsx_fl - rect_fl.xmin; -#endif - my = ofsy_fl - rect_fl.ymax; - - BLI_rctf_translate(&rect_fl, mx, my); - } - BLI_rcti_rctf_copy(&rect_i, &rect_fl); - -#undef TIP_BORDER_X -#undef TIP_BORDER_Y - - /* clip with window boundaries */ - if (rect_i.xmax > winx) { - /* super size */ - if (rect_i.xmax > winx + rect_i.xmin) { - rect_i.xmax = winx; - rect_i.xmin = 0; - } - else { - rect_i.xmin -= rect_i.xmax - winx; - rect_i.xmax = winx; - } - } - /* ensure at least 5 px above screen bounds - * 25 is just a guess to be above the menu item */ - if (rect_i.ymin < 5) { - rect_i.ymax += (-rect_i.ymin) + 30; - rect_i.ymin = 30; - } - - /* add padding */ - BLI_rcti_resize(&rect_i, - BLI_rcti_size_x(&rect_i) + pad_px, - BLI_rcti_size_y(&rect_i) + pad_px); - - /* widget rect, in region coords */ - { - const int margin = UI_POPUP_MARGIN; - - data->bbox.xmin = margin; - data->bbox.xmax = BLI_rcti_size_x(&rect_i) - margin; - data->bbox.ymin = margin; - data->bbox.ymax = BLI_rcti_size_y(&rect_i); - - /* region bigger for shadow */ - ar->winrct.xmin = rect_i.xmin - margin; - ar->winrct.xmax = rect_i.xmax + margin; - ar->winrct.ymin = rect_i.ymin - margin; - ar->winrct.ymax = rect_i.ymax + margin; - } - - /* adds subwindow */ - ED_region_init(C, ar); - - /* notify change and redraw */ - ED_region_tag_redraw(ar); - - return ar; -} - -void ui_tooltip_free(bContext *C, ARegion *ar) -{ - ui_region_temp_remove(C, CTX_wm_screen(C), ar); -} - - -/************************* Creating Search Box **********************/ - -struct uiSearchItems { - int maxitem, totitem, maxstrlen; - - int offset, offset_i; /* offset for inserting in array */ - int more; /* flag indicating there are more items */ - - char **names; - void **pointers; - int *icons; - - AutoComplete *autocpl; - void *active; -}; - -typedef struct uiSearchboxData { - rcti bbox; - uiFontStyle fstyle; - uiSearchItems items; - int active; /* index in items array */ - bool noback; /* when menu opened with enough space for this */ - bool preview; /* draw thumbnail previews, rather than list */ - bool use_sep; /* use the UI_SEP_CHAR char for splitting shortcuts (good for operators, bad for data) */ - int prv_rows, prv_cols; -} uiSearchboxData; - -#define SEARCH_ITEMS 10 - -/* exported for use by search callbacks */ -/* returns zero if nothing to add */ -bool UI_search_item_add(uiSearchItems *items, const char *name, void *poin, int iconid) -{ - /* hijack for autocomplete */ - if (items->autocpl) { - UI_autocomplete_update_name(items->autocpl, name); - return true; - } - - /* hijack for finding active item */ - if (items->active) { - if (poin == items->active) - items->offset_i = items->totitem; - items->totitem++; - return true; - } - - if (items->totitem >= items->maxitem) { - items->more = 1; - return false; - } - - /* skip first items in list */ - if (items->offset_i > 0) { - items->offset_i--; - return true; - } - - if (items->names) - BLI_strncpy(items->names[items->totitem], name, items->maxstrlen); - if (items->pointers) - items->pointers[items->totitem] = poin; - if (items->icons) - items->icons[items->totitem] = iconid; - - items->totitem++; - - return true; -} - -int UI_searchbox_size_y(void) -{ - return SEARCH_ITEMS * UI_UNIT_Y + 2 * UI_POPUP_MENU_TOP; -} - -int UI_searchbox_size_x(void) -{ - return 12 * UI_UNIT_X; -} - -int UI_search_items_find_index(uiSearchItems *items, const char *name) -{ - int i; - for (i = 0; i < items->totitem; i++) { - if (STREQ(name, items->names[i])) { - return i; - } - } - return -1; -} - -/* ar is the search box itself */ -static void ui_searchbox_select(bContext *C, ARegion *ar, uiBut *but, int step) -{ - uiSearchboxData *data = ar->regiondata; - - /* apply step */ - data->active += step; - - if (data->items.totitem == 0) { - data->active = -1; - } - else if (data->active >= data->items.totitem) { - if (data->items.more) { - data->items.offset++; - data->active = data->items.totitem - 1; - ui_searchbox_update(C, ar, but, false); - } - else { - data->active = data->items.totitem - 1; - } - } - else if (data->active < 0) { - if (data->items.offset) { - data->items.offset--; - data->active = 0; - ui_searchbox_update(C, ar, but, false); - } - else { - /* only let users step into an 'unset' state for unlink buttons */ - data->active = (but->flag & UI_BUT_VALUE_CLEAR) ? -1 : 0; - } - } - - ED_region_tag_redraw(ar); -} - -static void ui_searchbox_butrect(rcti *r_rect, uiSearchboxData *data, int itemnr) -{ - /* thumbnail preview */ - if (data->preview) { - int butw = (BLI_rcti_size_x(&data->bbox) - 2 * MENU_BORDER) / data->prv_cols; - int buth = (BLI_rcti_size_y(&data->bbox) - 2 * MENU_BORDER) / data->prv_rows; - int row, col; - - *r_rect = data->bbox; - - col = itemnr % data->prv_cols; - row = itemnr / data->prv_cols; - - r_rect->xmin += MENU_BORDER + (col * butw); - r_rect->xmax = r_rect->xmin + butw; - - r_rect->ymax -= MENU_BORDER + (row * buth); - r_rect->ymin = r_rect->ymax - buth; - } - /* list view */ - else { - int buth = (BLI_rcti_size_y(&data->bbox) - 2 * UI_POPUP_MENU_TOP) / SEARCH_ITEMS; - - *r_rect = data->bbox; - r_rect->xmin = data->bbox.xmin + 3.0f; - r_rect->xmax = data->bbox.xmax - 3.0f; - - r_rect->ymax = data->bbox.ymax - UI_POPUP_MENU_TOP - itemnr * buth; - r_rect->ymin = r_rect->ymax - buth; - } - -} - -int ui_searchbox_find_index(ARegion *ar, const char *name) -{ - uiSearchboxData *data = ar->regiondata; - return UI_search_items_find_index(&data->items, name); -} - -/* x and y in screencoords */ -bool ui_searchbox_inside(ARegion *ar, int x, int y) -{ - uiSearchboxData *data = ar->regiondata; - - return BLI_rcti_isect_pt(&data->bbox, x - ar->winrct.xmin, y - ar->winrct.ymin); -} - -/* string validated to be of correct length (but->hardmax) */ -bool ui_searchbox_apply(uiBut *but, ARegion *ar) -{ - uiSearchboxData *data = ar->regiondata; - - but->func_arg2 = NULL; - - if (data->active != -1) { - const char *name = data->items.names[data->active]; - const char *name_sep = data->use_sep ? strrchr(name, UI_SEP_CHAR) : NULL; - - BLI_strncpy(but->editstr, name, name_sep ? (name_sep - name) : data->items.maxstrlen); - - but->func_arg2 = data->items.pointers[data->active]; - - return true; - } - else if (but->flag & UI_BUT_VALUE_CLEAR) { - /* It is valid for _VALUE_CLEAR flavor to have no active element (it's a valid way to unlink). */ - but->editstr[0] = '\0'; - - return true; - } - else { - return false; - } -} - -void ui_searchbox_event(bContext *C, ARegion *ar, uiBut *but, const wmEvent *event) -{ - uiSearchboxData *data = ar->regiondata; - int type = event->type, val = event->val; - - if (type == MOUSEPAN) - ui_pan_to_scroll(event, &type, &val); - - switch (type) { - case WHEELUPMOUSE: - case UPARROWKEY: - ui_searchbox_select(C, ar, but, -1); - break; - case WHEELDOWNMOUSE: - case DOWNARROWKEY: - ui_searchbox_select(C, ar, but, 1); - break; - case MOUSEMOVE: - if (BLI_rcti_isect_pt(&ar->winrct, event->x, event->y)) { - rcti rect; - int a; - - for (a = 0; a < data->items.totitem; a++) { - ui_searchbox_butrect(&rect, data, a); - if (BLI_rcti_isect_pt(&rect, event->x - ar->winrct.xmin, event->y - ar->winrct.ymin)) { - if (data->active != a) { - data->active = a; - ui_searchbox_select(C, ar, but, 0); - break; - } - } - } - } - break; - } -} - -/* ar is the search box itself */ -void ui_searchbox_update(bContext *C, ARegion *ar, uiBut *but, const bool reset) -{ - uiSearchboxData *data = ar->regiondata; - - /* reset vars */ - data->items.totitem = 0; - data->items.more = 0; - if (reset == false) { - data->items.offset_i = data->items.offset; - } - else { - data->items.offset_i = data->items.offset = 0; - data->active = -1; - - /* handle active */ - if (but->search_func && but->func_arg2) { - data->items.active = but->func_arg2; - but->search_func(C, but->search_arg, but->editstr, &data->items); - data->items.active = NULL; - - /* found active item, calculate real offset by centering it */ - if (data->items.totitem) { - /* first case, begin of list */ - if (data->items.offset_i < data->items.maxitem) { - data->active = data->items.offset_i; - data->items.offset_i = 0; - } - else { - /* second case, end of list */ - if (data->items.totitem - data->items.offset_i <= data->items.maxitem) { - data->active = data->items.offset_i - data->items.totitem + data->items.maxitem; - data->items.offset_i = data->items.totitem - data->items.maxitem; - } - else { - /* center active item */ - data->items.offset_i -= data->items.maxitem / 2; - data->active = data->items.maxitem / 2; - } - } - } - data->items.offset = data->items.offset_i; - data->items.totitem = 0; - } - } - - /* callback */ - if (but->search_func) - but->search_func(C, but->search_arg, but->editstr, &data->items); - - /* handle case where editstr is equal to one of items */ - if (reset && data->active == -1) { - int a; - - for (a = 0; a < data->items.totitem; a++) { - const char *name = data->items.names[a]; - const char *name_sep = data->use_sep ? strrchr(name, UI_SEP_CHAR) : NULL; - if (STREQLEN(but->editstr, name, name_sep ? (name_sep - name) : data->items.maxstrlen)) { - data->active = a; - break; - } - } - if (data->items.totitem == 1 && but->editstr[0]) - data->active = 0; - } - - /* validate selected item */ - ui_searchbox_select(C, ar, but, 0); - - ED_region_tag_redraw(ar); -} - -int ui_searchbox_autocomplete(bContext *C, ARegion *ar, uiBut *but, char *str) -{ - uiSearchboxData *data = ar->regiondata; - int match = AUTOCOMPLETE_NO_MATCH; - - if (str[0]) { - data->items.autocpl = UI_autocomplete_begin(str, ui_but_string_get_max_length(but)); - - but->search_func(C, but->search_arg, but->editstr, &data->items); - - match = UI_autocomplete_end(data->items.autocpl, str); - data->items.autocpl = NULL; - } - - return match; -} - -static void ui_searchbox_region_draw_cb(const bContext *UNUSED(C), ARegion *ar) -{ - uiSearchboxData *data = ar->regiondata; - - /* pixel space */ - wmOrtho2_region_pixelspace(ar); - - if (data->noback == false) - ui_draw_search_back(NULL, NULL, &data->bbox); /* style not used yet */ - - /* draw text */ - if (data->items.totitem) { - rcti rect; - int a; - - if (data->preview) { - /* draw items */ - for (a = 0; a < data->items.totitem; a++) { - ui_searchbox_butrect(&rect, data, a); - - /* widget itself */ - ui_draw_preview_item(&data->fstyle, &rect, data->items.names[a], data->items.icons[a], - (a == data->active) ? UI_ACTIVE : 0); - } - - /* indicate more */ - if (data->items.more) { - ui_searchbox_butrect(&rect, data, data->items.maxitem - 1); - glEnable(GL_BLEND); - UI_icon_draw(rect.xmax - 18, rect.ymin - 7, ICON_TRIA_DOWN); - glDisable(GL_BLEND); - } - if (data->items.offset) { - ui_searchbox_butrect(&rect, data, 0); - glEnable(GL_BLEND); - UI_icon_draw(rect.xmin, rect.ymax - 9, ICON_TRIA_UP); - glDisable(GL_BLEND); - } - - } - else { - /* draw items */ - for (a = 0; a < data->items.totitem; a++) { - ui_searchbox_butrect(&rect, data, a); - - /* widget itself */ - ui_draw_menu_item(&data->fstyle, &rect, data->items.names[a], data->items.icons[a], - (a == data->active) ? UI_ACTIVE : 0, data->use_sep); - - } - /* indicate more */ - if (data->items.more) { - ui_searchbox_butrect(&rect, data, data->items.maxitem - 1); - glEnable(GL_BLEND); - UI_icon_draw((BLI_rcti_size_x(&rect)) / 2, rect.ymin - 9, ICON_TRIA_DOWN); - glDisable(GL_BLEND); - } - if (data->items.offset) { - ui_searchbox_butrect(&rect, data, 0); - glEnable(GL_BLEND); - UI_icon_draw((BLI_rcti_size_x(&rect)) / 2, rect.ymax - 7, ICON_TRIA_UP); - glDisable(GL_BLEND); - } - } - } -} - -static void ui_searchbox_region_free_cb(ARegion *ar) -{ - uiSearchboxData *data = ar->regiondata; - int a; - - /* free search data */ - for (a = 0; a < data->items.maxitem; a++) { - MEM_freeN(data->items.names[a]); - } - MEM_freeN(data->items.names); - MEM_freeN(data->items.pointers); - MEM_freeN(data->items.icons); - - MEM_freeN(data); - ar->regiondata = NULL; -} - -ARegion *ui_searchbox_create_generic(bContext *C, ARegion *butregion, uiBut *but) -{ - wmWindow *win = CTX_wm_window(C); - uiStyle *style = UI_style_get(); - static ARegionType type; - ARegion *ar; - uiSearchboxData *data; - float aspect = but->block->aspect; - rctf rect_fl; - rcti rect_i; - const int margin = UI_POPUP_MARGIN; - int winx /*, winy */, ofsx, ofsy; - int i; - - /* create area region */ - ar = ui_region_temp_add(CTX_wm_screen(C)); - - memset(&type, 0, sizeof(ARegionType)); - type.draw = ui_searchbox_region_draw_cb; - type.free = ui_searchbox_region_free_cb; - type.regionid = RGN_TYPE_TEMPORARY; - ar->type = &type; - - /* create searchbox data */ - data = MEM_callocN(sizeof(uiSearchboxData), "uiSearchboxData"); - - /* set font, get bb */ - data->fstyle = style->widget; /* copy struct */ - data->fstyle.align = UI_STYLE_TEXT_CENTER; - ui_fontscale(&data->fstyle.points, aspect); - UI_fontstyle_set(&data->fstyle); - - ar->regiondata = data; - - /* special case, hardcoded feature, not draw backdrop when called from menus, - * assume for design that popup already added it */ - if (but->block->flag & UI_BLOCK_SEARCH_MENU) - data->noback = true; - - if (but->a1 > 0 && but->a2 > 0) { - data->preview = true; - data->prv_rows = but->a1; - data->prv_cols = but->a2; - } - - /* only show key shortcuts when needed (not rna buttons) [#36699] */ - if (but->rnaprop == NULL) { - data->use_sep = true; - } - - /* compute position */ - if (but->block->flag & UI_BLOCK_SEARCH_MENU) { - const int search_but_h = BLI_rctf_size_y(&but->rect) + 10; - /* this case is search menu inside other menu */ - /* we copy region size */ - - ar->winrct = butregion->winrct; - - /* widget rect, in region coords */ - data->bbox.xmin = margin; - data->bbox.xmax = BLI_rcti_size_x(&ar->winrct) - margin; - data->bbox.ymin = margin; - data->bbox.ymax = BLI_rcti_size_y(&ar->winrct) - margin; - - /* check if button is lower half */ - if (but->rect.ymax < BLI_rctf_cent_y(&but->block->rect)) { - data->bbox.ymin += search_but_h; - } - else { - data->bbox.ymax -= search_but_h; - } - } - else { - const int searchbox_width = UI_searchbox_size_x(); - - rect_fl.xmin = but->rect.xmin - 5; /* align text with button */ - rect_fl.xmax = but->rect.xmax + 5; /* symmetrical */ - rect_fl.ymax = but->rect.ymin; - rect_fl.ymin = rect_fl.ymax - UI_searchbox_size_y(); - - ofsx = (but->block->panel) ? but->block->panel->ofsx : 0; - ofsy = (but->block->panel) ? but->block->panel->ofsy : 0; - - BLI_rctf_translate(&rect_fl, ofsx, ofsy); - - /* minimal width */ - if (BLI_rctf_size_x(&rect_fl) < searchbox_width) { - rect_fl.xmax = rect_fl.xmin + searchbox_width; - } - - /* copy to int, gets projected if possible too */ - BLI_rcti_rctf_copy(&rect_i, &rect_fl); - - if (butregion->v2d.cur.xmin != butregion->v2d.cur.xmax) { - UI_view2d_view_to_region_rcti(&butregion->v2d, &rect_fl, &rect_i); - } - - BLI_rcti_translate(&rect_i, butregion->winrct.xmin, butregion->winrct.ymin); - - winx = WM_window_pixels_x(win); - // winy = WM_window_pixels_y(win); /* UNUSED */ - //wm_window_get_size(win, &winx, &winy); - - if (rect_i.xmax > winx) { - /* super size */ - if (rect_i.xmax > winx + rect_i.xmin) { - rect_i.xmax = winx; - rect_i.xmin = 0; - } - else { - rect_i.xmin -= rect_i.xmax - winx; - rect_i.xmax = winx; - } - } - - if (rect_i.ymin < 0) { - int newy1 = but->rect.ymax + ofsy; - - if (butregion->v2d.cur.xmin != butregion->v2d.cur.xmax) - newy1 = UI_view2d_view_to_region_y(&butregion->v2d, newy1); - - newy1 += butregion->winrct.ymin; - - rect_i.ymax = BLI_rcti_size_y(&rect_i) + newy1; - rect_i.ymin = newy1; - } - - /* widget rect, in region coords */ - data->bbox.xmin = margin; - data->bbox.xmax = BLI_rcti_size_x(&rect_i) + margin; - data->bbox.ymin = margin; - data->bbox.ymax = BLI_rcti_size_y(&rect_i) + margin; - - /* region bigger for shadow */ - ar->winrct.xmin = rect_i.xmin - margin; - ar->winrct.xmax = rect_i.xmax + margin; - ar->winrct.ymin = rect_i.ymin - margin; - ar->winrct.ymax = rect_i.ymax; - } - - /* adds subwindow */ - ED_region_init(C, ar); - - /* notify change and redraw */ - ED_region_tag_redraw(ar); - - /* prepare search data */ - if (data->preview) { - data->items.maxitem = data->prv_rows * data->prv_cols; - } - else { - data->items.maxitem = SEARCH_ITEMS; - } - data->items.maxstrlen = but->hardmax; - data->items.totitem = 0; - data->items.names = MEM_callocN(data->items.maxitem * sizeof(void *), "search names"); - data->items.pointers = MEM_callocN(data->items.maxitem * sizeof(void *), "search pointers"); - data->items.icons = MEM_callocN(data->items.maxitem * sizeof(int), "search icons"); - for (i = 0; i < data->items.maxitem; i++) - data->items.names[i] = MEM_callocN(but->hardmax + 1, "search pointers"); - - return ar; -} - -/** - * Similar to Python's `str.title` except... - * - * - we know words are upper case and ascii only. - * - '_' are replaces by spaces. - */ -static void str_tolower_titlecaps_ascii(char *str, const size_t len) -{ - size_t i; - bool prev_delim = true; - - for (i = 0; (i < len) && str[i]; i++) { - if (str[i] >= 'A' && str[i] <= 'Z') { - if (prev_delim == false) { - str[i] += 'a' - 'A'; - } - } - else if (str[i] == '_') { - str[i] = ' '; - } - - prev_delim = ELEM(str[i], ' ') || (str[i] >= '0' && str[i] <= '9'); - } - -} - -static void ui_searchbox_region_draw_cb__operator(const bContext *UNUSED(C), ARegion *ar) -{ - uiSearchboxData *data = ar->regiondata; - - /* pixel space */ - wmOrtho2_region_pixelspace(ar); - - if (data->noback == false) - ui_draw_search_back(NULL, NULL, &data->bbox); /* style not used yet */ - - /* draw text */ - if (data->items.totitem) { - rcti rect; - int a; - - /* draw items */ - for (a = 0; a < data->items.totitem; a++) { - rcti rect_pre, rect_post; - ui_searchbox_butrect(&rect, data, a); - - rect_pre = rect; - rect_post = rect; - - rect_pre.xmax = rect_post.xmin = rect.xmin + ((rect.xmax - rect.xmin) / 4); - - /* widget itself */ - /* NOTE: i18n messages extracting tool does the same, please keep it in sync. */ - { - wmOperatorType *ot = data->items.pointers[a]; - - int state = (a == data->active) ? UI_ACTIVE : 0; - char text_pre[128]; - char *text_pre_p = strstr(ot->idname, "_OT_"); - if (text_pre_p == NULL) { - text_pre[0] = '\0'; - } - else { - int text_pre_len; - text_pre_p += 1; - text_pre_len = BLI_strncpy_rlen( - text_pre, ot->idname, min_ii(sizeof(text_pre), text_pre_p - ot->idname)); - text_pre[text_pre_len] = ':'; - text_pre[text_pre_len + 1] = '\0'; - str_tolower_titlecaps_ascii(text_pre, sizeof(text_pre)); - } - - rect_pre.xmax += 4; /* sneaky, avoid showing ugly margin */ - ui_draw_menu_item(&data->fstyle, &rect_pre, CTX_IFACE_(BLT_I18NCONTEXT_OPERATOR_DEFAULT, text_pre), - data->items.icons[a], state, false); - ui_draw_menu_item(&data->fstyle, &rect_post, data->items.names[a], 0, state, data->use_sep); - } - - } - /* indicate more */ - if (data->items.more) { - ui_searchbox_butrect(&rect, data, data->items.maxitem - 1); - glEnable(GL_BLEND); - UI_icon_draw((BLI_rcti_size_x(&rect)) / 2, rect.ymin - 9, ICON_TRIA_DOWN); - glDisable(GL_BLEND); - } - if (data->items.offset) { - ui_searchbox_butrect(&rect, data, 0); - glEnable(GL_BLEND); - UI_icon_draw((BLI_rcti_size_x(&rect)) / 2, rect.ymax - 7, ICON_TRIA_UP); - glDisable(GL_BLEND); - } - } -} - -ARegion *ui_searchbox_create_operator(bContext *C, ARegion *butregion, uiBut *but) -{ - ARegion *ar; - - ar = ui_searchbox_create_generic(C, butregion, but); - - ar->type->draw = ui_searchbox_region_draw_cb__operator; - - return ar; -} - -void ui_searchbox_free(bContext *C, ARegion *ar) -{ - ui_region_temp_remove(C, CTX_wm_screen(C), ar); -} - -/* sets red alert if button holds a string it can't find */ -/* XXX weak: search_func adds all partial matches... */ -void ui_but_search_refresh(uiBut *but) -{ - uiSearchItems *items; - int x1; - - /* possibly very large lists (such as ID datablocks) only - * only validate string RNA buts (not pointers) */ - if (but->rnaprop && RNA_property_type(but->rnaprop) != PROP_STRING) { - return; - } - - items = MEM_callocN(sizeof(uiSearchItems), "search items"); - - /* setup search struct */ - items->maxitem = 10; - items->maxstrlen = 256; - items->names = MEM_callocN(items->maxitem * sizeof(void *), "search names"); - for (x1 = 0; x1 < items->maxitem; x1++) - items->names[x1] = MEM_callocN(but->hardmax + 1, "search names"); - - but->search_func(but->block->evil_C, but->search_arg, but->drawstr, items); - - /* only redalert when we are sure of it, this can miss cases when >10 matches */ - if (items->totitem == 0) { - UI_but_flag_enable(but, UI_BUT_REDALERT); - } - else if (items->more == 0) { - if (UI_search_items_find_index(items, but->drawstr) == -1) { - UI_but_flag_enable(but, UI_BUT_REDALERT); - } - } - - for (x1 = 0; x1 < items->maxitem; x1++) { - MEM_freeN(items->names[x1]); - } - MEM_freeN(items->names); - MEM_freeN(items); -} - - -/************************* Creating Menu Blocks **********************/ - -/* position block relative to but, result is in window space */ -static void ui_block_position(wmWindow *window, ARegion *butregion, uiBut *but, uiBlock *block) -{ - uiBut *bt; - uiSafetyRct *saferct; - rctf butrct; - /*float aspect;*/ /*UNUSED*/ - int xsize, ysize, xof = 0, yof = 0, center; - short dir1 = 0, dir2 = 0; - - /* transform to window coordinates, using the source button region/block */ - ui_block_to_window_rctf(butregion, but->block, &butrct, &but->rect); - - /* widget_roundbox_set has this correction too, keep in sync */ - if (but->type != UI_BTYPE_PULLDOWN) { - if (but->drawflag & UI_BUT_ALIGN_TOP) - butrct.ymax += U.pixelsize; - if (but->drawflag & UI_BUT_ALIGN_LEFT) - butrct.xmin -= U.pixelsize; - } - - /* calc block rect */ - if (block->rect.xmin == 0.0f && block->rect.xmax == 0.0f) { - if (block->buttons.first) { - BLI_rctf_init_minmax(&block->rect); - - for (bt = block->buttons.first; bt; bt = bt->next) { - BLI_rctf_union(&block->rect, &bt->rect); - } - } - else { - /* we're nice and allow empty blocks too */ - block->rect.xmin = block->rect.ymin = 0; - block->rect.xmax = block->rect.ymax = 20; - } - } - - /* aspect = (float)(BLI_rcti_size_x(&block->rect) + 4);*/ /*UNUSED*/ - ui_block_to_window_rctf(butregion, but->block, &block->rect, &block->rect); - - //block->rect.xmin -= 2.0; block->rect.ymin -= 2.0; - //block->rect.xmax += 2.0; block->rect.ymax += 2.0; - - xsize = BLI_rctf_size_x(&block->rect) + 0.2f * UI_UNIT_X; /* 4 for shadow */ - ysize = BLI_rctf_size_y(&block->rect) + 0.2f * UI_UNIT_Y; - /* aspect /= (float)xsize;*/ /*UNUSED*/ - - { - bool left = 0, right = 0, top = 0, down = 0; - int winx, winy; - // int offscreen; - - winx = WM_window_pixels_x(window); - winy = WM_window_pixels_y(window); - // wm_window_get_size(window, &winx, &winy); - - if (block->direction & UI_DIR_CENTER_Y) { - center = ysize / 2; - } - else { - center = 0; - } - - /* check if there's space at all */ - if (butrct.xmin - xsize > 0.0f) left = 1; - if (butrct.xmax + xsize < winx) right = 1; - if (butrct.ymin - ysize + center > 0.0f) down = 1; - if (butrct.ymax + ysize - center < winy) top = 1; - - if (top == 0 && down == 0) { - if (butrct.ymin - ysize < winy - butrct.ymax - ysize) - top = 1; - else - down = 1; - } - - dir1 = (block->direction & UI_DIR_ALL); - - /* secundary directions */ - if (dir1 & (UI_DIR_UP | UI_DIR_DOWN)) { - if (dir1 & UI_DIR_LEFT) dir2 = UI_DIR_LEFT; - else if (dir1 & UI_DIR_RIGHT) dir2 = UI_DIR_RIGHT; - dir1 &= (UI_DIR_UP | UI_DIR_DOWN); - } - - if ((dir2 == 0) && (dir1 == UI_DIR_LEFT || dir1 == UI_DIR_RIGHT)) dir2 = UI_DIR_DOWN; - if ((dir2 == 0) && (dir1 == UI_DIR_UP || dir1 == UI_DIR_DOWN)) dir2 = UI_DIR_LEFT; - - /* no space at all? don't change */ - if (left || right) { - if (dir1 == UI_DIR_LEFT && left == 0) dir1 = UI_DIR_RIGHT; - if (dir1 == UI_DIR_RIGHT && right == 0) dir1 = UI_DIR_LEFT; - /* this is aligning, not append! */ - if (dir2 == UI_DIR_LEFT && right == 0) dir2 = UI_DIR_RIGHT; - if (dir2 == UI_DIR_RIGHT && left == 0) dir2 = UI_DIR_LEFT; - } - if (down || top) { - if (dir1 == UI_DIR_UP && top == 0) dir1 = UI_DIR_DOWN; - if (dir1 == UI_DIR_DOWN && down == 0) dir1 = UI_DIR_UP; - BLI_assert(dir2 != UI_DIR_UP); -// if (dir2 == UI_DIR_UP && top == 0) dir2 = UI_DIR_DOWN; - if (dir2 == UI_DIR_DOWN && down == 0) dir2 = UI_DIR_UP; - } - - if (dir1 == UI_DIR_LEFT) { - xof = butrct.xmin - block->rect.xmax; - if (dir2 == UI_DIR_UP) yof = butrct.ymin - block->rect.ymin - center - MENU_PADDING; - else yof = butrct.ymax - block->rect.ymax + center + MENU_PADDING; - } - else if (dir1 == UI_DIR_RIGHT) { - xof = butrct.xmax - block->rect.xmin; - if (dir2 == UI_DIR_UP) yof = butrct.ymin - block->rect.ymin - center - MENU_PADDING; - else yof = butrct.ymax - block->rect.ymax + center + MENU_PADDING; - } - else if (dir1 == UI_DIR_UP) { - yof = butrct.ymax - block->rect.ymin; - if (dir2 == UI_DIR_RIGHT) xof = butrct.xmax - block->rect.xmax; - else xof = butrct.xmin - block->rect.xmin; - /* changed direction? */ - if ((dir1 & block->direction) == 0) { - UI_block_order_flip(block); - } - } - else if (dir1 == UI_DIR_DOWN) { - yof = butrct.ymin - block->rect.ymax; - if (dir2 == UI_DIR_RIGHT) xof = butrct.xmax - block->rect.xmax; - else xof = butrct.xmin - block->rect.xmin; - /* changed direction? */ - if ((dir1 & block->direction) == 0) { - UI_block_order_flip(block); - } - } - - /* and now we handle the exception; no space below or to top */ - if (top == 0 && down == 0) { - if (dir1 == UI_DIR_LEFT || dir1 == UI_DIR_RIGHT) { - /* align with bottom of screen */ - // yof = ysize; (not with menu scrolls) - } - } - -#if 0 /* seems redundant and causes issues with blocks inside big regions */ - /* or no space left or right */ - if (left == 0 && right == 0) { - if (dir1 == UI_DIR_UP || dir1 == UI_DIR_DOWN) { - /* align with left size of screen */ - xof = -block->rect.xmin + 5; - } - } -#endif - -#if 0 - /* clamp to window bounds, could be made into an option if its ever annoying */ - if ( (offscreen = (block->rect.ymin + yof)) < 0) yof -= offscreen; /* bottom */ - else if ((offscreen = (block->rect.ymax + yof) - winy) > 0) yof -= offscreen; /* top */ - if ( (offscreen = (block->rect.xmin + xof)) < 0) xof -= offscreen; /* left */ - else if ((offscreen = (block->rect.xmax + xof) - winx) > 0) xof -= offscreen; /* right */ -#endif - } - - /* apply offset, buttons in window coords */ - - for (bt = block->buttons.first; bt; bt = bt->next) { - ui_block_to_window_rctf(butregion, but->block, &bt->rect, &bt->rect); - - BLI_rctf_translate(&bt->rect, xof, yof); - - /* ui_but_update recalculates drawstring size in pixels */ - ui_but_update(bt); - } - - BLI_rctf_translate(&block->rect, xof, yof); - - /* safety calculus */ - { - const float midx = BLI_rctf_cent_x(&butrct); - const float midy = BLI_rctf_cent_y(&butrct); - - /* when you are outside parent button, safety there should be smaller */ - - /* parent button to left */ - if (midx < block->rect.xmin) block->safety.xmin = block->rect.xmin - 3; - else block->safety.xmin = block->rect.xmin - 40; - /* parent button to right */ - if (midx > block->rect.xmax) block->safety.xmax = block->rect.xmax + 3; - else block->safety.xmax = block->rect.xmax + 40; - - /* parent button on bottom */ - if (midy < block->rect.ymin) block->safety.ymin = block->rect.ymin - 3; - else block->safety.ymin = block->rect.ymin - 40; - /* parent button on top */ - if (midy > block->rect.ymax) block->safety.ymax = block->rect.ymax + 3; - else block->safety.ymax = block->rect.ymax + 40; - - /* exception for switched pulldowns... */ - if (dir1 && (dir1 & block->direction) == 0) { - if (dir2 == UI_DIR_RIGHT) block->safety.xmax = block->rect.xmax + 3; - if (dir2 == UI_DIR_LEFT) block->safety.xmin = block->rect.xmin - 3; - } - block->direction = dir1; - } - - /* keep a list of these, needed for pulldown menus */ - saferct = MEM_callocN(sizeof(uiSafetyRct), "uiSafetyRct"); - saferct->parent = butrct; - saferct->safety = block->safety; - BLI_freelistN(&block->saferct); - BLI_duplicatelist(&block->saferct, &but->block->saferct); - BLI_addhead(&block->saferct, saferct); -} - -static void ui_block_region_draw(const bContext *C, ARegion *ar) -{ - uiBlock *block; - - if (ar->do_draw & RGN_DRAW_REFRESH_UI) { - uiBlock *block_next; - ar->do_draw &= ~RGN_DRAW_REFRESH_UI; - for (block = ar->uiblocks.first; block; block = block_next) { - block_next = block->next; - if (block->handle->can_refresh) { - ui_popup_block_refresh((bContext *)C, block->handle, NULL, NULL); - } - } - } - - for (block = ar->uiblocks.first; block; block = block->next) - UI_block_draw(C, block); -} - -/** - * Use to refresh centered popups on screen resizing (for splash). - */ -static void ui_block_region_popup_window_listener( - bScreen *UNUSED(sc), ScrArea *UNUSED(sa), ARegion *ar, wmNotifier *wmn) -{ - switch (wmn->category) { - case NC_WINDOW: - { - switch (wmn->action) { - case NA_EDITED: - { - /* window resize */ - ED_region_tag_refresh_ui(ar); - break; - } - } - break; - } - } -} - -static void ui_popup_block_clip(wmWindow *window, uiBlock *block) -{ - uiBut *bt; - float xofs = 0.0f; - int width = UI_SCREEN_MARGIN; - int winx, winy; - - if (block->flag & UI_BLOCK_NO_WIN_CLIP) { - return; - } - - winx = WM_window_pixels_x(window); - winy = WM_window_pixels_y(window); - - /* shift menus to right if outside of view */ - if (block->rect.xmin < width) { - xofs = (width - block->rect.xmin); - block->rect.xmin += xofs; - block->rect.xmax += xofs; - } - /* or shift to left if outside of view */ - if (block->rect.xmax > winx - width) { - xofs = winx - width - block->rect.xmax; - block->rect.xmin += xofs; - block->rect.xmax += xofs; - } - - if (block->rect.ymin < width) - block->rect.ymin = width; - if (block->rect.ymax > winy - UI_POPUP_MENU_TOP) - block->rect.ymax = winy - UI_POPUP_MENU_TOP; - - /* ensure menu items draw inside left/right boundary */ - for (bt = block->buttons.first; bt; bt = bt->next) { - bt->rect.xmin += xofs; - bt->rect.xmax += xofs; - } - -} - -void ui_popup_block_scrolltest(uiBlock *block) -{ - uiBut *bt; - - block->flag &= ~(UI_BLOCK_CLIPBOTTOM | UI_BLOCK_CLIPTOP); - - for (bt = block->buttons.first; bt; bt = bt->next) - bt->flag &= ~UI_SCROLLED; - - if (block->buttons.first == block->buttons.last) - return; - - /* mark buttons that are outside boundary */ - for (bt = block->buttons.first; bt; bt = bt->next) { - if (bt->rect.ymin < block->rect.ymin) { - bt->flag |= UI_SCROLLED; - block->flag |= UI_BLOCK_CLIPBOTTOM; - } - if (bt->rect.ymax > block->rect.ymax) { - bt->flag |= UI_SCROLLED; - block->flag |= UI_BLOCK_CLIPTOP; - } - } - - /* mark buttons overlapping arrows, if we have them */ - for (bt = block->buttons.first; bt; bt = bt->next) { - if (block->flag & UI_BLOCK_CLIPBOTTOM) { - if (bt->rect.ymin < block->rect.ymin + UI_MENU_SCROLL_ARROW) - bt->flag |= UI_SCROLLED; - } - if (block->flag & UI_BLOCK_CLIPTOP) { - if (bt->rect.ymax > block->rect.ymax - UI_MENU_SCROLL_ARROW) - bt->flag |= UI_SCROLLED; - } - } -} - -static void ui_popup_block_remove(bContext *C, uiPopupBlockHandle *handle) -{ - wmWindow *win = CTX_wm_window(C); - bScreen *sc = CTX_wm_screen(C); - - ui_region_temp_remove(C, sc, handle->region); - - /* reset to region cursor (only if there's not another menu open) */ - if (BLI_listbase_is_empty(&sc->regionbase)) { - ED_region_cursor_set(win, CTX_wm_area(C), CTX_wm_region(C)); - /* in case cursor needs to be changed again */ - WM_event_add_mousemove(C); - } - - if (handle->scrolltimer) - WM_event_remove_timer(CTX_wm_manager(C), win, handle->scrolltimer); -} - -/** - * Called for creating new popups and refreshing existing ones. - */ -uiBlock *ui_popup_block_refresh( - bContext *C, uiPopupBlockHandle *handle, - ARegion *butregion, uiBut *but) -{ - BLI_assert(handle->can_refresh == true); - - const int margin = UI_POPUP_MARGIN; - wmWindow *window = CTX_wm_window(C); - ARegion *ar = handle->region; - - uiBlockCreateFunc create_func = handle->popup_create_vars.create_func; - uiBlockHandleCreateFunc handle_create_func = handle->popup_create_vars.handle_create_func; - void *arg = handle->popup_create_vars.arg; - - uiBlock *block_old = ar->uiblocks.first; - uiBlock *block; - -#ifdef DEBUG - wmEvent *event_back = window->eventstate; -#endif - - /* create ui block */ - if (create_func) - block = create_func(C, ar, arg); - else - block = handle_create_func(C, handle, arg); - - /* callbacks _must_ leave this for us, otherwise we can't call UI_block_update_from_old */ - BLI_assert(!block->endblock); - - /* ensure we don't use mouse coords here! */ -#ifdef DEBUG - window->eventstate = NULL; -#endif - - if (block->handle) { - memcpy(block->handle, handle, sizeof(uiPopupBlockHandle)); - MEM_freeN(handle); - handle = block->handle; - } - else - block->handle = handle; - - ar->regiondata = handle; - - /* set UI_BLOCK_NUMSELECT before UI_block_end() so we get alphanumeric keys assigned */ - if (but == NULL) { - block->flag |= UI_BLOCK_POPUP; - } - - block->flag |= UI_BLOCK_LOOP; - - /* defer this until blocks are translated (below) */ - block->oldblock = NULL; - - if (!block->endblock) { - UI_block_end_ex(C, block, handle->popup_create_vars.event_xy, handle->popup_create_vars.event_xy); - } - - /* if this is being created from a button */ - if (but) { - block->aspect = but->block->aspect; - ui_block_position(window, butregion, but, block); - handle->direction = block->direction; - } - else { - uiSafetyRct *saferct; - /* keep a list of these, needed for pulldown menus */ - saferct = MEM_callocN(sizeof(uiSafetyRct), "uiSafetyRct"); - saferct->safety = block->safety; - BLI_addhead(&block->saferct, saferct); - } - - if (block->flag & UI_BLOCK_RADIAL) { - int win_width = UI_SCREEN_MARGIN; - int winx, winy; - - int x_offset = 0, y_offset = 0; - - winx = WM_window_pixels_x(window); - winy = WM_window_pixels_y(window); - - copy_v2_v2(block->pie_data.pie_center_init, block->pie_data.pie_center_spawned); - - /* only try translation if area is large enough */ - if (BLI_rctf_size_x(&block->rect) < winx - (2.0f * win_width)) { - if (block->rect.xmin < win_width ) x_offset += win_width - block->rect.xmin; - if (block->rect.xmax > winx - win_width) x_offset += winx - win_width - block->rect.xmax; - } - - if (BLI_rctf_size_y(&block->rect) < winy - (2.0f * win_width)) { - if (block->rect.ymin < win_width ) y_offset += win_width - block->rect.ymin; - if (block->rect.ymax > winy - win_width) y_offset += winy - win_width - block->rect.ymax; - } - /* if we are offsetting set up initial data for timeout functionality */ - - if ((x_offset != 0) || (y_offset != 0)) { - block->pie_data.pie_center_spawned[0] += x_offset; - block->pie_data.pie_center_spawned[1] += y_offset; - - ui_block_translate(block, x_offset, y_offset); - - if (U.pie_initial_timeout > 0) - block->pie_data.flags |= UI_PIE_INITIAL_DIRECTION; - } - - ar->winrct.xmin = 0; - ar->winrct.xmax = winx; - ar->winrct.ymin = 0; - ar->winrct.ymax = winy; - - ui_block_calc_pie_segment(block, block->pie_data.pie_center_init); - - /* lastly set the buttons at the center of the pie menu, ready for animation */ - if (U.pie_animation_timeout > 0) { - for (uiBut *but_iter = block->buttons.first; but_iter; but_iter = but_iter->next) { - if (but_iter->pie_dir != UI_RADIAL_NONE) { - BLI_rctf_recenter(&but_iter->rect, UNPACK2(block->pie_data.pie_center_spawned)); - } - } - } - } - else { - /* clip block with window boundary */ - ui_popup_block_clip(window, block); - /* the block and buttons were positioned in window space as in 2.4x, now - * these menu blocks are regions so we bring it back to region space. - * additionally we add some padding for the menu shadow or rounded menus */ - ar->winrct.xmin = block->rect.xmin - margin; - ar->winrct.xmax = block->rect.xmax + margin; - ar->winrct.ymin = block->rect.ymin - margin; - ar->winrct.ymax = block->rect.ymax + UI_POPUP_MENU_TOP; - - ui_block_translate(block, -ar->winrct.xmin, -ar->winrct.ymin); - } - - if (block_old) { - block->oldblock = block_old; - UI_block_update_from_old(C, block); - UI_blocklist_free_inactive(C, &ar->uiblocks); - } - - /* checks which buttons are visible, sets flags to prevent draw (do after region init) */ - ui_popup_block_scrolltest(block); - - /* adds subwindow */ - ED_region_init(C, ar); - - /* get winmat now that we actually have the subwindow */ - wmSubWindowSet(window, ar->swinid); - - wm_subwindow_matrix_get(window, ar->swinid, block->winmat); - - /* notify change and redraw */ - ED_region_tag_redraw(ar); - - ED_region_update_rect(C, ar); - -#ifdef DEBUG - window->eventstate = event_back; -#endif - - return block; -} - -uiPopupBlockHandle *ui_popup_block_create( - bContext *C, ARegion *butregion, uiBut *but, - uiBlockCreateFunc create_func, uiBlockHandleCreateFunc handle_create_func, - void *arg) -{ - wmWindow *window = CTX_wm_window(C); - uiBut *activebut = UI_context_active_but_get(C); - static ARegionType type; - ARegion *ar; - uiBlock *block; - uiPopupBlockHandle *handle; - - /* disable tooltips from buttons below */ - if (activebut) { - UI_but_tooltip_timer_remove(C, activebut); - } - /* standard cursor by default */ - WM_cursor_set(window, CURSOR_STD); - - /* create handle */ - handle = MEM_callocN(sizeof(uiPopupBlockHandle), "uiPopupBlockHandle"); - - /* store context for operator */ - handle->ctx_area = CTX_wm_area(C); - handle->ctx_region = CTX_wm_region(C); - - /* store vars to refresh popup (RGN_DRAW_REFRESH_UI) */ - handle->popup_create_vars.create_func = create_func; - handle->popup_create_vars.handle_create_func = handle_create_func; - handle->popup_create_vars.arg = arg; - handle->popup_create_vars.butregion = but ? butregion : NULL; - copy_v2_v2_int(handle->popup_create_vars.event_xy, &window->eventstate->x); - /* caller may free vars used to create this popup, in that case this variable should be disabled. */ - handle->can_refresh = true; - - /* create area region */ - ar = ui_region_temp_add(CTX_wm_screen(C)); - handle->region = ar; - - memset(&type, 0, sizeof(ARegionType)); - type.draw = ui_block_region_draw; - type.regionid = RGN_TYPE_TEMPORARY; - ar->type = &type; - - UI_region_handlers_add(&ar->handlers); - - block = ui_popup_block_refresh(C, handle, butregion, but); - handle = block->handle; - - /* keep centered on window resizing */ - if ((block->bounds_type == UI_BLOCK_BOUNDS_POPUP_CENTER) && handle->can_refresh) { - type.listener = ui_block_region_popup_window_listener; - } - - return handle; -} - -void ui_popup_block_free(bContext *C, uiPopupBlockHandle *handle) -{ - ui_popup_block_remove(C, handle); - - MEM_freeN(handle); -} - -/***************************** Menu Button ***************************/ - -#if 0 -static void ui_warp_pointer(int x, int y) -{ - /* XXX 2.50 which function to use for this? */ - /* OSX has very poor mouse-warp support, it sends events; - * this causes a menu being pressed immediately ... */ -# ifndef __APPLE__ - warp_pointer(x, y); -# endif -} -#endif - -/********************* Color Button ****************/ - -/* for picker, while editing hsv */ -void ui_but_hsv_set(uiBut *but) -{ - float col[3]; - ColorPicker *cpicker = but->custom_data; - float *hsv = cpicker->color_data; - - ui_color_picker_to_rgb_v(hsv, col); - - ui_but_v3_set(but, col); -} - -/* Updates all buttons who share the same color picker as the one passed - * also used by small picker, be careful with name checks below... */ -static void ui_update_color_picker_buts_rgb(uiBlock *block, ColorPicker *cpicker, const float rgb[3], bool is_display_space) -{ - uiBut *bt; - float *hsv = cpicker->color_data; - struct ColorManagedDisplay *display = NULL; - /* this is to keep the H and S value when V is equal to zero - * and we are working in HSV mode, of course! - */ - if (is_display_space) { - ui_rgb_to_color_picker_compat_v(rgb, hsv); - } - else { - /* we need to convert to display space to use hsv, because hsv is stored in display space */ - float rgb_display[3]; - - copy_v3_v3(rgb_display, rgb); - ui_block_cm_to_display_space_v3(block, rgb_display); - ui_rgb_to_color_picker_compat_v(rgb_display, hsv); - } - - if (block->color_profile) - display = ui_block_cm_display_get(block); - - /* this updates button strings, is hackish... but button pointers are on stack of caller function */ - for (bt = block->buttons.first; bt; bt = bt->next) { - if (bt->custom_data != cpicker) - continue; - - if (bt->rnaprop) { - ui_but_v3_set(bt, rgb); - - /* original button that created the color picker already does undo - * push, so disable it on RNA buttons in the color picker block */ - UI_but_flag_disable(bt, UI_BUT_UNDO); - } - else if (STREQ(bt->str, "Hex: ")) { - float rgb_gamma[3]; - unsigned char rgb_gamma_uchar[3]; - double intpart; - char col[16]; - - /* Hex code is assumed to be in sRGB space (coming from other applications, web, etc) */ - - copy_v3_v3(rgb_gamma, rgb); - - if (display) { - /* make a display version, for Hex code */ - IMB_colormanagement_scene_linear_to_display_v3(rgb_gamma, display); - } - - if (rgb_gamma[0] > 1.0f) rgb_gamma[0] = modf(rgb_gamma[0], &intpart); - if (rgb_gamma[1] > 1.0f) rgb_gamma[1] = modf(rgb_gamma[1], &intpart); - if (rgb_gamma[2] > 1.0f) rgb_gamma[2] = modf(rgb_gamma[2], &intpart); - - rgb_float_to_uchar(rgb_gamma_uchar, rgb_gamma); - BLI_snprintf(col, sizeof(col), "%02X%02X%02X", UNPACK3_EX((unsigned int), rgb_gamma_uchar, )); - - strcpy(bt->poin, col); - } - else if (bt->str[1] == ' ') { - if (bt->str[0] == 'R') { - ui_but_value_set(bt, rgb[0]); - } - else if (bt->str[0] == 'G') { - ui_but_value_set(bt, rgb[1]); - } - else if (bt->str[0] == 'B') { - ui_but_value_set(bt, rgb[2]); - } - else if (bt->str[0] == 'H') { - ui_but_value_set(bt, hsv[0]); - } - else if (bt->str[0] == 'S') { - ui_but_value_set(bt, hsv[1]); - } - else if (bt->str[0] == 'V') { - ui_but_value_set(bt, hsv[2]); - } - else if (bt->str[0] == 'L') { - ui_but_value_set(bt, hsv[2]); - } - } - - ui_but_update(bt); - } -} - -static void ui_colorpicker_rna_cb(bContext *UNUSED(C), void *bt1, void *UNUSED(arg)) -{ - uiBut *but = (uiBut *)bt1; - uiPopupBlockHandle *popup = but->block->handle; - PropertyRNA *prop = but->rnaprop; - PointerRNA ptr = but->rnapoin; - float rgb[4]; - - if (prop) { - RNA_property_float_get_array(&ptr, prop, rgb); - ui_update_color_picker_buts_rgb(but->block, but->custom_data, rgb, (RNA_property_subtype(prop) == PROP_COLOR_GAMMA)); - } - - if (popup) - popup->menuretval = UI_RETURN_UPDATE; -} - -static void ui_color_wheel_rna_cb(bContext *UNUSED(C), void *bt1, void *UNUSED(arg)) -{ - uiBut *but = (uiBut *)bt1; - uiPopupBlockHandle *popup = but->block->handle; - float rgb[3]; - ColorPicker *cpicker = but->custom_data; - float *hsv = cpicker->color_data; - bool use_display_colorspace = ui_but_is_colorpicker_display_space(but); - - ui_color_picker_to_rgb_v(hsv, rgb); - - /* hsv is saved in display space so convert back */ - if (use_display_colorspace) { - ui_block_cm_to_scene_linear_v3(but->block, rgb); - } - - ui_update_color_picker_buts_rgb(but->block, cpicker, rgb, !use_display_colorspace); - - if (popup) - popup->menuretval = UI_RETURN_UPDATE; -} - -static void ui_colorpicker_hex_rna_cb(bContext *UNUSED(C), void *bt1, void *hexcl) -{ - uiBut *but = (uiBut *)bt1; - uiPopupBlockHandle *popup = but->block->handle; - ColorPicker *cpicker = but->custom_data; - char *hexcol = (char *)hexcl; - float rgb[3]; - - hex_to_rgb(hexcol, rgb, rgb + 1, rgb + 2); - - /* Hex code is assumed to be in sRGB space (coming from other applications, web, etc) */ - if (but->block->color_profile) { - /* so we need to linearise it for Blender */ - ui_block_cm_to_scene_linear_v3(but->block, rgb); - } - - ui_update_color_picker_buts_rgb(but->block, cpicker, rgb, false); - - if (popup) - popup->menuretval = UI_RETURN_UPDATE; -} - -static void ui_popup_close_cb(bContext *UNUSED(C), void *bt1, void *UNUSED(arg)) -{ - uiBut *but = (uiBut *)bt1; - uiPopupBlockHandle *popup = but->block->handle; - - if (popup) - popup->menuretval = UI_RETURN_OK; -} - -static void ui_colorpicker_hide_reveal(uiBlock *block, short colormode) -{ - uiBut *bt; - - /* tag buttons */ - for (bt = block->buttons.first; bt; bt = bt->next) { - if ((bt->func == ui_colorpicker_rna_cb) && bt->type == UI_BTYPE_NUM_SLIDER && bt->rnaindex != 3) { - /* RGB sliders (color circle and alpha are always shown) */ - if (colormode == 0) bt->flag &= ~UI_HIDDEN; - else bt->flag |= UI_HIDDEN; - } - else if (bt->func == ui_color_wheel_rna_cb) { - /* HSV sliders */ - if (colormode == 1) bt->flag &= ~UI_HIDDEN; - else bt->flag |= UI_HIDDEN; - } - else if (bt->func == ui_colorpicker_hex_rna_cb || bt->type == UI_BTYPE_LABEL) { - /* hex input or gamma correction status label */ - if (colormode == 2) bt->flag &= ~UI_HIDDEN; - else bt->flag |= UI_HIDDEN; - } - } -} - -static void ui_colorpicker_create_mode_cb(bContext *UNUSED(C), void *bt1, void *UNUSED(arg)) -{ - uiBut *bt = bt1; - short colormode = ui_but_value_get(bt); - ui_colorpicker_hide_reveal(bt->block, colormode); -} - -#define PICKER_H (7.5f * U.widget_unit) -#define PICKER_W (7.5f * U.widget_unit) -#define PICKER_SPACE (0.3f * U.widget_unit) -#define PICKER_BAR (0.7f * U.widget_unit) - -#define PICKER_TOTAL_W (PICKER_W + PICKER_SPACE + PICKER_BAR) - -static void ui_colorpicker_circle(uiBlock *block, PointerRNA *ptr, PropertyRNA *prop, ColorPicker *cpicker) -{ - uiBut *bt; - - /* HS circle */ - bt = uiDefButR_prop(block, UI_BTYPE_HSVCIRCLE, 0, "", 0, 0, PICKER_H, PICKER_W, ptr, prop, -1, 0.0, 0.0, 0.0, 0, TIP_("Color")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - bt->custom_data = cpicker; - - /* value */ - if (U.color_picker_type == USER_CP_CIRCLE_HSL) { - bt = uiDefButR_prop(block, UI_BTYPE_HSVCUBE, 0, "", PICKER_W + PICKER_SPACE, 0, PICKER_BAR, PICKER_H, ptr, prop, -1, 0.0, 0.0, UI_GRAD_L_ALT, 0, "Lightness"); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - } - else { - bt = uiDefButR_prop(block, UI_BTYPE_HSVCUBE, 0, "", PICKER_W + PICKER_SPACE, 0, PICKER_BAR, PICKER_H, ptr, prop, -1, 0.0, 0.0, UI_GRAD_V_ALT, 0, TIP_("Value")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - } - bt->custom_data = cpicker; -} - - -static void ui_colorpicker_square(uiBlock *block, PointerRNA *ptr, PropertyRNA *prop, int type, ColorPicker *cpicker) -{ - uiBut *bt; - int bartype = type + 3; - - /* HS square */ - bt = uiDefButR_prop(block, UI_BTYPE_HSVCUBE, 0, "", 0, PICKER_BAR + PICKER_SPACE, PICKER_TOTAL_W, PICKER_H, ptr, prop, -1, 0.0, 0.0, type, 0, TIP_("Color")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - bt->custom_data = cpicker; - - /* value */ - bt = uiDefButR_prop(block, UI_BTYPE_HSVCUBE, 0, "", 0, 0, PICKER_TOTAL_W, PICKER_BAR, ptr, prop, -1, 0.0, 0.0, bartype, 0, TIP_("Value")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - bt->custom_data = cpicker; -} - - -/* a HS circle, V slider, rgb/hsv/hex sliders */ -static void ui_block_colorpicker(uiBlock *block, float rgba[4], PointerRNA *ptr, PropertyRNA *prop, bool show_picker) -{ - static short colormode = 0; /* temp? 0=rgb, 1=hsv, 2=hex */ - uiBut *bt; - int width, butwidth; - static char tip[50]; - static char hexcol[128]; - float rgb_gamma[3]; - unsigned char rgb_gamma_uchar[3]; - float softmin, softmax, hardmin, hardmax, step, precision; - int yco; - ColorPicker *cpicker = ui_block_colorpicker_create(block); - float *hsv = cpicker->color_data; - - width = PICKER_TOTAL_W; - butwidth = width - 1.5f * UI_UNIT_X; - - /* existence of profile means storage is in linear color space, with display correction */ - /* XXX That tip message is not use anywhere! */ - if (!block->color_profile) { - BLI_strncpy(tip, N_("Value in Display Color Space"), sizeof(tip)); - copy_v3_v3(rgb_gamma, rgba); - } - else { - BLI_strncpy(tip, N_("Value in Linear RGB Color Space"), sizeof(tip)); - - /* make a display version, for Hex code */ - copy_v3_v3(rgb_gamma, rgba); - ui_block_cm_to_display_space_v3(block, rgb_gamma); - } - - /* sneaky way to check for alpha */ - rgba[3] = FLT_MAX; - - RNA_property_float_ui_range(ptr, prop, &softmin, &softmax, &step, &precision); - RNA_property_float_range(ptr, prop, &hardmin, &hardmax); - RNA_property_float_get_array(ptr, prop, rgba); - - /* when the softmax isn't defined in the RNA, - * using very large numbers causes sRGB/linear round trip to fail. */ - if (softmax == FLT_MAX) { - softmax = 1.0f; - } - - switch (U.color_picker_type) { - case USER_CP_SQUARE_SV: - ui_colorpicker_square(block, ptr, prop, UI_GRAD_SV, cpicker); - break; - case USER_CP_SQUARE_HS: - ui_colorpicker_square(block, ptr, prop, UI_GRAD_HS, cpicker); - break; - case USER_CP_SQUARE_HV: - ui_colorpicker_square(block, ptr, prop, UI_GRAD_HV, cpicker); - break; - - /* user default */ - case USER_CP_CIRCLE_HSV: - case USER_CP_CIRCLE_HSL: - default: - ui_colorpicker_circle(block, ptr, prop, cpicker); - break; - } - - /* mode */ - yco = -1.5f * UI_UNIT_Y; - UI_block_align_begin(block); - bt = uiDefButS(block, UI_BTYPE_ROW, 0, IFACE_("RGB"), 0, yco, width / 3, UI_UNIT_Y, &colormode, 0.0, 0.0, 0, 0, ""); - UI_but_func_set(bt, ui_colorpicker_create_mode_cb, bt, NULL); - bt->custom_data = cpicker; - if (U.color_picker_type == USER_CP_CIRCLE_HSL) - bt = uiDefButS(block, UI_BTYPE_ROW, 0, IFACE_("HSL"), width / 3, yco, width / 3, UI_UNIT_Y, &colormode, 0.0, 1.0, 0, 0, ""); - else - bt = uiDefButS(block, UI_BTYPE_ROW, 0, IFACE_("HSV"), width / 3, yco, width / 3, UI_UNIT_Y, &colormode, 0.0, 1.0, 0, 0, ""); - UI_but_func_set(bt, ui_colorpicker_create_mode_cb, bt, NULL); - bt->custom_data = cpicker; - bt = uiDefButS(block, UI_BTYPE_ROW, 0, IFACE_("Hex"), 2 * width / 3, yco, width / 3, UI_UNIT_Y, &colormode, 0.0, 2.0, 0, 0, ""); - UI_but_func_set(bt, ui_colorpicker_create_mode_cb, bt, NULL); - bt->custom_data = cpicker; - UI_block_align_end(block); - - yco = -3.0f * UI_UNIT_Y; - if (show_picker) { - bt = uiDefIconButO(block, UI_BTYPE_BUT, "UI_OT_eyedropper_color", WM_OP_INVOKE_DEFAULT, ICON_EYEDROPPER, butwidth + 10, yco, UI_UNIT_X, UI_UNIT_Y, NULL); - UI_but_func_set(bt, ui_popup_close_cb, bt, NULL); - bt->custom_data = cpicker; - } - - /* RGB values */ - UI_block_align_begin(block); - bt = uiDefButR_prop(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("R:"), 0, yco, butwidth, UI_UNIT_Y, ptr, prop, 0, 0.0, 0.0, 0, 3, TIP_("Red")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - bt->custom_data = cpicker; - bt = uiDefButR_prop(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("G:"), 0, yco -= UI_UNIT_Y, butwidth, UI_UNIT_Y, ptr, prop, 1, 0.0, 0.0, 0, 3, TIP_("Green")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - bt->custom_data = cpicker; - bt = uiDefButR_prop(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("B:"), 0, yco -= UI_UNIT_Y, butwidth, UI_UNIT_Y, ptr, prop, 2, 0.0, 0.0, 0, 3, TIP_("Blue")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - bt->custom_data = cpicker; - - /* could use uiItemFullR(col, ptr, prop, -1, 0, UI_ITEM_R_EXPAND|UI_ITEM_R_SLIDER, "", ICON_NONE); - * but need to use UI_but_func_set for updating other fake buttons */ - - /* HSV values */ - yco = -3.0f * UI_UNIT_Y; - UI_block_align_begin(block); - bt = uiDefButF(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("H:"), 0, yco, butwidth, UI_UNIT_Y, hsv, 0.0, 1.0, 10, 3, TIP_("Hue")); - UI_but_func_set(bt, ui_color_wheel_rna_cb, bt, hsv); - bt->custom_data = cpicker; - bt = uiDefButF(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("S:"), 0, yco -= UI_UNIT_Y, butwidth, UI_UNIT_Y, hsv + 1, 0.0, 1.0, 10, 3, TIP_("Saturation")); - UI_but_func_set(bt, ui_color_wheel_rna_cb, bt, hsv); - bt->custom_data = cpicker; - if (U.color_picker_type == USER_CP_CIRCLE_HSL) - bt = uiDefButF(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("L:"), 0, yco -= UI_UNIT_Y, butwidth, UI_UNIT_Y, hsv + 2, 0.0, 1.0, 10, 3, TIP_("Lightness")); - else - bt = uiDefButF(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("V:"), 0, yco -= UI_UNIT_Y, butwidth, UI_UNIT_Y, hsv + 2, 0.0, softmax, 10, 3, TIP_("Value")); - - bt->hardmax = hardmax; /* not common but rgb may be over 1.0 */ - UI_but_func_set(bt, ui_color_wheel_rna_cb, bt, hsv); - bt->custom_data = cpicker; - - UI_block_align_end(block); - - if (rgba[3] != FLT_MAX) { - bt = uiDefButR_prop(block, UI_BTYPE_NUM_SLIDER, 0, IFACE_("A: "), 0, yco -= UI_UNIT_Y, butwidth, UI_UNIT_Y, ptr, prop, 3, 0.0, 0.0, 0, 3, TIP_("Alpha")); - UI_but_func_set(bt, ui_colorpicker_rna_cb, bt, NULL); - bt->custom_data = cpicker; - } - else { - rgba[3] = 1.0f; - } - - rgb_float_to_uchar(rgb_gamma_uchar, rgb_gamma); - BLI_snprintf(hexcol, sizeof(hexcol), "%02X%02X%02X", UNPACK3_EX((unsigned int), rgb_gamma_uchar, )); - - yco = -3.0f * UI_UNIT_Y; - bt = uiDefBut(block, UI_BTYPE_TEXT, 0, IFACE_("Hex: "), 0, yco, butwidth, UI_UNIT_Y, hexcol, 0, 8, 0, 0, TIP_("Hex triplet for color (#RRGGBB)")); - UI_but_func_set(bt, ui_colorpicker_hex_rna_cb, bt, hexcol); - bt->custom_data = cpicker; - uiDefBut(block, UI_BTYPE_LABEL, 0, IFACE_("(Gamma Corrected)"), 0, yco - UI_UNIT_Y, butwidth, UI_UNIT_Y, NULL, 0.0, 0.0, 0, 0, ""); - - ui_rgb_to_color_picker_v(rgb_gamma, hsv); - - ui_colorpicker_hide_reveal(block, colormode); -} - - -static int ui_colorpicker_small_wheel_cb(const bContext *UNUSED(C), uiBlock *block, const wmEvent *event) -{ - float add = 0.0f; - - if (event->type == WHEELUPMOUSE) - add = 0.05f; - else if (event->type == WHEELDOWNMOUSE) - add = -0.05f; - - if (add != 0.0f) { - uiBut *but; - - for (but = block->buttons.first; but; but = but->next) { - if (but->type == UI_BTYPE_HSVCUBE && but->active == NULL) { - uiPopupBlockHandle *popup = block->handle; - float rgb[3]; - ColorPicker *cpicker = but->custom_data; - float *hsv = cpicker->color_data; - bool use_display_colorspace = ui_but_is_colorpicker_display_space(but); - - ui_but_v3_get(but, rgb); - - if (use_display_colorspace) - ui_block_cm_to_display_space_v3(block, rgb); - - ui_rgb_to_color_picker_compat_v(rgb, hsv); - - hsv[2] = CLAMPIS(hsv[2] + add, 0.0f, 1.0f); - ui_color_picker_to_rgb_v(hsv, rgb); - - if (use_display_colorspace) - ui_block_cm_to_scene_linear_v3(block, rgb); - - ui_but_v3_set(but, rgb); - - ui_update_color_picker_buts_rgb(block, cpicker, rgb, !use_display_colorspace); - if (popup) - popup->menuretval = UI_RETURN_UPDATE; - - return 1; - } - } - } - return 0; -} - -uiBlock *ui_block_func_COLOR(bContext *C, uiPopupBlockHandle *handle, void *arg_but) -{ - uiBut *but = arg_but; - uiBlock *block; - bool show_picker = true; - - block = UI_block_begin(C, handle->region, __func__, UI_EMBOSS); - - if (RNA_property_subtype(but->rnaprop) == PROP_COLOR_GAMMA) { - block->color_profile = false; - } - - if (but->block) { - /* if color block is invoked from a popup we wouldn't be able to set color properly - * this is because color picker will close popups first and then will try to figure - * out active button RNA, and of course it'll fail - */ - show_picker = (but->block->flag & UI_BLOCK_POPUP) == 0; - } - - copy_v3_v3(handle->retvec, but->editvec); - - ui_block_colorpicker(block, handle->retvec, &but->rnapoin, but->rnaprop, show_picker); - - block->flag = UI_BLOCK_LOOP | UI_BLOCK_KEEP_OPEN | UI_BLOCK_OUT_1 | UI_BLOCK_MOVEMOUSE_QUIT; - UI_block_bounds_set_normal(block, 0.5 * UI_UNIT_X); - - block->block_event_func = ui_colorpicker_small_wheel_cb; - - /* and lets go */ - block->direction = UI_DIR_UP; - - return block; -} - -/************************ Popup Menu Memory ****************************/ - -static unsigned int ui_popup_string_hash(const char *str) -{ - /* sometimes button contains hotkey, sometimes not, strip for proper compare */ - int hash; - const char *delimit = strrchr(str, UI_SEP_CHAR); - - if (delimit) { - hash = BLI_ghashutil_strhash_n(str, delimit - str); - } - else { - hash = BLI_ghashutil_strhash(str); - } - - return hash; -} - -static unsigned int ui_popup_menu_hash(const char *str) -{ - return BLI_ghashutil_strhash(str); -} - -/* but == NULL read, otherwise set */ -static uiBut *ui_popup_menu_memory__internal(uiBlock *block, uiBut *but) -{ - static unsigned int mem[256]; - static bool first = true; - - const unsigned int hash = block->puphash; - const unsigned int hash_mod = hash & 255; - - if (first) { - /* init */ - memset(mem, -1, sizeof(mem)); - first = 0; - } - - if (but) { - /* set */ - mem[hash_mod] = ui_popup_string_hash(but->str); - return NULL; - } - else { - /* get */ - for (but = block->buttons.first; but; but = but->next) - if (ui_popup_string_hash(but->str) == mem[hash_mod]) - return but; - - return NULL; - } -} - -uiBut *ui_popup_menu_memory_get(uiBlock *block) -{ - return ui_popup_menu_memory__internal(block, NULL); -} - -void ui_popup_menu_memory_set(uiBlock *block, uiBut *but) -{ - ui_popup_menu_memory__internal(block, but); -} - -/** - * Translate any popup regions (so we can drag them). - */ -void ui_popup_translate(bContext *C, ARegion *ar, const int mdiff[2]) -{ - uiBlock *block; - - BLI_rcti_translate(&ar->winrct, UNPACK2(mdiff)); - - ED_region_update_rect(C, ar); - - ED_region_tag_redraw(ar); - - /* update blocks */ - for (block = ar->uiblocks.first; block; block = block->next) { - uiSafetyRct *saferct; - for (saferct = block->saferct.first; saferct; saferct = saferct->next) { - BLI_rctf_translate(&saferct->parent, UNPACK2(mdiff)); - BLI_rctf_translate(&saferct->safety, UNPACK2(mdiff)); - } - } -} - -/******************** Popup Menu with callback or string **********************/ - -struct uiPopupMenu { - uiBlock *block; - uiLayout *layout; - uiBut *but; - ARegion *butregion; - - int mx, my; - bool popup, slideout; - - uiMenuCreateFunc menu_func; - void *menu_arg; -}; - -struct uiPieMenu { - uiBlock *block_radial; /* radial block of the pie menu (more could be added later) */ - uiLayout *layout; - int mx, my; -}; - -static uiBlock *ui_block_func_POPUP(bContext *C, uiPopupBlockHandle *handle, void *arg_pup) -{ - uiBlock *block; - uiBut *bt; - uiPopupMenu *pup = arg_pup; - int offset[2], minwidth, width, height; - char direction; - bool flip; - - if (pup->menu_func) { - pup->block->handle = handle; - pup->menu_func(C, pup->layout, pup->menu_arg); - pup->block->handle = NULL; - } - - if (pup->but) { - /* minimum width to enforece */ - minwidth = BLI_rctf_size_x(&pup->but->rect); - - /* settings (typically rna-enum-popups) show above the button, - * menu's like file-menu, show below */ - if (pup->block->direction != 0) { - /* allow overriding the direction from menu_func */ - direction = pup->block->direction; - } - else if ((pup->but->type == UI_BTYPE_PULLDOWN) || - (UI_but_menutype_get(pup->but) != NULL)) - { - direction = UI_DIR_DOWN; - } - else { - direction = UI_DIR_UP; - } - } - else { - minwidth = 50; - direction = UI_DIR_DOWN; - } - - flip = (direction == UI_DIR_DOWN); - - block = pup->block; - - /* in some cases we create the block before the region, - * so we set it delayed here if necessary */ - if (BLI_findindex(&handle->region->uiblocks, block) == -1) - UI_block_region_set(block, handle->region); - - block->direction = direction; - - UI_block_layout_resolve(block, &width, &height); - - UI_block_flag_enable(block, UI_BLOCK_MOVEMOUSE_QUIT); - - if (pup->popup) { - uiBut *but_activate = NULL; - UI_block_flag_enable(block, UI_BLOCK_LOOP | UI_BLOCK_NUMSELECT); - UI_block_direction_set(block, direction); - - /* offset the mouse position, possibly based on earlier selection */ - if ((block->flag & UI_BLOCK_POPUP_MEMORY) && - (bt = ui_popup_menu_memory_get(block))) - { - /* position mouse on last clicked item, at 0.8*width of the - * button, so it doesn't overlap the text too much, also note - * the offset is negative because we are inverse moving the - * block to be under the mouse */ - offset[0] = -(bt->rect.xmin + 0.8f * BLI_rctf_size_x(&bt->rect)); - offset[1] = -(bt->rect.ymin + 0.5f * UI_UNIT_Y); - - if (ui_but_is_editable(bt)) { - but_activate = bt; - } - } - else { - /* position mouse at 0.8*width of the button and below the tile - * on the first item */ - offset[0] = 0; - for (bt = block->buttons.first; bt; bt = bt->next) - offset[0] = min_ii(offset[0], -(bt->rect.xmin + 0.8f * BLI_rctf_size_x(&bt->rect))); - - offset[1] = 2.1 * UI_UNIT_Y; - - for (bt = block->buttons.first; bt; bt = bt->next) { - if (ui_but_is_editable(bt)) { - but_activate = bt; - break; - } - } - } - - /* in rare cases this is needed since moving the popup - * to be within the window bounds may move it away from the mouse, - * This ensures we set an item to be active. */ - if (but_activate) { - ui_but_activate_over(C, handle->region, but_activate); - } - - block->minbounds = minwidth; - UI_block_bounds_set_menu(block, 1, offset[0], offset[1]); - } - else { - /* for a header menu we set the direction automatic */ - if (!pup->slideout && flip) { - ScrArea *sa = CTX_wm_area(C); - if (sa && sa->headertype == HEADERDOWN) { - ARegion *ar = CTX_wm_region(C); - if (ar && ar->regiontype == RGN_TYPE_HEADER) { - UI_block_direction_set(block, UI_DIR_UP); - UI_block_order_flip(block); - } - } - } - - block->minbounds = minwidth; - UI_block_bounds_set_text(block, 3.0f * UI_UNIT_X); - } - - /* if menu slides out of other menu, override direction */ - if (pup->slideout) - UI_block_direction_set(block, UI_DIR_RIGHT); - - return pup->block; -} - -uiPopupBlockHandle *ui_popup_menu_create( - bContext *C, ARegion *butregion, uiBut *but, - uiMenuCreateFunc menu_func, void *arg) -{ - wmWindow *window = CTX_wm_window(C); - uiStyle *style = UI_style_get_dpi(); - uiPopupBlockHandle *handle; - uiPopupMenu *pup; - - pup = MEM_callocN(sizeof(uiPopupMenu), __func__); - pup->block = UI_block_begin(C, NULL, __func__, UI_EMBOSS_PULLDOWN); - pup->block->flag |= UI_BLOCK_NUMSELECT; /* default menus to numselect */ - pup->layout = UI_block_layout(pup->block, UI_LAYOUT_VERTICAL, UI_LAYOUT_MENU, 0, 0, 200, 0, MENU_PADDING, style); - pup->slideout = but ? ui_block_is_menu(but->block) : false; - pup->but = but; - uiLayoutSetOperatorContext(pup->layout, WM_OP_INVOKE_REGION_WIN); - - if (!but) { - /* no button to start from, means we are a popup */ - pup->mx = window->eventstate->x; - pup->my = window->eventstate->y; - pup->popup = true; - pup->block->flag |= UI_BLOCK_NO_FLIP; - } - /* some enums reversing is strange, currently we have no good way to - * reverse some enum's but not others, so reverse all so the first menu - * items are always close to the mouse cursor */ - else { -#if 0 - /* if this is an rna button then we can assume its an enum - * flipping enums is generally not good since the order can be - * important [#28786] */ - if (but->rnaprop && RNA_property_type(but->rnaprop) == PROP_ENUM) { - pup->block->flag |= UI_BLOCK_NO_FLIP; - } -#endif - if (but->context) - uiLayoutContextCopy(pup->layout, but->context); - } - - /* menu is created from a callback */ - pup->menu_func = menu_func; - pup->menu_arg = arg; - - handle = ui_popup_block_create(C, butregion, but, NULL, ui_block_func_POPUP, pup); - - if (!but) { - handle->popup = true; - - UI_popup_handlers_add(C, &window->modalhandlers, handle, 0); - WM_event_add_mousemove(C); - } - - handle->can_refresh = false; - MEM_freeN(pup); - - return handle; -} - -/******************** Popup Menu API with begin and end ***********************/ - -/** - * Only return handler, and set optional title. - * \param block_name: Assigned to uiBlock.name (useful info for debugging). - */ -uiPopupMenu *UI_popup_menu_begin_ex(bContext *C, const char *title, const char *block_name, int icon) -{ - uiStyle *style = UI_style_get_dpi(); - uiPopupMenu *pup = MEM_callocN(sizeof(uiPopupMenu), "popup menu"); - uiBut *but; - - pup->block = UI_block_begin(C, NULL, block_name, UI_EMBOSS_PULLDOWN); - pup->block->flag |= UI_BLOCK_POPUP_MEMORY | UI_BLOCK_IS_FLIP; - pup->block->puphash = ui_popup_menu_hash(title); - pup->layout = UI_block_layout(pup->block, UI_LAYOUT_VERTICAL, UI_LAYOUT_MENU, 0, 0, 200, 0, MENU_PADDING, style); - - /* note, this intentionally differs from the menu & submenu default because many operators - * use popups like this to select one of their options - where having invoke doesn't make sense */ - uiLayoutSetOperatorContext(pup->layout, WM_OP_EXEC_REGION_WIN); - - /* create in advance so we can let buttons point to retval already */ - pup->block->handle = MEM_callocN(sizeof(uiPopupBlockHandle), "uiPopupBlockHandle"); - - /* create title button */ - if (title[0]) { - char titlestr[256]; - - if (icon) { - BLI_snprintf(titlestr, sizeof(titlestr), " %s", title); - uiDefIconTextBut(pup->block, UI_BTYPE_LABEL, 0, icon, titlestr, 0, 0, 200, UI_UNIT_Y, NULL, 0.0, 0.0, 0, 0, ""); - } - else { - but = uiDefBut(pup->block, UI_BTYPE_LABEL, 0, title, 0, 0, 200, UI_UNIT_Y, NULL, 0.0, 0.0, 0, 0, ""); - but->drawflag = UI_BUT_TEXT_LEFT; - } - - uiItemS(pup->layout); - } - - return pup; -} - -uiPopupMenu *UI_popup_menu_begin(bContext *C, const char *title, int icon) -{ - return UI_popup_menu_begin_ex(C, title, __func__, icon); -} - -/** - * Setting the button makes the popup open from the button instead of the cursor. - */ -void UI_popup_menu_but_set(uiPopupMenu *pup, struct ARegion *butregion, uiBut *but) -{ - pup->but = but; - pup->butregion = butregion; -} - -/* set the whole structure to work */ -void UI_popup_menu_end(bContext *C, uiPopupMenu *pup) -{ - wmWindow *window = CTX_wm_window(C); - uiPopupBlockHandle *menu; - uiBut *but = NULL; - ARegion *butregion = NULL; - - pup->popup = true; - pup->mx = window->eventstate->x; - pup->my = window->eventstate->y; - - if (pup->but) { - but = pup->but; - butregion = pup->butregion; - } - - menu = ui_popup_block_create(C, butregion, but, NULL, ui_block_func_POPUP, pup); - menu->popup = true; - - UI_popup_handlers_add(C, &window->modalhandlers, menu, 0); - WM_event_add_mousemove(C); - - menu->can_refresh = false; - MEM_freeN(pup); -} - -uiLayout *UI_popup_menu_layout(uiPopupMenu *pup) -{ - return pup->layout; -} - -/*************************** Pie Menus ***************************************/ - -static uiBlock *ui_block_func_PIE(bContext *UNUSED(C), uiPopupBlockHandle *handle, void *arg_pie) -{ - uiBlock *block; - uiPieMenu *pie = arg_pie; - int minwidth, width, height; - - minwidth = 50; - block = pie->block_radial; - - /* in some cases we create the block before the region, - * so we set it delayed here if necessary */ - if (BLI_findindex(&handle->region->uiblocks, block) == -1) - UI_block_region_set(block, handle->region); - - UI_block_layout_resolve(block, &width, &height); - - UI_block_flag_enable(block, UI_BLOCK_LOOP | UI_BLOCK_NUMSELECT); - - block->minbounds = minwidth; - block->bounds = 1; - block->mx = 0; - block->my = 0; - block->bounds_type = UI_BLOCK_BOUNDS_PIE_CENTER; - - block->pie_data.pie_center_spawned[0] = pie->mx; - block->pie_data.pie_center_spawned[1] = pie->my; - - return pie->block_radial; -} - -static float ui_pie_menu_title_width(const char *name, int icon) -{ - const uiFontStyle *fstyle = UI_FSTYLE_WIDGET; - return (UI_fontstyle_string_width(fstyle, name) + - (UI_UNIT_X * (1.50f + (icon ? 0.25f : 0.0f)))); -} - -uiPieMenu *UI_pie_menu_begin(struct bContext *C, const char *title, int icon, const wmEvent *event) -{ - uiStyle *style; - uiPieMenu *pie; - short event_type; - - wmWindow *win = CTX_wm_window(C); - - style = UI_style_get_dpi(); - pie = MEM_callocN(sizeof(*pie), "pie menu"); - - pie->block_radial = UI_block_begin(C, NULL, __func__, UI_EMBOSS); - /* may be useful later to allow spawning pies - * from old positions */ - /* pie->block_radial->flag |= UI_BLOCK_POPUP_MEMORY; */ - pie->block_radial->puphash = ui_popup_menu_hash(title); - pie->block_radial->flag |= UI_BLOCK_RADIAL; - - /* if pie is spawned by a left click, release or click event, it is always assumed to be click style */ - if (event->type == LEFTMOUSE || ELEM(event->val, KM_RELEASE, KM_CLICK)) { - pie->block_radial->pie_data.flags |= UI_PIE_CLICK_STYLE; - pie->block_radial->pie_data.event = EVENT_NONE; - win->lock_pie_event = EVENT_NONE; - } - else { - if (win->last_pie_event != EVENT_NONE) { - /* original pie key has been released, so don't propagate the event */ - if (win->lock_pie_event == EVENT_NONE) { - event_type = EVENT_NONE; - pie->block_radial->pie_data.flags |= UI_PIE_CLICK_STYLE; - } - else - event_type = win->last_pie_event; - } - else { - event_type = event->type; - } - - pie->block_radial->pie_data.event = event_type; - win->lock_pie_event = event_type; - } - - pie->layout = UI_block_layout(pie->block_radial, UI_LAYOUT_VERTICAL, UI_LAYOUT_PIEMENU, 0, 0, 200, 0, 0, style); - pie->mx = event->x; - pie->my = event->y; - - /* create title button */ - if (title[0]) { - uiBut *but; - char titlestr[256]; - int w; - if (icon) { - BLI_snprintf(titlestr, sizeof(titlestr), " %s", title); - w = ui_pie_menu_title_width(titlestr, icon); - but = uiDefIconTextBut(pie->block_radial, UI_BTYPE_LABEL, 0, icon, titlestr, 0, 0, w, UI_UNIT_Y, NULL, 0.0, 0.0, 0, 0, ""); - } - else { - w = ui_pie_menu_title_width(title, 0); - but = uiDefBut(pie->block_radial, UI_BTYPE_LABEL, 0, title, 0, 0, w, UI_UNIT_Y, NULL, 0.0, 0.0, 0, 0, ""); - } - /* do not align left */ - but->drawflag &= ~UI_BUT_TEXT_LEFT; - pie->block_radial->pie_data.title = but->str; - pie->block_radial->pie_data.icon = icon; - } - - return pie; -} - -void UI_pie_menu_end(bContext *C, uiPieMenu *pie) -{ - wmWindow *window = CTX_wm_window(C); - uiPopupBlockHandle *menu; - - menu = ui_popup_block_create(C, NULL, NULL, NULL, ui_block_func_PIE, pie); - menu->popup = true; - menu->towardstime = PIL_check_seconds_timer(); - - UI_popup_handlers_add( - C, &window->modalhandlers, - menu, WM_HANDLER_ACCEPT_DBL_CLICK); - WM_event_add_mousemove(C); - - menu->can_refresh = false; - MEM_freeN(pie); -} - -uiLayout *UI_pie_menu_layout(uiPieMenu *pie) -{ - return pie->layout; -} - -int UI_pie_menu_invoke(struct bContext *C, const char *idname, const wmEvent *event) -{ - uiPieMenu *pie; - uiLayout *layout; - MenuType *mt = WM_menutype_find(idname, true); - - if (mt == NULL) { - printf("%s: named menu \"%s\" not found\n", __func__, idname); - return OPERATOR_CANCELLED; - } - - if (mt->poll && mt->poll(C, mt) == 0) - /* cancel but allow event to pass through, just like operators do */ - return (OPERATOR_CANCELLED | OPERATOR_PASS_THROUGH); - - pie = UI_pie_menu_begin(C, IFACE_(mt->label), ICON_NONE, event); - layout = UI_pie_menu_layout(pie); - - UI_menutype_draw(C, mt, layout); - - UI_pie_menu_end(C, pie); - - return OPERATOR_INTERFACE; -} - -int UI_pie_menu_invoke_from_operator_enum( - struct bContext *C, const char *title, const char *opname, - const char *propname, const wmEvent *event) -{ - uiPieMenu *pie; - uiLayout *layout; - - pie = UI_pie_menu_begin(C, IFACE_(title), ICON_NONE, event); - layout = UI_pie_menu_layout(pie); - - layout = uiLayoutRadial(layout); - uiItemsEnumO(layout, opname, propname); - - UI_pie_menu_end(C, pie); - - return OPERATOR_INTERFACE; -} - -int UI_pie_menu_invoke_from_rna_enum( - struct bContext *C, const char *title, const char *path, - const wmEvent *event) -{ - PointerRNA ctx_ptr; - PointerRNA r_ptr; - PropertyRNA *r_prop; - uiPieMenu *pie; - uiLayout *layout; - - RNA_pointer_create(NULL, &RNA_Context, C, &ctx_ptr); - - if (!RNA_path_resolve(&ctx_ptr, path, &r_ptr, &r_prop)) { - return OPERATOR_CANCELLED; - } - - /* invalid property, only accept enums */ - if (RNA_property_type(r_prop) != PROP_ENUM) { - BLI_assert(0); - return OPERATOR_CANCELLED; - } - - pie = UI_pie_menu_begin(C, IFACE_(title), ICON_NONE, event); - - layout = UI_pie_menu_layout(pie); - - layout = uiLayoutRadial(layout); - uiItemFullR(layout, &r_ptr, r_prop, RNA_NO_INDEX, 0, UI_ITEM_R_EXPAND, NULL, 0); - - UI_pie_menu_end(C, pie); - - return OPERATOR_INTERFACE; -} - -/** - * \name Pie Menu Levels - * - * Pie menus can't contain more than 8 items (yet). When using #uiItemsFullEnumO, a "More" button is created that calls - * a new pie menu if the enum has too many items. We call this a new "level". - * Indirect recursion is used, so that a theoretically unlimited number of items is supported. - * - * This is a implementation specifically for operator enums, needed since the object mode pie now has more than 8 - * items. Ideally we'd have some way of handling this for all kinds of pie items, but that's tricky. - * - * - Julian (Feb 2016) - * - * \{ */ - -typedef struct PieMenuLevelData { - char title[UI_MAX_NAME_STR]; /* parent pie title, copied for level */ - int icon; /* parent pie icon, copied for level */ - int totitem; /* total count of *remaining* items */ - - /* needed for calling uiItemsFullEnumO_array again for new level */ - wmOperatorType *ot; - const char *propname; - IDProperty *properties; - int context, flag; -} PieMenuLevelData; - -/** - * Invokes a new pie menu for a new level. - */ -static void ui_pie_menu_level_invoke(bContext *C, void *argN, void *arg2) -{ - EnumPropertyItem *item_array = (EnumPropertyItem *)argN; - PieMenuLevelData *lvl = (PieMenuLevelData *)arg2; - wmWindow *win = CTX_wm_window(C); - - uiPieMenu *pie = UI_pie_menu_begin(C, IFACE_(lvl->title), lvl->icon, win->eventstate); - uiLayout *layout = UI_pie_menu_layout(pie); - - layout = uiLayoutRadial(layout); - - PointerRNA ptr; - - WM_operator_properties_create_ptr(&ptr, lvl->ot); - /* so the context is passed to itemf functions (some need it) */ - WM_operator_properties_sanitize(&ptr, false); - PropertyRNA *prop = RNA_struct_find_property(&ptr, lvl->propname); - - if (prop) { - uiItemsFullEnumO_items( - layout, lvl->ot, ptr, prop, lvl->properties, lvl->context, lvl->flag, - item_array, lvl->totitem); - } - else { - RNA_warning("%s.%s not found", RNA_struct_identifier(ptr.type), lvl->propname); - } - - UI_pie_menu_end(C, pie); -} - -/** - * Set up data for defining a new pie menu level and add button that invokes it. - */ -void ui_pie_menu_level_create( - uiBlock *block, wmOperatorType *ot, const char *propname, IDProperty *properties, - const EnumPropertyItem *items, int totitem, int context, int flag) -{ - const int totitem_parent = PIE_MAX_ITEMS - 1; - const int totitem_remain = totitem - totitem_parent; - size_t array_size = sizeof(EnumPropertyItem) * totitem_remain; - - /* used as but->func_argN so freeing is handled elsewhere */ - EnumPropertyItem *remaining = MEM_mallocN(array_size + sizeof(EnumPropertyItem), "pie_level_item_array"); - memcpy(remaining, items + totitem_parent, array_size); - /* a NULL terminating sentinal element is required */ - memset(&remaining[totitem_remain], 0, sizeof(EnumPropertyItem)); - - - /* yuk, static... issue is we can't reliably free this without doing dangerous changes */ - static PieMenuLevelData lvl; - BLI_strncpy(lvl.title, block->pie_data.title, UI_MAX_NAME_STR); - lvl.totitem = totitem_remain; - lvl.ot = ot; - lvl.propname = propname; - lvl.properties = properties; - lvl.context = context; - lvl.flag = flag; - - /* add a 'more' menu entry */ - uiBut *but = uiDefIconTextBut(block, UI_BTYPE_BUT, 0, ICON_PLUS, "More", 0, 0, UI_UNIT_X * 3, UI_UNIT_Y, NULL, - 0.0f, 0.0f, 0.0f, 0.0f, "Show more items of this menu"); - UI_but_funcN_set(but, ui_pie_menu_level_invoke, remaining, &lvl); -} - -/** \} */ /* Pie Menu Levels */ - - -/*************************** Standard Popup Menus ****************************/ - -void UI_popup_menu_reports(bContext *C, ReportList *reports) -{ - Report *report; - - uiPopupMenu *pup = NULL; - uiLayout *layout; - - if (!CTX_wm_window(C)) - return; - - for (report = reports->list.first; report; report = report->next) { - int icon; - const char *msg, *msg_next; - - if (report->type < reports->printlevel) { - continue; - } - - if (pup == NULL) { - char title[UI_MAX_DRAW_STR]; - BLI_snprintf(title, sizeof(title), "%s: %s", IFACE_("Report"), report->typestr); - /* popup_menu stuff does just what we need (but pass meaningful block name) */ - pup = UI_popup_menu_begin_ex(C, title, __func__, ICON_NONE); - layout = UI_popup_menu_layout(pup); - } - else { - uiItemS(layout); - } - - /* split each newline into a label */ - msg = report->message; - icon = UI_icon_from_report_type(report->type); - do { - char buf[UI_MAX_DRAW_STR]; - msg_next = strchr(msg, '\n'); - if (msg_next) { - msg_next++; - BLI_strncpy(buf, msg, MIN2(sizeof(buf), msg_next - msg)); - msg = buf; - } - uiItemL(layout, msg, icon); - icon = ICON_NONE; - } while ((msg = msg_next) && *msg); - } - - if (pup) { - UI_popup_menu_end(C, pup); - } -} - -int UI_popup_menu_invoke(bContext *C, const char *idname, ReportList *reports) -{ - uiPopupMenu *pup; - uiLayout *layout; - MenuType *mt = WM_menutype_find(idname, true); - - if (mt == NULL) { - BKE_reportf(reports, RPT_ERROR, "Menu \"%s\" not found", idname); - return OPERATOR_CANCELLED; - } - - if (mt->poll && mt->poll(C, mt) == 0) - /* cancel but allow event to pass through, just like operators do */ - return (OPERATOR_CANCELLED | OPERATOR_PASS_THROUGH); - - pup = UI_popup_menu_begin(C, IFACE_(mt->label), ICON_NONE); - layout = UI_popup_menu_layout(pup); - - UI_menutype_draw(C, mt, layout); - - UI_popup_menu_end(C, pup); - - return OPERATOR_INTERFACE; -} - - -/*************************** Popup Block API **************************/ - -void UI_popup_block_invoke_ex(bContext *C, uiBlockCreateFunc func, void *arg, const char *opname, int opcontext) -{ - wmWindow *window = CTX_wm_window(C); - uiPopupBlockHandle *handle; - - handle = ui_popup_block_create(C, NULL, NULL, func, NULL, arg); - handle->popup = true; - handle->optype = (opname) ? WM_operatortype_find(opname, 0) : NULL; - handle->opcontext = opcontext; - - UI_popup_handlers_add(C, &window->modalhandlers, handle, 0); - WM_event_add_mousemove(C); -} - -void UI_popup_block_invoke(bContext *C, uiBlockCreateFunc func, void *arg) -{ - UI_popup_block_invoke_ex(C, func, arg, NULL, WM_OP_INVOKE_DEFAULT); -} - -void UI_popup_block_ex(bContext *C, uiBlockCreateFunc func, uiBlockHandleFunc popup_func, uiBlockCancelFunc cancel_func, void *arg, wmOperator *op) -{ - wmWindow *window = CTX_wm_window(C); - uiPopupBlockHandle *handle; - - handle = ui_popup_block_create(C, NULL, NULL, func, NULL, arg); - handle->popup = true; - handle->retvalue = 1; - - handle->popup_op = op; - handle->popup_arg = arg; - handle->popup_func = popup_func; - handle->cancel_func = cancel_func; - // handle->opcontext = opcontext; - - UI_popup_handlers_add(C, &window->modalhandlers, handle, 0); - WM_event_add_mousemove(C); -} - -#if 0 /* UNUSED */ -void uiPupBlockOperator(bContext *C, uiBlockCreateFunc func, wmOperator *op, int opcontext) -{ - wmWindow *window = CTX_wm_window(C); - uiPopupBlockHandle *handle; - - handle = ui_popup_block_create(C, NULL, NULL, func, NULL, op); - handle->popup = 1; - handle->retvalue = 1; - - handle->popup_arg = op; - handle->popup_func = operator_cb; - handle->cancel_func = confirm_cancel_operator; - handle->opcontext = opcontext; - - UI_popup_handlers_add(C, &window->modalhandlers, handle, 0); - WM_event_add_mousemove(C); -} -#endif - -void UI_popup_block_close(bContext *C, wmWindow *win, uiBlock *block) -{ - /* if loading new .blend while popup is open, window will be NULL */ - if (block->handle) { - if (win) { - UI_popup_handlers_remove(&win->modalhandlers, block->handle); - ui_popup_block_free(C, block->handle); - - /* In the case we have nested popups, closing one may need to redraw another, see: T48874 */ - for (ARegion *ar = win->screen->regionbase.first; ar; ar = ar->next) { - ED_region_tag_refresh_ui(ar); - } - } - } -} - -ColorPicker *ui_block_colorpicker_create(struct uiBlock *block) -{ - ColorPicker *cpicker = MEM_callocN(sizeof(ColorPicker), "color_picker"); - BLI_addhead(&block->color_pickers.list, cpicker); - - return cpicker; -} - -void ui_rgb_to_color_picker_compat_v(const float rgb[3], float r_cp[3]) -{ - switch (U.color_picker_type) { - case USER_CP_CIRCLE_HSL: - rgb_to_hsl_compat_v(rgb, r_cp); - break; - default: - rgb_to_hsv_compat_v(rgb, r_cp); - break; - } -} - -void ui_rgb_to_color_picker_v(const float rgb[3], float r_cp[3]) -{ - switch (U.color_picker_type) { - case USER_CP_CIRCLE_HSL: - rgb_to_hsl_v(rgb, r_cp); - break; - default: - rgb_to_hsv_v(rgb, r_cp); - break; - } -} - -void ui_color_picker_to_rgb_v(const float r_cp[3], float rgb[3]) -{ - switch (U.color_picker_type) { - case USER_CP_CIRCLE_HSL: - hsl_to_rgb_v(r_cp, rgb); - break; - default: - hsv_to_rgb_v(r_cp, rgb); - break; - } -} - -void ui_color_picker_to_rgb(float r_cp0, float r_cp1, float r_cp2, float *r, float *g, float *b) -{ - switch (U.color_picker_type) { - case USER_CP_CIRCLE_HSL: - hsl_to_rgb(r_cp0, r_cp1, r_cp2, r, g, b); - break; - default: - hsv_to_rgb(r_cp0, r_cp1, r_cp2, r, g, b); - break; - } -} -- cgit v1.2.3