Blender V5.0
MOD_grease_pencil_mirror.cc
Go to the documentation of this file.
1/* SPDX-FileCopyrightText: 2024 Blender Authors
2 *
3 * SPDX-License-Identifier: GPL-2.0-or-later */
4
8
9#include "BLI_math_matrix.hh"
10
11#include "DNA_defaults.h"
12#include "DNA_modifier_types.h"
13
14#include "BKE_curves.hh"
15#include "BKE_geometry_set.hh"
16#include "BKE_grease_pencil.hh"
17#include "BKE_instances.hh"
18#include "BKE_lib_query.hh"
19#include "BKE_modifier.hh"
20
21#include "BLO_read_write.hh"
22
24
26#include "UI_resources.hh"
27
28#include "BLT_translation.hh"
29
30#include "WM_types.hh"
31
32#include "RNA_prototypes.hh"
33
35#include "MOD_ui_common.hh"
36
37namespace blender {
38
48
49static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
50{
51 const auto *mmd = reinterpret_cast<const GreasePencilMirrorModifierData *>(md);
52 auto *tmmd = reinterpret_cast<GreasePencilMirrorModifierData *>(target);
53
55
57 modifier::greasepencil::copy_influence_data(&mmd->influence, &tmmd->influence, flag);
58}
59
60static void free_data(ModifierData *md)
61{
62 auto *mmd = reinterpret_cast<GreasePencilMirrorModifierData *>(md);
64}
65
66static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
67{
68 auto *mmd = reinterpret_cast<GreasePencilMirrorModifierData *>(md);
69 walk(user_data, ob, (ID **)&mmd->object, IDWALK_CB_NOP);
70 modifier::greasepencil::foreach_influence_ID_link(&mmd->influence, ob, walk, user_data);
71}
72
74{
75 auto *mmd = reinterpret_cast<GreasePencilMirrorModifierData *>(md);
76 if (mmd->object != nullptr) {
78 ctx->node, mmd->object, DEG_OB_COMP_TRANSFORM, "Grease Pencil Mirror Modifier");
79 DEG_add_depends_on_transform_relation(ctx->node, "Grease Pencil Mirror Modifier");
80 }
81}
82
85 const bool mirror_x,
86 const bool mirror_y,
87 const bool mirror_z)
88{
90 float3(mirror_x ? -1.0f : 1.0f, mirror_y ? -1.0f : 1.0f, mirror_z ? -1.0f : 1.0f));
91
92 if (mmd.object) {
93 /* Transforms from parent object space to target object space. */
94 const float4x4 to_target = math::invert(mmd.object->object_to_world()) * ob.object_to_world();
95 /* Mirror points in the target object space. */
96 matrix = math::invert(to_target) * matrix * to_target;
97 }
98 return matrix;
99}
100
103 const bke::CurvesGeometry &base_curves,
104 const bke::CurvesGeometry &mirror_curves)
105{
106 const bool use_mirror_x = (mmd.flag & MOD_GREASE_PENCIL_MIRROR_AXIS_X);
107 const bool use_mirror_y = (mmd.flag & MOD_GREASE_PENCIL_MIRROR_AXIS_Y);
108 const bool use_mirror_z = (mmd.flag & MOD_GREASE_PENCIL_MIRROR_AXIS_Z);
109
110 Curves *base_curves_id = bke::curves_new_nomain(base_curves);
111 Curves *mirror_curves_id = bke::curves_new_nomain(mirror_curves);
112 bke::GeometrySet base_geo = bke::GeometrySet::from_curves(base_curves_id);
113 bke::GeometrySet mirror_geo = bke::GeometrySet::from_curves(mirror_curves_id);
114
115 std::unique_ptr<bke::Instances> instances = std::make_unique<bke::Instances>();
116 const int base_handle = instances->add_reference(bke::InstanceReference{base_geo});
117 const int mirror_handle = instances->add_reference(bke::InstanceReference{mirror_geo});
118 for (const int mirror_x : IndexRange(use_mirror_x ? 2 : 1)) {
119 for (const int mirror_y : IndexRange(use_mirror_y ? 2 : 1)) {
120 for (const int mirror_z : IndexRange(use_mirror_z ? 2 : 1)) {
121 if (mirror_x == 0 && mirror_y == 0 && mirror_z == 0) {
122 instances->add_instance(base_handle, float4x4::identity());
123 }
124 else {
125 const float4x4 matrix = get_mirror_matrix(
126 ob, mmd, bool(mirror_x), bool(mirror_y), bool(mirror_z));
127 instances->add_instance(mirror_handle, matrix);
128 }
129 }
130 }
131 }
132
134 options.keep_original_ids = true;
135 options.realize_instance_attributes = false;
137 bke::GeometrySet::from_instances(instances.release()), options)
138 .geometry;
139 return std::move(result_geo.get_curves_for_write()->geometry.wrap());
140}
141
143 const ModifierEvalContext &ctx,
145{
146 const bool use_mirror_x = (mmd.flag & MOD_GREASE_PENCIL_MIRROR_AXIS_X);
147 const bool use_mirror_y = (mmd.flag & MOD_GREASE_PENCIL_MIRROR_AXIS_Y);
148 const bool use_mirror_z = (mmd.flag & MOD_GREASE_PENCIL_MIRROR_AXIS_Z);
149 if (!use_mirror_x && !use_mirror_y && !use_mirror_z) {
150 return;
151 }
152
153 const bke::CurvesGeometry &src_curves = drawing.strokes();
154 if (src_curves.curve_num == 0) {
155 return;
156 }
157 /* Selected source curves. */
158 IndexMaskMemory curve_mask_memory;
160 ctx.object, src_curves, mmd.influence, curve_mask_memory);
161
162 if (curves_mask.size() == src_curves.curve_num) {
163 /* All geometry gets mirrored. */
164 drawing.strokes_for_write() = create_mirror_copies(*ctx.object, mmd, src_curves, src_curves);
165 }
166 else {
167 /* Create masked geometry, then mirror it. */
169 src_curves, curves_mask, {});
170
172 *ctx.object, mmd, src_curves, masked_curves);
173 }
174
175 drawing.tag_topology_changed();
176}
177
179 const ModifierEvalContext *ctx,
180 bke::GeometrySet *geometry_set)
181{
183
184 auto *mmd = reinterpret_cast<GreasePencilMirrorModifierData *>(md);
185
186 if (!geometry_set->has_grease_pencil()) {
187 return;
188 }
189 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
190 const int frame = grease_pencil.runtime->eval_frame;
191
192 IndexMaskMemory mask_memory;
194 grease_pencil, mmd->influence, mask_memory);
195
197 grease_pencil, layer_mask, frame);
199 [&](Drawing *drawing) { modify_drawing(*mmd, *ctx, *drawing); });
200}
201
202static void panel_draw(const bContext *C, Panel *panel)
203{
204 uiLayout *layout = panel->layout;
205
206 PointerRNA ob_ptr;
209
210 layout->use_property_split_set(true);
211
212 uiLayout *row = &layout->row(true, IFACE_("Axis"));
213 row->prop(ptr, "use_axis_x", toggles_flag, std::nullopt, ICON_NONE);
214 row->prop(ptr, "use_axis_y", toggles_flag, std::nullopt, ICON_NONE);
215 row->prop(ptr, "use_axis_z", toggles_flag, std::nullopt, ICON_NONE);
216
217 layout->prop(ptr, "object", UI_ITEM_NONE, std::nullopt, ICON_NONE);
218
219 if (uiLayout *influence_panel = layout->panel_prop(
220 C, ptr, "open_influence_panel", IFACE_("Influence")))
221 {
224 }
225
227}
228
233
234static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
235{
236 const auto *mmd = reinterpret_cast<const GreasePencilMirrorModifierData *>(md);
237
239 modifier::greasepencil::write_influence_data(writer, &mmd->influence);
240}
241
242static void blend_read(BlendDataReader *reader, ModifierData *md)
243{
244 auto *mmd = reinterpret_cast<GreasePencilMirrorModifierData *>(md);
245
246 modifier::greasepencil::read_influence_data(reader, &mmd->influence);
247}
248
249} // namespace blender
250
252 /*idname*/ "GreasePencilMirror",
253 /*name*/ N_("Mirror"),
254 /*struct_name*/ "GreasePencilMirrorModifierData",
255 /*struct_size*/ sizeof(GreasePencilMirrorModifierData),
256 /*srna*/ &RNA_GreasePencilMirrorModifier,
260 /*icon*/ ICON_MOD_MIRROR,
261
262 /*copy_data*/ blender::copy_data,
263
264 /*deform_verts*/ nullptr,
265 /*deform_matrices*/ nullptr,
266 /*deform_verts_EM*/ nullptr,
267 /*deform_matrices_EM*/ nullptr,
268 /*modify_mesh*/ nullptr,
269 /*modify_geometry_set*/ blender::modify_geometry_set,
270
271 /*init_data*/ blender::init_data,
272 /*required_data_mask*/ nullptr,
273 /*free_data*/ blender::free_data,
274 /*is_disabled*/ nullptr,
275 /*update_depsgraph*/ blender::update_depsgraph,
276 /*depends_on_time*/ nullptr,
277 /*depends_on_normals*/ nullptr,
278 /*foreach_ID_link*/ blender::foreach_ID_link,
279 /*foreach_tex_link*/ nullptr,
280 /*free_runtime_data*/ nullptr,
281 /*panel_register*/ blender::panel_register,
282 /*blend_write*/ blender::blend_write,
283 /*blend_read*/ blender::blend_read,
284};
Low-level operations for curves.
Low-level operations for grease pencil.
@ IDWALK_CB_NOP
void(*)(void *user_data, Object *ob, ID **idpoin, LibraryForeachIDCallbackFlag cb_flag) IDWalkFunc
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
@ eModifierTypeFlag_SupportsMapping
@ eModifierTypeFlag_AcceptsGreasePencil
@ eModifierTypeFlag_EnableInEditmode
@ eModifierTypeFlag_SupportsEditmode
#define BLI_assert(a)
Definition BLI_assert.h:46
#define MEMCMP_STRUCT_AFTER_IS_ZERO(struct_var, member)
#define MEMCPY_STRUCT_AFTER(struct_dst, struct_src, member)
#define BLO_write_struct(writer, struct_name, data_ptr)
#define IFACE_(msgid)
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)
@ MOD_GREASE_PENCIL_MIRROR_AXIS_Z
@ MOD_GREASE_PENCIL_MIRROR_AXIS_X
@ MOD_GREASE_PENCIL_MIRROR_AXIS_Y
@ eModifierType_GreasePencilMirror
ModifierTypeInfo modifierType_GreasePencilMirror
PanelType * modifier_panel_register(ARegionType *region_type, ModifierType type, PanelDrawFn draw)
PointerRNA * modifier_panel_get_property_pointers(Panel *panel, PointerRNA *r_ob_ptr)
void modifier_error_message_draw(uiLayout *layout, PointerRNA *ptr)
#define C
Definition RandGen.cpp:29
@ UI_ITEM_R_TOGGLE
@ UI_ITEM_R_FORCE_BLANK_DECORATE
#define UI_ITEM_NONE
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
CCL_NAMESPACE_BEGIN struct Options options
CurvesGeometry curves_copy_curve_selection(const CurvesGeometry &curves, const IndexMask &curves_to_copy, const AttributeFilter &attribute_filter)
Curves * curves_new_nomain(int points_num, int curves_num)
RealizeInstancesResult realize_instances(bke::GeometrySet geometry_set, const RealizeInstancesOptions &options)
CartesianBasis invert(const CartesianBasis &basis)
MatT from_scale(const VecBase< typename MatT::base_type, ScaleDim > &scale)
void read_influence_data(BlendDataReader *reader, GreasePencilModifierInfluenceData *influence_data)
void init_influence_data(GreasePencilModifierInfluenceData *influence_data, const bool has_custom_curve)
static IndexMask get_filtered_stroke_mask(const Object *ob, const bke::CurvesGeometry &curves, const Material *material_filter, const std::optional< int > material_pass_filter, const bool material_filter_invert, const bool material_pass_filter_invert, IndexMaskMemory &memory)
void write_influence_data(BlendWriter *writer, const GreasePencilModifierInfluenceData *influence_data)
static IndexMask get_filtered_layer_mask(const GreasePencil &grease_pencil, const std::optional< StringRef > tree_node_name_filter, const std::optional< int > layer_pass_filter, const bool layer_filter_invert, const bool layer_pass_filter_invert, IndexMaskMemory &memory)
Vector< bke::greasepencil::Drawing * > get_drawings_for_write(GreasePencil &grease_pencil, const IndexMask &layer_mask, const int frame)
void draw_material_filter_settings(const bContext *, uiLayout *layout, PointerRNA *ptr)
void draw_layer_filter_settings(const bContext *, uiLayout *layout, PointerRNA *ptr)
void free_influence_data(GreasePencilModifierInfluenceData *influence_data)
void foreach_influence_ID_link(GreasePencilModifierInfluenceData *influence_data, Object *ob, IDWalkFunc walk, void *user_data)
void copy_influence_data(const GreasePencilModifierInfluenceData *influence_data_src, GreasePencilModifierInfluenceData *influence_data_dst, const int)
void parallel_for_each(Range &&range, const Function &function)
Definition BLI_task.hh:56
static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
static void blend_write(BlendWriter *writer, const ID *, const ModifierData *md)
static bke::CurvesGeometry create_mirror_copies(const Object &ob, const GreasePencilMirrorModifierData &mmd, const bke::CurvesGeometry &base_curves, const bke::CurvesGeometry &mirror_curves)
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)
MatBase< float, 4, 4 > float4x4
static void modify_geometry_set(ModifierData *md, const ModifierEvalContext *ctx, bke::GeometrySet *geometry_set)
static float4x4 get_mirror_matrix(const Object &ob, const GreasePencilMirrorModifierData &mmd, const bool mirror_x, const bool mirror_y, const bool mirror_z)
static void free_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
static void modify_drawing(const GreasePencilArrayModifierData &mmd, const ModifierEvalContext &ctx, bke::greasepencil::Drawing &drawing)
static void update_depsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
VecBase< float, 3 > float3
static void blend_read(BlendDataReader *reader, ModifierData *md)
CurvesGeometry geometry
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:414
struct uiLayout * layout
static GeometrySet from_instances(Instances *instances, GeometryOwnershipType ownership=GeometryOwnershipType::Owned)
static GeometrySet from_curves(Curves *curves, GeometryOwnershipType ownership=GeometryOwnershipType::Owned)
GreasePencil * get_grease_pencil_for_write()
PanelLayout panel_prop(const bContext *C, PointerRNA *open_prop_owner, blender::StringRefNull open_prop_name)
uiLayout & row(bool align)
void use_property_split_set(bool value)
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)
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4238
uint8_t flag
Definition wm_window.cc:145