Blender V4.3
MOD_grease_pencil_array.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
9#include "DNA_defaults.h"
10#include "DNA_modifier_types.h"
11
12#include "BKE_curves.hh"
13#include "BKE_geometry_set.hh"
14#include "BKE_grease_pencil.hh"
15#include "BKE_instances.hh"
16#include "BKE_lib_query.hh"
17#include "BKE_material.h"
18#include "BKE_modifier.hh"
19#include "BKE_screen.hh"
20
21#include "BLO_read_write.hh"
22
24
25#include "UI_interface.hh"
26#include "UI_resources.hh"
27
28#include "BLT_translation.hh"
29
30#include "BLI_bounds_types.hh"
31#include "BLI_hash.h"
32#include "BLI_math_matrix.hh"
33#include "BLI_rand.h"
34
35#include "WM_types.hh"
36
37#include "RNA_access.hh"
38#include "RNA_prototypes.hh"
39
41#include "MOD_modifiertypes.hh"
42#include "MOD_ui_common.hh"
43
44namespace blender {
45
46static void init_data(ModifierData *md)
47{
48 auto *mmd = reinterpret_cast<GreasePencilArrayModifierData *>(md);
49
51
53 modifier::greasepencil::init_influence_data(&mmd->influence, false);
54}
55
56static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
57{
58 const auto *mmd = reinterpret_cast<const GreasePencilArrayModifierData *>(md);
59 auto *tmmd = reinterpret_cast<GreasePencilArrayModifierData *>(target);
60
62
64 modifier::greasepencil::copy_influence_data(&mmd->influence, &tmmd->influence, flag);
65}
66
67static void free_data(ModifierData *md)
68{
69 auto *mmd = reinterpret_cast<GreasePencilArrayModifierData *>(md);
71}
72
73static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
74{
75 auto *mmd = reinterpret_cast<GreasePencilArrayModifierData *>(md);
76 walk(user_data, ob, (ID **)&mmd->object, IDWALK_CB_NOP);
77 modifier::greasepencil::foreach_influence_ID_link(&mmd->influence, ob, walk, user_data);
78}
79
81{
82 auto *mmd = reinterpret_cast<GreasePencilArrayModifierData *>(md);
83 if (mmd->object != nullptr) {
85 ctx->node, mmd->object, DEG_OB_COMP_TRANSFORM, "Grease Pencil Array Modifier");
86 DEG_add_depends_on_transform_relation(ctx->node, "Grease Pencil Array Modifier");
87 }
88}
89
92 const int elem_idx,
93 const bool use_object_offset)
94{
95
96 if (use_object_offset) {
97 float4x4 mat_offset = float4x4::identity();
98
100 mat_offset[3] += mmd.offset;
101 }
102 const float4x4 &obinv = ob.world_to_object();
103
104 return mat_offset * obinv * mmd.object->object_to_world();
105 }
106
107 const float3 offset = [&]() {
109 return float3(mmd.offset) * elem_idx;
110 }
111 return float3(0.0f);
112 }();
113
114 return math::from_location<float4x4>(offset);
115}
116
118 const Object &ob,
119 const int elem_id)
120{
121 int seed = mmd.seed;
122 seed += BLI_hash_string(ob.id.name + 2);
124 const float rand_offset = BLI_hash_int_01(seed);
125 float3x3 rand;
126 for (int j = 0; j < 3; j++) {
127 const uint3 primes(2, 3, 7);
128 double3 offset(0.0);
129 double3 r;
130 /* To ensure a nice distribution, we use halton sequence and offset using the seed. */
131 BLI_halton_3d(primes, offset, elem_id, r);
132
134 float rand_value;
135 rand_value = math::mod(r[0] * 2.0 - 1.0 + rand_offset, 1.0);
136 rand_value = math::mod(math::sin(rand_value * 12.9898 + j * 78.233) * 43758.5453, 1.0);
137 rand[j] = float3(rand_value);
138 }
139 else {
140 for (int i = 0; i < 3; i++) {
141 rand[j][i] = math::mod(r[i] * 2.0 - 1.0 + rand_offset, 1.0);
142 rand[j][i] = math::mod(math::sin(rand[j][i] * 12.9898 + j * 78.233) * 43758.5453, 1.0);
143 }
144 }
145 }
146 /* Calculate Random matrix. */
148 mmd.rnd_offset * rand[0], mmd.rnd_rot * rand[1], float3(1.0f) + mmd.rnd_scale * rand[2]);
149};
150
153 const bke::CurvesGeometry &base_curves,
154 bke::CurvesGeometry filtered_curves)
155{
156 /* Assign replacement material on filtered curves so all copies can have this material when later
157 * when they get instanced. */
158 if (mmd.mat_rpl > 0) {
159 bke::MutableAttributeAccessor attributes = filtered_curves.attributes_for_write();
160 bke::SpanAttributeWriter<int> stroke_materials = attributes.lookup_or_add_for_write_span<int>(
161 "material_index", bke::AttrDomain::Curve);
162 stroke_materials.span.fill(mmd.mat_rpl - 1);
163 stroke_materials.finish();
164 }
165
166 Curves *base_curves_id = bke::curves_new_nomain(base_curves);
167 Curves *filtered_curves_id = bke::curves_new_nomain(filtered_curves);
168 bke::GeometrySet base_geo = bke::GeometrySet::from_curves(base_curves_id);
169 bke::GeometrySet filtered_geo = bke::GeometrySet::from_curves(filtered_curves_id);
170
171 std::unique_ptr<bke::Instances> instances = std::make_unique<bke::Instances>();
172 const int base_handle = instances->add_reference(bke::InstanceReference{base_geo});
173 const int filtered_handle = instances->add_reference(bke::InstanceReference{filtered_geo});
174
175 /* Always add untouched original curves. */
176 instances->add_instance(base_handle, float4x4::identity());
177
178 float3 size(0.0f);
180 std::optional<blender::Bounds<float3>> bounds = filtered_curves.bounds_min_max();
181 if (bounds.has_value()) {
182 size = bounds.value().max - bounds.value().min;
183 /* Need a minimum size (for flat drawings). */
184 size = math::max(size, float3(0.01f));
185 }
186 }
187
188 float4x4 current_offset = float4x4::identity();
189 for (const int elem_id : IndexRange(1, mmd.count - 1)) {
190 const bool use_object_offset = (mmd.flag & MOD_GREASE_PENCIL_ARRAY_USE_OB_OFFSET) &&
191 (mmd.object);
192 const float4x4 mat = get_array_matrix(ob, mmd, elem_id, use_object_offset);
193
194 if (use_object_offset) {
195 current_offset = current_offset * mat;
196 }
197 else {
198 current_offset = mat;
199 }
200
201 /* Apply relative offset. */
203 float3 relative = size * float3(mmd.shift);
204 float3 translate = relative * float3(float(elem_id));
205 current_offset.w += float4(translate, 1.0f);
206 }
207
208 current_offset *= get_rand_matrix(mmd, ob, elem_id);
209
210 instances->add_instance(filtered_handle, current_offset);
211 }
212
214 options.keep_original_ids = true;
215 options.realize_instance_attributes = false; /* Should this be true? */
217 bke::GeometrySet::from_instances(instances.release()), options);
218 return std::move(result_geo.get_curves_for_write()->geometry.wrap());
219}
220
222 const ModifierEvalContext &ctx,
224{
225 const bke::CurvesGeometry &src_curves = drawing.strokes();
226 if (src_curves.curve_num == 0) {
227 return;
228 }
229
230 IndexMaskMemory curve_mask_memory;
232 ctx.object, src_curves, mmd.influence, curve_mask_memory);
233
234 if (curves_mask.size() == src_curves.curve_num) {
235 /* Make a full copy so we can modify materials inside #create_array_copies before instancing.
236 */
238
240 *ctx.object, mmd, src_curves, std::move(copy));
241 }
242 else {
244 src_curves, curves_mask, {});
245
247 *ctx.object, mmd, src_curves, std::move(masked_curves));
248 }
249
250 drawing.tag_topology_changed();
251}
252
254 const ModifierEvalContext *ctx,
255 bke::GeometrySet *geometry_set)
256{
258
259 auto *mmd = reinterpret_cast<GreasePencilArrayModifierData *>(md);
260
261 if (!geometry_set->has_grease_pencil()) {
262 return;
263 }
264 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
265 const int frame = grease_pencil.runtime->eval_frame;
266
267 IndexMaskMemory mask_memory;
269 grease_pencil, mmd->influence, mask_memory);
270
272 grease_pencil, layer_mask, frame);
274 [&](Drawing *drawing) { modify_drawing(*mmd, *ctx, *drawing); });
275}
276
277static void panel_draw(const bContext *C, Panel *panel)
278{
279 uiLayout *layout = panel->layout;
280
281 PointerRNA ob_ptr;
283
284 uiLayoutSetPropSep(layout, true);
285
286 uiItemR(layout, ptr, "count", UI_ITEM_NONE, nullptr, ICON_NONE);
287 uiItemR(layout, ptr, "replace_material", UI_ITEM_NONE, IFACE_("Material Override"), ICON_NONE);
288
289 if (uiLayout *sub = uiLayoutPanelProp(
290 C, layout, ptr, "open_relative_offset_panel", IFACE_("Relative Offset")))
291 {
292 uiLayoutSetPropSep(sub, true);
293 uiItemR(sub, ptr, "use_relative_offset", UI_ITEM_NONE, IFACE_("Enable"), ICON_NONE);
294
295 uiLayout *col = uiLayoutColumn(sub, false);
296 uiLayoutSetActive(col, RNA_boolean_get(ptr, "use_relative_offset"));
297 uiItemR(col, ptr, "relative_offset", UI_ITEM_NONE, IFACE_("Factor"), ICON_NONE);
298 }
299
300 if (uiLayout *sub = uiLayoutPanelProp(
301 C, layout, ptr, "open_constant_offset_panel", IFACE_("Constant Offset")))
302 {
303 uiLayoutSetPropSep(sub, true);
304 uiItemR(sub, ptr, "use_constant_offset", UI_ITEM_NONE, IFACE_("Enable"), ICON_NONE);
305
306 uiLayout *col = uiLayoutColumn(sub, false);
307 uiLayoutSetActive(col, RNA_boolean_get(ptr, "use_constant_offset"));
308 uiItemR(col, ptr, "constant_offset", UI_ITEM_NONE, IFACE_("Distance"), ICON_NONE);
309 }
310
311 if (uiLayout *sub = uiLayoutPanelProp(
312 C, layout, ptr, "open_object_offset_panel", IFACE_("Object Offset")))
313 {
314 uiLayoutSetPropSep(sub, true);
315 uiItemR(sub, ptr, "use_object_offset", UI_ITEM_NONE, IFACE_("Enable"), ICON_NONE);
316
317 uiLayout *col = uiLayoutColumn(sub, false);
318 uiLayoutSetActive(col, RNA_boolean_get(ptr, "use_object_offset"));
319 uiItemR(col, ptr, "offset_object", UI_ITEM_NONE, IFACE_("Object"), ICON_NONE);
320 }
321
322 if (uiLayout *sub = uiLayoutPanelProp(
323 C, layout, ptr, "open_randomize_panel", IFACE_("Randomize")))
324 {
325 uiLayoutSetPropSep(sub, true);
326 uiItemR(sub, ptr, "random_offset", UI_ITEM_NONE, IFACE_("Offset"), ICON_NONE);
327 uiItemR(sub, ptr, "random_rotation", UI_ITEM_NONE, IFACE_("Rotation"), ICON_NONE);
328 uiItemR(sub, ptr, "random_scale", UI_ITEM_NONE, IFACE_("Scale"), ICON_NONE);
329 uiItemR(sub, ptr, "use_uniform_random_scale", UI_ITEM_NONE, nullptr, ICON_NONE);
330 uiItemR(sub, ptr, "seed", UI_ITEM_NONE, nullptr, ICON_NONE);
331 }
332
333 if (uiLayout *influence_panel = uiLayoutPanelProp(
334 C, layout, ptr, "open_influence_panel", IFACE_("Influence")))
335 {
338 }
339
340 modifier_panel_end(layout, ptr);
341}
342
347
348static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
349{
350 const auto *mmd = reinterpret_cast<const GreasePencilArrayModifierData *>(md);
351
353 modifier::greasepencil::write_influence_data(writer, &mmd->influence);
354}
355
356static void blend_read(BlendDataReader *reader, ModifierData *md)
357{
358 auto *mmd = reinterpret_cast<GreasePencilArrayModifierData *>(md);
359
360 modifier::greasepencil::read_influence_data(reader, &mmd->influence);
361}
362
363} // namespace blender
364
366 /*idname*/ "GreasePencilArrayModifier",
367 /*name*/ N_("Array"),
368 /*struct_name*/ "GreasePencilArrayModifierData",
369 /*struct_size*/ sizeof(GreasePencilArrayModifierData),
370 /*srna*/ &RNA_GreasePencilArrayModifier,
374 /*icon*/ ICON_MOD_ARRAY,
375
376 /*copy_data*/ blender::copy_data,
377
378 /*deform_verts*/ nullptr,
379 /*deform_matrices*/ nullptr,
380 /*deform_verts_EM*/ nullptr,
381 /*deform_matrices_EM*/ nullptr,
382 /*modify_mesh*/ nullptr,
383 /*modify_geometry_set*/ blender::modify_geometry_set,
384
385 /*init_data*/ blender::init_data,
386 /*required_data_mask*/ nullptr,
387 /*free_data*/ blender::free_data,
388 /*is_disabled*/ nullptr,
389 /*update_depsgraph*/ blender::update_depsgraph,
390 /*depends_on_time*/ nullptr,
391 /*depends_on_normals*/ nullptr,
392 /*foreach_ID_link*/ blender::foreach_ID_link,
393 /*foreach_tex_link*/ nullptr,
394 /*free_runtime_data*/ nullptr,
395 /*panel_register*/ blender::panel_register,
396 /*blend_write*/ blender::blend_write,
397 /*blend_read*/ blender::blend_read,
398};
Low-level operations for curves.
Low-level operations for grease pencil.
@ IDWALK_CB_NOP
General operations, lookup, etc. for materials.
void BKE_modifier_copydata_generic(const ModifierData *md, ModifierData *md_dst, int flag)
@ eModifierTypeFlag_SupportsMapping
@ eModifierTypeFlag_AcceptsGreasePencil
@ eModifierTypeFlag_EnableInEditmode
@ eModifierTypeFlag_SupportsEditmode
void(*)(void *user_data, Object *ob, ID **idpoin, int cb_flag) IDWalkFunc
#define BLI_assert(a)
Definition BLI_assert.h:50
BLI_INLINE float BLI_hash_int_01(unsigned int k)
Definition BLI_hash.h:96
BLI_INLINE unsigned int BLI_hash_string(const char *str)
Definition BLI_hash.h:71
Random number functions.
void BLI_halton_3d(const unsigned int prime[3], double offset[3], int n, double *r)
Definition rand.cc:308
#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)
struct GreasePencilArrayModifierData GreasePencilArrayModifierData
@ MOD_GREASE_PENCIL_ARRAY_USE_OFFSET
@ MOD_GREASE_PENCIL_ARRAY_USE_RELATIVE
@ MOD_GREASE_PENCIL_ARRAY_USE_OB_OFFSET
@ MOD_GREASE_PENCIL_ARRAY_UNIFORM_RANDOM_SCALE
@ eModifierType_GreasePencilArray
ModifierTypeInfo modifierType_GreasePencilArray
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)
void uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
#define UI_ITEM_NONE
PanelLayout uiLayoutPanelProp(const bContext *C, uiLayout *layout, PointerRNA *open_prop_owner, const char *open_prop_name)
uiLayout * uiLayoutColumn(uiLayout *layout, bool align)
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
static DBVT_INLINE btScalar size(const btDbvtVolume &a)
Definition btDbvt.cpp:52
static unsigned long seed
Definition btSoftBody.h:39
MutableAttributeAccessor attributes_for_write()
std::optional< Bounds< float3 > > bounds_min_max() const
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
CCL_NAMESPACE_BEGIN struct Options options
uint col
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)
bke::GeometrySet realize_instances(bke::GeometrySet geometry_set, const RealizeInstancesOptions &options)
T sin(const AngleRadianBase< T > &a)
T max(const T &a, const T &b)
T mod(const T &a, const T &b)
MatT from_location(const typename MatT::loc_type &location)
MatT from_loc_rot_scale(const typename MatT::loc_type &location, const RotationT &rotation, 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_layer_mask(const GreasePencil &grease_pencil, const std::optional< StringRef > layer_name_filter, const std::optional< int > layer_pass_filter, const bool layer_filter_invert, const bool layer_pass_filter_invert, IndexMaskMemory &memory)
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)
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:58
static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
static void blend_write(BlendWriter *writer, const ID *, const ModifierData *md)
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)
VecBase< float, 4 > float4
static void modify_geometry_set(ModifierData *md, const ModifierEvalContext *ctx, bke::GeometrySet *geometry_set)
static void free_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
static float4x4 get_array_matrix(const Object &ob, const GreasePencilArrayModifierData &mmd, const int elem_idx, const bool use_object_offset)
static bke::CurvesGeometry create_array_copies(const Object &ob, const GreasePencilArrayModifierData &mmd, const bke::CurvesGeometry &base_curves, bke::CurvesGeometry filtered_curves)
static float4x4 get_rand_matrix(const GreasePencilArrayModifierData &mmd, const Object &ob, const int elem_id)
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)
static void copy(bNodeTree *dest_ntree, bNode *dest_node, const bNode *src_node)
bool RNA_boolean_get(PointerRNA *ptr, const char *name)
CurvesGeometry geometry
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:413
char name[66]
Definition DNA_ID.h:425
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()
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4126
uint8_t flag
Definition wm_window.cc:138