Blender V5.0
MOD_ui_common.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2023 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
8
9#include "BLI_listbase.h"
10#include "BLI_string_utf8.h"
11
12#include "MEM_guardedalloc.h"
13
14#include "BKE_context.hh"
15#include "BKE_library.hh"
16#include "BKE_modifier.hh"
17#include "BKE_screen.hh"
18
19#include "DNA_object_types.h"
20#include "DNA_particle_types.h"
21#include "DNA_screen_types.h"
22
23#include "ED_object.hh"
24
25#include "BLT_translation.hh"
26
27#include "UI_interface.hh"
29#include "UI_resources.hh"
30
31#include "RNA_access.hh"
32#include "RNA_prototypes.hh"
33
34#include "WM_api.hh"
35#include "WM_types.hh"
36
37#include "MOD_ui_common.hh" /* Self include */
38
40
45static bool modifier_ui_poll(const bContext *C, PanelType * /*pt*/)
46{
48 return ob != nullptr;
49}
50
51/* -------------------------------------------------------------------- */
54
58static void modifier_reorder(bContext *C, Panel *panel, int new_index)
59{
60 PointerRNA *md_ptr = UI_panel_custom_data_get(panel);
61 ModifierData *md = (ModifierData *)md_ptr->data;
62
63 PointerRNA props_ptr;
64 wmOperatorType *ot = WM_operatortype_find("OBJECT_OT_modifier_move_to_index", false);
66 RNA_string_set(&props_ptr, "modifier", md->name);
67 RNA_int_set(&props_ptr, "index", new_index);
70}
71
72static short get_modifier_expand_flag(const bContext * /*C*/, Panel *panel)
73{
74 PointerRNA *md_ptr = UI_panel_custom_data_get(panel);
75 ModifierData *md = (ModifierData *)md_ptr->data;
76 return md->ui_expand_flag;
77}
78
79static void set_modifier_expand_flag(const bContext * /*C*/, Panel *panel, short expand_flag)
80{
81 PointerRNA *md_ptr = UI_panel_custom_data_get(panel);
82 ModifierData *md = (ModifierData *)md_ptr->data;
83 md->ui_expand_flag = expand_flag;
84}
85
87
88/* -------------------------------------------------------------------- */
91
93{
94 ModifierData *md = static_cast<ModifierData *>(ptr->data);
95 if (md->error) {
96 uiLayout *row = &layout->row(false);
97 row->label(RPT_(md->error), ICON_ERROR);
98 }
99}
100
107#define ERROR_LIBDATA_MESSAGE N_("External library data")
109{
112 BLI_assert(RNA_struct_is_a(ptr->type, &RNA_Modifier));
113
114 if (r_ob_ptr != nullptr) {
115 *r_ob_ptr = RNA_pointer_create_discrete(ptr->owner_id, &RNA_Object, ptr->owner_id);
116 }
117
118 uiBlock *block = panel->layout->block();
120
121 UI_panel_context_pointer_set(panel, "modifier", ptr);
122
123 return ptr;
124}
125
128 PointerRNA *ob_ptr,
129 const StringRefNull vgroup_prop,
130 const std::optional<StringRefNull> invert_vgroup_prop,
131 const std::optional<StringRefNull> text)
132{
133 bool has_vertex_group = RNA_string_length(ptr, vgroup_prop.c_str()) != 0;
134
135 uiLayout *row = &layout->row(true);
136 row->prop_search(ptr, vgroup_prop, ob_ptr, "vertex_groups", text, ICON_GROUP_VERTEX);
137 if (invert_vgroup_prop) {
138 uiLayout *sub = &row->row(true);
139 sub->active_set(has_vertex_group);
140 sub->use_property_decorate_set(false);
141 sub->prop(ptr, *invert_vgroup_prop, UI_ITEM_NONE, "", ICON_ARROW_LEFTRIGHT);
142 }
143}
144
146{
147 uiLayout *layout = panel->layout;
148
150
151 layout->prop(ptr, "use_custom_curve", UI_ITEM_NONE, std::nullopt, ICON_NONE);
152}
153
155{
156 uiLayout *layout = panel->layout;
157
159
160 uiTemplateCurveMapping(layout, ptr, "curve", 0, false, false, false, false, false);
161}
162
168{
169 /* Physic Tab */
170 if (ELEM(md->type,
178 {
179 return 1;
180 }
181 /* Particle Tab */
183 return 2;
184 }
185
186 return 0;
187}
188
190{
191 /* fluid particle modifier can't be deleted here */
193 short particle_type = ((ParticleSystemModifierData *)md)->psys->part->type;
194 if (ELEM(particle_type,
205 {
206 return false;
207 }
208 }
209 return true;
210}
211
212static void modifier_ops_extra_draw(bContext *C, uiLayout *layout, void *md_v)
213{
214 PointerRNA op_ptr;
215 ModifierData *md = (ModifierData *)md_v;
216
218 PointerRNA ptr = RNA_pointer_create_discrete(&ob->id, &RNA_Modifier, md);
219 layout->context_ptr_set("modifier", &ptr);
221
222 layout->ui_units_x_set(4.0f);
223
224 /* Apply. */
225 if (ob->type == OB_GREASE_PENCIL) {
226 layout->op("OBJECT_OT_modifier_apply",
227 CTX_IFACE_(BLT_I18NCONTEXT_OPERATOR_DEFAULT, "Apply (Active Keyframe)"),
228 ICON_CHECKMARK);
229
230 op_ptr = layout->op("OBJECT_OT_modifier_apply",
231 IFACE_("Apply (All Keyframes)"),
232 ICON_KEYFRAME,
235 RNA_boolean_set(&op_ptr, "all_keyframes", true);
236 }
237 else {
238 layout->op("OBJECT_OT_modifier_apply",
240 ICON_CHECKMARK);
241 }
242
243 /* Apply as shapekey. */
245 PointerRNA op_ptr = layout->op(
246 "OBJECT_OT_modifier_apply_as_shapekey",
247 CTX_IFACE_(BLT_I18NCONTEXT_OPERATOR_DEFAULT, "Apply as Shape Key"),
248 ICON_SHAPEKEY_DATA);
249 RNA_boolean_set(&op_ptr, "keep_modifier", false);
250
251 op_ptr = layout->op("OBJECT_OT_modifier_apply_as_shapekey",
252 CTX_IFACE_(BLT_I18NCONTEXT_OPERATOR_DEFAULT, "Save as Shape Key"),
253 ICON_NONE);
254 RNA_boolean_set(&op_ptr, "keep_modifier", true);
255 layout->separator();
256 }
257
258 /* Duplicate. */
259 if (!ELEM(md->type,
265 {
266 layout->op("OBJECT_OT_modifier_copy",
268 ICON_DUPLICATE);
269 }
270
271 layout->op("OBJECT_OT_modifier_copy_to_selected",
272 CTX_IFACE_(BLT_I18NCONTEXT_OPERATOR_DEFAULT, "Copy to Selected"),
273 0);
274
275 layout->separator();
276
277 /* Move to first. */
278 op_ptr = layout->op("OBJECT_OT_modifier_move_to_index",
279 IFACE_("Move to First"),
280 ICON_TRIA_UP,
283 RNA_int_set(&op_ptr, "index", 0);
284
285 /* Move to last. */
286 op_ptr = layout->op("OBJECT_OT_modifier_move_to_index",
287 IFACE_("Move to Last"),
288 ICON_TRIA_DOWN,
291 RNA_int_set(&op_ptr, "index", BLI_listbase_count(&ob->modifiers) - 1);
292
293 layout->separator();
294
295 layout->prop(&ptr, "use_pin_to_last", UI_ITEM_NONE, std::nullopt, ICON_NONE);
296
297 if (md->type == eModifierType_Nodes) {
298 layout->separator();
299 op_ptr = layout->op("OBJECT_OT_geometry_nodes_move_to_nodes",
300 std::nullopt,
301 ICON_NONE,
304 layout->prop(&ptr, "show_group_selector", UI_ITEM_NONE, std::nullopt, ICON_NONE);
305 layout->prop(&ptr, "show_manage_panel", UI_ITEM_NONE, std::nullopt, ICON_NONE);
306 }
307}
308
309static void modifier_panel_header(const bContext *C, Panel *panel)
310{
311 uiLayout *row, *sub, *name_row;
312 uiLayout *layout = panel->layout;
313
314 /* Don't use #modifier_panel_get_property_pointers, we don't want to lock the header. */
316 ModifierData *md = (ModifierData *)ptr->data;
317 Object *ob = (Object *)ptr->owner_id;
318
319 UI_panel_context_pointer_set(panel, "modifier", ptr);
320
322 Scene *scene = CTX_data_scene(C);
323 int index = BLI_findindex(&ob->modifiers, md);
324
325 /* Modifier Icon. */
326 sub = &layout->row(true);
328 if (mti->is_disabled && mti->is_disabled(scene, md, false)) {
329 sub->red_alert_set(true);
330 }
331 PointerRNA op_ptr = sub->op("OBJECT_OT_modifier_set_active", "", RNA_struct_ui_icon(ptr->type));
332 RNA_string_set(&op_ptr, "modifier", md->name);
333
334 row = &layout->row(true);
335
336 /* Modifier Name.
337 * Count how many buttons are added to the header to check if there is enough space. */
338 int buttons_number = 0;
339 name_row = &row->row(true);
340
341 /* Display mode switching buttons. */
342 if (ob->type == OB_MESH) {
343 int last_cage_index;
344 int cage_index = BKE_modifiers_get_cage_index(scene, ob, &last_cage_index, false);
345 if (BKE_modifier_supports_cage(scene, md) && (index <= last_cage_index)) {
346 sub = &row->row(true);
347 if (index < cage_index || !BKE_modifier_couldbe_cage(scene, md)) {
348 sub->active_set(false);
349 }
350 sub->prop(ptr, "show_on_cage", UI_ITEM_NONE, "", ICON_NONE);
351 buttons_number++;
352 }
353 } /* Tessellation point for curve-typed objects. */
354 else if (ELEM(ob->type, OB_CURVES_LEGACY, OB_SURF, OB_FONT)) {
355 /* Smooth modifier can work with tessellated curves only (works on mesh edges explicitly). */
356 if (md->type == eModifierType_Smooth) {
357 /* Add button (appearing to be OFF) and add tip why this can't be changed. */
358 sub = &row->row(true);
359 uiBlock *block = sub->block();
360 static int apply_on_spline_always_off_hack = 0;
361 uiBut *but = uiDefIconButBitI(block,
364 0,
365 ICON_SURFACE_DATA,
366 0,
367 0,
368 UI_UNIT_X - 2,
369 UI_UNIT_Y,
370 &apply_on_spline_always_off_hack,
371 0.0,
372 0.0,
373 RPT_("Apply on Spline"));
374 UI_but_disable(but,
375 "This modifier can only deform filled curve/surface, not the control points");
376 buttons_number++;
377 }
378 /* Some modifiers can work with pre-tessellated curves only. */
380 {
381 /* Add button (appearing to be ON) and add tip why this can't be changed. */
382 sub = &row->row(true);
383 uiBlock *block = sub->block();
384 static int apply_on_spline_always_on_hack = eModifierMode_ApplyOnSpline;
385 uiBut *but = uiDefIconButBitI(block,
388 0,
389 ICON_SURFACE_DATA,
390 0,
391 0,
392 UI_UNIT_X - 2,
393 UI_UNIT_Y,
394 &apply_on_spline_always_on_hack,
395 0.0,
396 0.0,
397 RPT_("Apply on Spline"));
398 UI_but_disable(but,
399 "This modifier can only deform control points, not the filled curve/surface");
400 buttons_number++;
401 }
402 else if (mti->type != ModifierTypeType::Constructive) {
403 /* Constructive modifiers tessellates curve before applying. */
404 row->prop(ptr, "use_apply_on_spline", UI_ITEM_NONE, "", ICON_NONE);
405 buttons_number++;
406 }
407 }
408 /* Collision and Surface are always enabled, hide buttons. */
411 sub = &row->row(true);
413 sub->prop(ptr, "show_in_editmode", UI_ITEM_NONE, "", ICON_NONE);
414 buttons_number++;
415 }
416 row->prop(ptr, "show_viewport", UI_ITEM_NONE, "", ICON_NONE);
417 row->prop(ptr, "show_render", UI_ITEM_NONE, "", ICON_NONE);
418 buttons_number += 2;
419 }
420
421 /* Extra operators menu. */
422 row->menu_fn("", ICON_DOWNARROW_HLT, modifier_ops_extra_draw, md);
423
424 /* Delete button. */
426 sub = &row->row(false);
428 sub->op("OBJECT_OT_modifier_remove", "", ICON_X);
429 buttons_number++;
430 }
431
432 /* Switch context buttons. */
433 if (modifier_is_simulation(md) == 1) {
434 PointerRNA op_ptr = row->op("WM_OT_properties_context_change", "", ICON_PROPERTIES);
435 if (!RNA_pointer_is_null(&op_ptr)) {
436 RNA_string_set(&op_ptr, "context", "PHYSICS");
437 }
438 buttons_number++;
439 }
440 else if (modifier_is_simulation(md) == 2) {
441 PointerRNA op_ptr = row->op("WM_OT_properties_context_change", "", ICON_PROPERTIES);
442 if (!RNA_pointer_is_null(&op_ptr)) {
443 RNA_string_set(&op_ptr, "context", "PARTICLES");
444 }
445 buttons_number++;
446 }
447
448 bool display_name = (panel->sizex / UI_UNIT_X - buttons_number > 5) || (panel->sizex == 0);
449 if (display_name) {
450 name_row->prop(ptr, "name", UI_ITEM_NONE, "", ICON_NONE);
451 }
452 else {
454 }
455
456 /* Extra padding for delete button. */
457 layout->separator();
458}
459
461
462/* -------------------------------------------------------------------- */
465
467{
468 PanelType *panel_type = MEM_callocN<PanelType>(__func__);
469
470 BKE_modifier_type_panel_id(type, panel_type->idname);
471 STRNCPY_UTF8(panel_type->label, "");
472 STRNCPY_UTF8(panel_type->context, "modifier");
474 STRNCPY_UTF8(panel_type->active_property, "is_active");
475 STRNCPY_UTF8(panel_type->pin_to_last_property, "use_pin_to_last");
476
478 panel_type->draw = draw;
479 panel_type->poll = modifier_ui_poll;
480
481 /* Give the panel the special flag that says it was built here and corresponds to a
482 * modifier rather than a #PanelType. */
484 panel_type->reorder = modifier_reorder;
487
488 BLI_addtail(&region_type->paneltypes, panel_type);
489
490 return panel_type;
491}
492
494 const char *name,
495 const char *label,
496 PanelDrawFn draw_header,
497 PanelDrawFn draw,
498 PanelType *parent)
499{
500 PanelType *panel_type = MEM_callocN<PanelType>(__func__);
501
502 BLI_assert(parent != nullptr);
503 SNPRINTF_UTF8(panel_type->idname, "%s_%s", parent->idname, name);
504 STRNCPY_UTF8(panel_type->label, label);
505 STRNCPY_UTF8(panel_type->context, "modifier");
507 STRNCPY_UTF8(panel_type->active_property, "is_active");
508
509 panel_type->draw_header = draw_header;
510 panel_type->draw = draw;
511 panel_type->poll = modifier_ui_poll;
512 panel_type->flag = PANEL_TYPE_DEFAULT_CLOSED;
513
514 STRNCPY_UTF8(panel_type->parent_id, parent->idname);
515 panel_type->parent = parent;
516 BLI_addtail(&parent->children, BLI_genericNodeN(panel_type));
517 BLI_addtail(&region_type->paneltypes, panel_type);
518
519 return panel_type;
520}
521
Scene * CTX_data_scene(const bContext *C)
bool BKE_modifier_is_non_geometrical(ModifierData *md)
int BKE_modifiers_get_cage_index(const Scene *scene, Object *ob, int *r_lastPossibleCageIndex, bool is_virtual)
const ModifierTypeInfo * BKE_modifier_get_info(ModifierType type)
@ eModifierTypeFlag_SupportsEditmode
bool BKE_modifier_supports_cage(Scene *scene, ModifierData *md)
bool BKE_modifier_is_same_topology(ModifierData *md)
void BKE_modifier_type_panel_id(ModifierType type, char *r_idname)
bool BKE_modifier_couldbe_cage(Scene *scene, ModifierData *md)
@ PANEL_TYPE_INSTANCED
@ PANEL_TYPE_DEFAULT_CLOSED
@ PANEL_TYPE_HEADER_EXPAND
#define BLI_assert(a)
Definition BLI_assert.h:46
int BLI_findindex(const ListBase *listbase, const void *vlink) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL(1)
Definition listbase.cc:586
LinkData * BLI_genericNodeN(void *data)
Definition listbase.cc:922
void BLI_addtail(ListBase *listbase, void *vlink) ATTR_NONNULL(1)
Definition listbase.cc:111
int BLI_listbase_count(const ListBase *listbase) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL(1)
Definition listbase.cc:524
#define SNPRINTF_UTF8(dst, format,...)
#define STRNCPY_UTF8(dst, src)
#define ELEM(...)
#define RPT_(msgid)
#define CTX_IFACE_(context, msgid)
#define BLT_I18NCONTEXT_OPERATOR_DEFAULT
#define IFACE_(msgid)
#define BLT_I18NCONTEXT_DEFAULT_BPYRNA
#define ID_IS_EDITABLE(_id)
Definition DNA_ID.h:705
@ eModifierMode_ApplyOnSpline
@ eModifierMode_Realtime
@ eModifierType_ParticleSystem
@ eModifierType_MeshDeform
@ eModifierType_Fluidsim
@ eModifierType_Surface
@ eModifierType_Cloth
@ eModifierType_Fluid
@ eModifierType_Hook
@ eModifierType_Nodes
@ eModifierType_Collision
@ eModifierType_Smooth
@ eModifierType_DynamicPaint
@ eModifierType_Softbody
Object is a sort of wrapper for general info.
@ OB_SURF
@ OB_FONT
@ OB_GREASE_PENCIL
@ OB_MESH
@ OB_CURVES_LEGACY
@ PART_FLUID_FLIP
@ PART_FLUID_BUBBLE
@ PART_FLUID_SPRAYBUBBLE
@ PART_FLUID_TRACER
@ PART_FLUID
@ PART_FLUID_FOAM
@ PART_FLUID_SPRAYFOAMBUBBLE
@ PART_FLUID_SPRAYFOAM
@ PART_FLUID_SPRAY
@ PART_FLUID_FOAMBUBBLE
Read Guarded memory(de)allocation.
void modifier_vgroup_ui(uiLayout *layout, PointerRNA *ptr, PointerRNA *ob_ptr, const StringRefNull vgroup_prop, const std::optional< StringRefNull > invert_vgroup_prop, const std::optional< StringRefNull > text)
static void modifier_reorder(bContext *C, Panel *panel, int new_index)
PanelType * modifier_subpanel_register(ARegionType *region_type, const char *name, const char *label, PanelDrawFn draw_header, PanelDrawFn draw, PanelType *parent)
static void modifier_panel_header(const bContext *C, Panel *panel)
static int modifier_is_simulation(const ModifierData *md)
void modifier_grease_pencil_curve_header_draw(const bContext *, Panel *panel)
static bool modifier_ui_poll(const bContext *C, PanelType *)
void modifier_grease_pencil_curve_panel_draw(const bContext *, Panel *panel)
PanelType * modifier_panel_register(ARegionType *region_type, ModifierType type, PanelDrawFn draw)
static bool modifier_can_delete(ModifierData *md)
static void set_modifier_expand_flag(const bContext *, Panel *panel, short expand_flag)
static short get_modifier_expand_flag(const bContext *, Panel *panel)
PointerRNA * modifier_panel_get_property_pointers(Panel *panel, PointerRNA *r_ob_ptr)
void modifier_error_message_draw(uiLayout *layout, PointerRNA *ptr)
static void modifier_ops_extra_draw(bContext *C, uiLayout *layout, void *md_v)
#define C
Definition RandGen.cpp:29
void UI_but_disable(uiBut *but, const char *disabled_hint)
#define UI_UNIT_Y
uiBut * uiDefIconButBitI(uiBlock *block, ButType type, int bit, int retval, int icon, int x, int y, short width, short height, int *poin, float min, float max, std::optional< blender::StringRef > tip)
void uiTemplateCurveMapping(uiLayout *layout, PointerRNA *ptr, blender::StringRefNull propname, int type, bool levels, bool brush, bool neg_slope, bool tone, bool presets)
#define UI_UNIT_X
PointerRNA * UI_panel_custom_data_get(const Panel *panel)
void UI_panel_context_pointer_set(Panel *panel, const char *name, PointerRNA *ptr)
void UI_block_lock_set(uiBlock *block, bool val, const char *lockstr)
#define UI_ITEM_NONE
constexpr const char * c_str() const
void(*)(const bContext *, Panel *) PanelDrawFn
#define ERROR_LIBDATA_MESSAGE
void * MEM_callocN(size_t len, const char *str)
Definition mallocn.cc:118
Object * context_active_object(const bContext *C)
const char * name
bool RNA_struct_is_a(const StructRNA *type, const StructRNA *srna)
void RNA_string_set(PointerRNA *ptr, const char *name, const char *value)
void RNA_boolean_set(PointerRNA *ptr, const char *name, bool value)
void RNA_int_set(PointerRNA *ptr, const char *name, int value)
int RNA_struct_ui_icon(const StructRNA *type)
int RNA_string_length(PointerRNA *ptr, const char *name)
bool RNA_pointer_is_null(const PointerRNA *ptr)
PointerRNA RNA_pointer_create_discrete(ID *id, StructRNA *type, void *data)
ListBase paneltypes
bool(* is_disabled)(const Scene *scene, ModifierData *md, bool use_render_params)
ModifierTypeFlag flags
ModifierTypeType type
ListBase modifiers
void(* reorder)(bContext *C, Panel *pa, int new_index)
void(* set_list_data_expand_flag)(const bContext *C, Panel *pa, short expand_flag)
void(* draw)(const bContext *C, Panel *panel)
char idname[BKE_ST_MAXNAME]
bool(* poll)(const bContext *C, PanelType *pt)
char pin_to_last_property[BKE_ST_MAXNAME]
char context[BKE_ST_MAXNAME]
char translation_context[BKE_ST_MAXNAME]
char active_property[BKE_ST_MAXNAME]
ListBase children
char label[BKE_ST_MAXNAME]
short(* get_list_data_expand_flag)(const bContext *C, Panel *pa)
char parent_id[BKE_ST_MAXNAME]
PanelType * parent
void(* draw_header)(const bContext *C, Panel *panel)
struct uiLayout * layout
void * data
Definition RNA_types.hh:53
void use_property_decorate_set(bool is_sep)
void alignment_set(blender::ui::LayoutAlign alignment)
uiBlock * block() const
void operator_context_set(blender::wm::OpCallContext opcontext)
void label(blender::StringRef name, int icon)
void red_alert_set(bool red_alert)
void prop_search(PointerRNA *ptr, PropertyRNA *prop, PointerRNA *searchptr, PropertyRNA *searchprop, PropertyRNA *item_searchpropname, std::optional< blender::StringRefNull > name, int icon, bool results_are_suggestions)
void active_set(bool active)
void separator(float factor=1.0f, LayoutSeparatorType type=LayoutSeparatorType::Auto)
void menu_fn(blender::StringRefNull name, int icon, uiMenuCreateFunc func, void *arg)
void context_ptr_set(blender::StringRef name, const PointerRNA *ptr)
void ui_units_x_set(float width)
uiLayout & row(bool align)
void emboss_set(blender::ui::EmbossType emboss)
PointerRNA op(wmOperatorType *ot, std::optional< blender::StringRef > name, int icon, blender::wm::OpCallContext context, eUI_Item_Flag flag)
void prop(PointerRNA *ptr, PropertyRNA *prop, int index, int value, eUI_Item_Flag flag, std::optional< blender::StringRef > name_opt, int icon, std::optional< blender::StringRef > placeholder=std::nullopt)
wmOperatorStatus WM_operator_name_call_ptr(bContext *C, wmOperatorType *ot, blender::wm::OpCallContext context, PointerRNA *properties, const wmEvent *event)
PointerRNA * ptr
Definition wm_files.cc:4238
wmOperatorType * ot
Definition wm_files.cc:4237
wmOperatorType * WM_operatortype_find(const char *idname, bool quiet)
void WM_operator_properties_create_ptr(PointerRNA *ptr, wmOperatorType *ot)
void WM_operator_properties_free(PointerRNA *ptr)