Blender V4.3
MOD_mirror.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2005 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
9#include "BLI_span.hh"
10
11#include "BLT_translation.hh"
12
13#include "DNA_defaults.h"
14#include "DNA_mesh_types.h"
15#include "DNA_screen_types.h"
16
17#include "BKE_lib_id.hh"
18#include "BKE_lib_query.hh"
19#include "BKE_mesh_mirror.hh"
20#include "BKE_modifier.hh"
21
22#include "UI_interface.hh"
23#include "UI_resources.hh"
24
25#include "RNA_access.hh"
26#include "RNA_prototypes.hh"
27
28#include "MEM_guardedalloc.h"
29
31
32#include "MOD_ui_common.hh"
33
35
36using namespace blender;
37
46
47static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
48{
50
51 walk(user_data, ob, (ID **)&mmd->mirror_ob, IDWALK_CB_NOP);
52}
53
55{
57 if (mmd->mirror_ob != nullptr) {
58 DEG_add_object_relation(ctx->node, mmd->mirror_ob, DEG_OB_COMP_TRANSFORM, "Mirror Modifier");
59 DEG_add_depends_on_transform_relation(ctx->node, "Mirror Modifier");
60 }
61}
62
64 Object *ob,
65 Mesh *mesh,
66 const int axis,
67 const bool use_correct_order_on_merge)
68{
69 int *vert_merge_map = nullptr;
70 int vert_merge_map_len;
71 Mesh *result = mesh;
73 mmd, ob, result, axis, use_correct_order_on_merge, &vert_merge_map, &vert_merge_map_len);
74
75 if (vert_merge_map) {
76 /* Slow - so only call if one or more merge verts are found,
77 * users may leave this on and not realize there is nothing to merge - campbell */
78
79 /* TODO(mano-wii): Polygons with all vertices merged are the ones that form duplicates.
80 * Therefore the duplicate face test can be skipped. */
81 if (vert_merge_map_len) {
82 Mesh *tmp = result;
83 result = geometry::mesh_merge_verts(
84 *tmp, MutableSpan<int>{vert_merge_map, result->verts_num}, vert_merge_map_len, false);
85 BKE_id_free(nullptr, tmp);
86 }
87 MEM_freeN(vert_merge_map);
88 }
89
90 return result;
91}
92
94{
95 Mesh *result = mesh;
96 const bool use_correct_order_on_merge = mmd->use_correct_order_on_merge;
97
98 /* check which axes have been toggled and mirror accordingly */
99 if (mmd->flag & MOD_MIR_AXIS_X) {
100 result = mirror_apply_on_axis(mmd, ob, result, 0, use_correct_order_on_merge);
101 }
102 if (mmd->flag & MOD_MIR_AXIS_Y) {
103 Mesh *tmp = result;
104 result = mirror_apply_on_axis(mmd, ob, result, 1, use_correct_order_on_merge);
105 if (tmp != mesh) {
106 /* free intermediate results */
107 BKE_id_free(nullptr, tmp);
108 }
109 }
110 if (mmd->flag & MOD_MIR_AXIS_Z) {
111 Mesh *tmp = result;
112 result = mirror_apply_on_axis(mmd, ob, result, 2, use_correct_order_on_merge);
113 if (tmp != mesh) {
114 /* free intermediate results */
115 BKE_id_free(nullptr, tmp);
116 }
117 }
118
119 return result;
120}
121
122static Mesh *modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
123{
124 Mesh *result;
126
127 result = mirrorModifier__doMirror(mmd, ctx->object, mesh);
128
129 return result;
130}
131
132static void panel_draw(const bContext * /*C*/, Panel *panel)
133{
134 uiLayout *row, *col, *sub;
135 uiLayout *layout = panel->layout;
137
138 PropertyRNA *prop;
139 PointerRNA ob_ptr;
142 bool has_bisect = (mmd->flag &
144
145 col = uiLayoutColumn(layout, false);
146 uiLayoutSetPropSep(col, true);
147
148 prop = RNA_struct_find_property(ptr, "use_axis");
149 row = uiLayoutRowWithHeading(col, true, IFACE_("Axis"));
150 uiItemFullR(row, ptr, prop, 0, 0, toggles_flag, IFACE_("X"), ICON_NONE);
151 uiItemFullR(row, ptr, prop, 1, 0, toggles_flag, IFACE_("Y"), ICON_NONE);
152 uiItemFullR(row, ptr, prop, 2, 0, toggles_flag, IFACE_("Z"), ICON_NONE);
153
154 prop = RNA_struct_find_property(ptr, "use_bisect_axis");
155 row = uiLayoutRowWithHeading(col, true, IFACE_("Bisect"));
156 uiItemFullR(row, ptr, prop, 0, 0, toggles_flag, IFACE_("X"), ICON_NONE);
157 uiItemFullR(row, ptr, prop, 1, 0, toggles_flag, IFACE_("Y"), ICON_NONE);
158 uiItemFullR(row, ptr, prop, 2, 0, toggles_flag, IFACE_("Z"), ICON_NONE);
159
160 prop = RNA_struct_find_property(ptr, "use_bisect_flip_axis");
161 row = uiLayoutRowWithHeading(col, true, IFACE_("Flip"));
162 uiLayoutSetActive(row, has_bisect);
163 uiItemFullR(row, ptr, prop, 0, 0, toggles_flag, IFACE_("X"), ICON_NONE);
164 uiItemFullR(row, ptr, prop, 1, 0, toggles_flag, IFACE_("Y"), ICON_NONE);
165 uiItemFullR(row, ptr, prop, 2, 0, toggles_flag, IFACE_("Z"), ICON_NONE);
166
167 uiItemS(col);
168
169 uiItemR(col, ptr, "mirror_object", UI_ITEM_NONE, nullptr, ICON_NONE);
170
171 uiItemR(col,
172 ptr,
173 "use_clip",
176 ICON_NONE);
177
178 row = uiLayoutRowWithHeading(col, true, IFACE_("Merge"));
179 uiItemR(row, ptr, "use_mirror_merge", UI_ITEM_NONE, "", ICON_NONE);
180 sub = uiLayoutRow(row, true);
181 uiLayoutSetActive(sub, RNA_boolean_get(ptr, "use_mirror_merge"));
182 uiItemR(sub, ptr, "merge_threshold", UI_ITEM_NONE, "", ICON_NONE);
183
184 sub = uiLayoutRow(col, true);
185 uiLayoutSetActive(sub, has_bisect);
186 uiItemR(sub, ptr, "bisect_threshold", UI_ITEM_NONE, IFACE_("Bisect Distance"), ICON_NONE);
187
188 modifier_panel_end(layout, ptr);
189}
190
191static void data_panel_draw(const bContext * /*C*/, Panel *panel)
192{
193 uiLayout *col, *row, *sub;
194 uiLayout *layout = panel->layout;
195
197
198 uiLayoutSetPropSep(layout, true);
199
200 col = uiLayoutColumn(layout, true);
201 row = uiLayoutRowWithHeading(col, true, IFACE_("Mirror U"));
202 uiLayoutSetPropDecorate(row, false);
203 sub = uiLayoutRow(row, true);
204 uiItemR(sub, ptr, "use_mirror_u", UI_ITEM_NONE, "", ICON_NONE);
205 sub = uiLayoutRow(sub, true);
206 uiLayoutSetActive(sub, RNA_boolean_get(ptr, "use_mirror_u"));
207 uiItemR(sub, ptr, "mirror_offset_u", UI_ITEM_R_SLIDER, "", ICON_NONE);
208 uiItemDecoratorR(row, ptr, "mirror_offset_u", 0);
209
210 row = uiLayoutRowWithHeading(col, true, IFACE_("V"));
211 uiLayoutSetPropDecorate(row, false);
212 sub = uiLayoutRow(row, true);
213 uiItemR(sub, ptr, "use_mirror_v", UI_ITEM_NONE, "", ICON_NONE);
214 sub = uiLayoutRow(sub, true);
215 uiLayoutSetActive(sub, RNA_boolean_get(ptr, "use_mirror_v"));
216 uiItemR(sub, ptr, "mirror_offset_v", UI_ITEM_R_SLIDER, "", ICON_NONE);
217 uiItemDecoratorR(row, ptr, "mirror_offset_v", 0);
218
219 col = uiLayoutColumn(layout, true);
220 uiItemR(col, ptr, "offset_u", UI_ITEM_R_SLIDER, IFACE_("Offset U"), ICON_NONE);
221 uiItemR(col, ptr, "offset_v", UI_ITEM_R_SLIDER, IFACE_("V"), ICON_NONE);
222
223 uiItemR(
224 layout, ptr, "use_mirror_vertex_groups", UI_ITEM_NONE, IFACE_("Vertex Groups"), ICON_NONE);
225 uiItemR(layout, ptr, "use_mirror_udim", UI_ITEM_NONE, IFACE_("Flip UDIM"), ICON_NONE);
226}
227
228static void panel_register(ARegionType *region_type)
229{
231 modifier_subpanel_register(region_type, "data", "Data", nullptr, data_panel_draw, panel_type);
232}
233
235 /*idname*/ "Mirror",
236 /*name*/ N_("Mirror"),
237 /*struct_name*/ "MirrorModifierData",
238 /*struct_size*/ sizeof(MirrorModifierData),
239 /*srna*/ &RNA_MirrorModifier,
244 /*icon*/ ICON_MOD_MIRROR,
245
246 /*copy_data*/ BKE_modifier_copydata_generic,
247
248 /*deform_verts*/ nullptr,
249 /*deform_matrices*/ nullptr,
250 /*deform_verts_EM*/ nullptr,
251 /*deform_matrices_EM*/ nullptr,
252 /*modify_mesh*/ modify_mesh,
253 /*modify_geometry_set*/ nullptr,
254
255 /*init_data*/ init_data,
256 /*required_data_mask*/ nullptr,
257 /*free_data*/ nullptr,
258 /*is_disabled*/ nullptr,
259 /*update_depsgraph*/ update_depsgraph,
260 /*depends_on_time*/ nullptr,
261 /*depends_on_normals*/ nullptr,
262 /*foreach_ID_link*/ foreach_ID_link,
263 /*foreach_tex_link*/ nullptr,
264 /*free_runtime_data*/ nullptr,
265 /*panel_register*/ panel_register,
266 /*blend_write*/ nullptr,
267 /*blend_read*/ nullptr,
268 /*foreach_cache*/ nullptr,
269};
void BKE_id_free(Main *bmain, void *idv)
@ IDWALK_CB_NOP
Mesh * BKE_mesh_mirror_apply_mirror_on_axis_for_modifier(MirrorModifierData *mmd, Object *ob, const Mesh *mesh, int axis, bool use_correct_order_on_merge, int **r_vert_merge_map, int *r_vert_merge_map_len)
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
@ eModifierTypeFlag_AcceptsCVs
@ eModifierTypeFlag_SupportsMapping
@ eModifierTypeFlag_EnableInEditmode
@ eModifierTypeFlag_SupportsEditmode
@ eModifierTypeFlag_AcceptsMesh
void(*)(void *user_data, Object *ob, ID **idpoin, int cb_flag) IDWalkFunc
#define BLI_assert(a)
Definition BLI_assert.h:50
#define MEMCMP_STRUCT_AFTER_IS_ZERO(struct_var, member)
#define MEMCPY_STRUCT_AFTER(struct_dst, struct_src, member)
#define CTX_IFACE_(context, msgid)
#define IFACE_(msgid)
#define BLT_I18NCONTEXT_ID_MESH
void DEG_add_depends_on_transform_relation(DepsNodeHandle *node_handle, const char *description)
void DEG_add_object_relation(DepsNodeHandle *node_handle, Object *object, eDepsObjectComponentType component, const char *description)
@ DEG_OB_COMP_TRANSFORM
#define DNA_struct_default_get(struct_name)
@ eModifierType_Mirror
struct MirrorModifierData MirrorModifierData
@ MOD_MIR_BISECT_AXIS_Z
@ MOD_MIR_AXIS_Z
@ MOD_MIR_BISECT_AXIS_X
@ MOD_MIR_AXIS_X
@ MOD_MIR_AXIS_Y
@ MOD_MIR_BISECT_AXIS_Y
Read Guarded memory(de)allocation.
static Mesh * mirror_apply_on_axis(MirrorModifierData *mmd, Object *ob, Mesh *mesh, const int axis, const bool use_correct_order_on_merge)
Definition MOD_mirror.cc:63
ModifierTypeInfo modifierType_Mirror
static Mesh * modify_mesh(ModifierData *md, const ModifierEvalContext *ctx, Mesh *mesh)
static Mesh * mirrorModifier__doMirror(MirrorModifierData *mmd, Object *ob, Mesh *mesh)
Definition MOD_mirror.cc:93
static void data_panel_draw(const bContext *, Panel *panel)
PanelType * modifier_subpanel_register(ARegionType *region_type, const char *name, const char *label, PanelDrawFn draw_header, PanelDrawFn draw, PanelType *parent)
void modifier_panel_end(uiLayout *layout, PointerRNA *ptr)
PanelType * modifier_panel_register(ARegionType *region_type, ModifierType type, PanelDrawFn draw)
PointerRNA * modifier_panel_get_property_pointers(Panel *panel, PointerRNA *r_ob_ptr)
void uiLayoutSetActive(uiLayout *layout, bool active)
uiLayout * uiLayoutRow(uiLayout *layout, bool align)
void uiItemFullR(uiLayout *layout, PointerRNA *ptr, PropertyRNA *prop, int index, int value, eUI_Item_Flag flag, const char *name, int icon, const char *placeholder=nullptr)
void uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
void uiItemS(uiLayout *layout)
uiLayout * uiLayoutRowWithHeading(uiLayout *layout, bool align, const char *heading)
#define UI_ITEM_NONE
void uiItemDecoratorR(uiLayout *layout, PointerRNA *ptr, const char *propname, int index)
void uiLayoutSetPropDecorate(uiLayout *layout, bool is_sep)
uiLayout * uiLayoutColumn(uiLayout *layout, bool align)
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
eUI_Item_Flag
@ UI_ITEM_R_TOGGLE
@ UI_ITEM_R_FORCE_BLANK_DECORATE
@ UI_ITEM_R_SLIDER
uint col
void MEM_freeN(void *vmemh)
Definition mallocn.cc:105
static void init_data(ModifierData *md)
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
static void panel_draw(const bContext *C, Panel *panel)
static void panel_register(ARegionType *region_type)
static void update_depsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
PropertyRNA * RNA_struct_find_property(PointerRNA *ptr, const char *identifier)
bool RNA_boolean_get(PointerRNA *ptr, const char *name)
Definition DNA_ID.h:413
struct Object * mirror_ob
struct uiLayout * layout
void * data
Definition RNA_types.hh:42
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4126