Blender V5.0
MOD_grease_pencil_noise.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
8
9#include "BLI_hash.h"
10#include "BLI_math_vector.hh"
11
12#include "BLT_translation.hh"
13
14#include "BLO_read_write.hh"
15
16#include "DNA_defaults.h"
18#include "DNA_object_types.h"
19#include "DNA_screen_types.h"
20
21#include "BKE_colortools.hh"
22#include "BKE_curves.hh"
23#include "BKE_geometry_set.hh"
24#include "BKE_grease_pencil.hh"
25
27#include "UI_resources.hh"
28
30#include "MOD_ui_common.hh"
31
32#include "RNA_access.hh"
33#include "RNA_prototypes.hh"
34
35namespace blender {
36
46
53
54static void copy_data(const ModifierData *md, ModifierData *target, int flag)
55{
57 reinterpret_cast<const GreasePencilNoiseModifierData *>(md);
58 GreasePencilNoiseModifierData *tgmd = reinterpret_cast<GreasePencilNoiseModifierData *>(target);
59
62}
63
64static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
65{
67 reinterpret_cast<const GreasePencilNoiseModifierData *>(md);
68
71}
72
78
79static bool depends_on_time(Scene * /*scene*/, ModifierData *md)
80{
82 return (mmd->flag & GP_NOISE_USE_RANDOM) != 0;
83}
84
85static Array<float> noise_table(const int len, const int offset, const int seed)
86{
87 Array<float> table(len);
88 for (int i = 0; i < len; i++) {
89 table[i] = BLI_hash_int_01(BLI_hash_int_2d(seed, i + offset + 1));
90 }
91 return table;
92}
93
98 const Object &ob,
99 const int ctime,
100 const int start_frame_number,
102{
104 bke::CurvesGeometry &strokes = drawing.strokes_for_write();
106 if (strokes.is_empty()) {
107 return;
108 }
109
110 IndexMaskMemory memory;
112 &ob, strokes, mmd.influence, memory);
113
114 const bool use_curve = (mmd.influence.flag & GREASE_PENCIL_INFLUENCE_USE_CUSTOM_CURVE) != 0;
115 const bool is_keyframe = (mmd.noise_mode == GP_NOISE_RANDOM_KEYFRAME);
116
117 /* Sanitize as it can create out of bound reads. */
118 const float noise_scale = math::clamp(mmd.noise_scale, 0.0f, 1.0f);
119
120 if (filtered_strokes.is_empty()) {
121 return;
122 }
123
124 int seed = mmd.seed;
125 /* Make sure different modifiers get different seeds. */
126 seed += BLI_hash_string(ob.id.name + 2);
128 if (mmd.flag & GP_NOISE_USE_RANDOM) {
129 if (!is_keyframe) {
130 seed += math::floor(ctime / mmd.step);
131 }
132 else {
133 /* If change every keyframe, use the last keyframe. */
134 seed += start_frame_number;
135 }
136 }
137
138 const OffsetIndices<int> points_by_curve = strokes.points_by_curve();
140 strokes, mmd.influence);
141
142 auto get_weight = [&](const IndexRange points, const int point_i) {
143 const float vertex_weight = vgroup_weights[points[point_i]];
144 if (!use_curve) {
145 return vertex_weight;
146 }
147 const float value = float(point_i) / float(points.size() - 1);
148 return vertex_weight * BKE_curvemapping_evaluateF(mmd.influence.custom_curve, 0, value);
149 };
150
151 auto get_noise = [](const Span<float> noise_table, const float value) {
152 return math::interpolate(noise_table[int(math::ceil(value))],
153 noise_table[int(math::floor(value))],
154 math::fract(value));
155 };
156
157 if (mmd.factor > 0.0f) {
158 const Span<float3> curve_plane_normals = drawing.curve_plane_normals();
159 const Span<float3> tangents = strokes.evaluated_tangents();
160 MutableSpan<float3> positions = strokes.positions_for_write();
161
162 filtered_strokes.foreach_index(GrainSize(512), [&](const int stroke_i) {
163 const IndexRange points = points_by_curve[stroke_i];
164 const int noise_len = math::ceil(points.size() * noise_scale) + 2;
165 const Array<float> table = noise_table(
166 noise_len, int(math::floor(mmd.noise_offset)), seed + 2 + stroke_i);
167 for (const int i : points.index_range()) {
168 const int point = points[i];
169 float weight = get_weight(points, i);
170 /* Vector orthogonal to normal. */
171 const float3 bi_normal = math::normalize(
172 math::cross(tangents[point], curve_plane_normals[stroke_i]));
173 const float noise = get_noise(table, i * noise_scale + math::fract(mmd.noise_offset));
174 positions[point] += bi_normal * (noise * 2.0f - 1.0f) * weight * mmd.factor * 0.1f;
175 }
176 });
177 drawing.tag_positions_changed();
178 }
179
180 if (mmd.factor_thickness > 0.0f) {
181 MutableSpan<float> radii = drawing.radii_for_write();
182
183 filtered_strokes.foreach_index(GrainSize(512), [&](const int stroke_i) {
184 const IndexRange points = points_by_curve[stroke_i];
185 const int noise_len = math::ceil(points.size() * noise_scale) + 2;
186 const Array<float> table = noise_table(
187 noise_len, int(math::floor(mmd.noise_offset)), seed + stroke_i);
188 for (const int i : points.index_range()) {
189 const int point = points[i];
190 const float weight = get_weight(points, i);
191 const float noise = get_noise(table, i * noise_scale + math::fract(mmd.noise_offset));
192 radii[point] *= math::max(1.0f + (noise * 2.0f - 1.0f) * weight * mmd.factor_thickness,
193 0.0f);
194 }
195 });
196 }
197
198 if (mmd.factor_strength > 0.0f) {
199 MutableSpan<float> opacities = drawing.opacities_for_write();
200
201 filtered_strokes.foreach_index(GrainSize(512), [&](const int stroke_i) {
202 const IndexRange points = points_by_curve[stroke_i];
203 const int noise_len = math::ceil(points.size() * noise_scale) + 2;
204 const Array<float> table = noise_table(
205 noise_len, int(math::floor(mmd.noise_offset)), seed + 3 + stroke_i);
206 for (const int i : points.index_range()) {
207 const int point = points[i];
208 const float weight = get_weight(points, i);
209 const float noise = get_noise(table, i * noise_scale + math::fract(mmd.noise_offset));
210 opacities[point] *= math::max(1.0f - noise * weight * mmd.factor_strength, 0.0f);
211 }
212 });
213 }
214
215 if (mmd.factor_uvs > 0.0f) {
216 if (bke::SpanAttributeWriter<float> rotations = attributes.lookup_or_add_for_write_span<float>(
217 "rotation", bke::AttrDomain::Point))
218 {
219 filtered_strokes.foreach_index(GrainSize(512), [&](const int stroke_i) {
220 const IndexRange points = points_by_curve[stroke_i];
221 const int noise_len = math::ceil(points.size() * noise_scale) + 2;
222 const Array<float> table = noise_table(
223 noise_len, int(math::floor(mmd.noise_offset)), seed + 4 + stroke_i);
224 for (const int i : points.index_range()) {
225 const int point = points[i];
226 const float weight = get_weight(points, i);
227 const float noise = get_noise(table, i * noise_scale + math::fract(mmd.noise_offset));
228 const float delta_rot = (noise * 2.0f - 1.0f) * weight * mmd.factor_uvs * M_PI_2;
229 rotations.span[point] = math::clamp(
230 rotations.span[point] + delta_rot, float(-M_PI_2), float(M_PI_2));
231 }
232 });
233 rotations.finish();
234 }
235 }
236}
237
239 const ModifierEvalContext *ctx,
240 bke::GeometrySet *geometry_set)
241{
242 const auto *mmd = reinterpret_cast<GreasePencilNoiseModifierData *>(md);
243
244 if (!geometry_set->has_grease_pencil()) {
245 return;
246 }
247
248 if (!mmd->factor && !mmd->factor_strength && !mmd->factor_thickness && !mmd->factor_uvs) {
249 return;
250 }
251
252 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
253 const int current_frame = grease_pencil.runtime->eval_frame;
254
255 IndexMaskMemory mask_memory;
257 grease_pencil, mmd->influence, mask_memory);
259 modifier::greasepencil::get_drawing_infos_by_frame(grease_pencil, layer_mask, current_frame);
260
262 drawing_infos, [&](const modifier::greasepencil::FrameDrawingInfo &info) {
263 deform_drawing(*mmd, *ctx->object, current_frame, info.start_frame_number, *info.drawing);
264 });
265}
266
267static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
268{
270
272}
273
274static void panel_draw(const bContext *C, Panel *panel)
275{
276 uiLayout *col;
277 uiLayout *layout = panel->layout;
278
280
281 layout->use_property_split_set(true);
282
283 col = &layout->column(false);
284 col->prop(ptr, "factor", UI_ITEM_NONE, IFACE_("Position"), ICON_NONE);
285 col->prop(ptr,
286 "factor_strength",
289 ICON_NONE);
290 col->prop(ptr, "factor_thickness", UI_ITEM_NONE, IFACE_("Thickness"), ICON_NONE);
291 col->prop(ptr, "factor_uvs", UI_ITEM_NONE, IFACE_("UV"), ICON_NONE);
292 col->prop(ptr, "noise_scale", UI_ITEM_NONE, std::nullopt, ICON_NONE);
293 col->prop(ptr, "noise_offset", UI_ITEM_NONE, std::nullopt, ICON_NONE);
294 col->prop(ptr, "seed", UI_ITEM_NONE, std::nullopt, ICON_NONE);
295 PanelLayout random_panel_layout = layout->panel_prop_with_bool_header(
296 C, ptr, "open_random_panel", ptr, "use_random", IFACE_("Random"));
297 if (uiLayout *random_layout = random_panel_layout.body) {
298 uiLayout *random_col = &random_layout->column(false);
299 random_col->active_set(RNA_boolean_get(ptr, "use_random"));
300
301 random_col->prop(ptr, "random_mode", UI_ITEM_NONE, std::nullopt, ICON_NONE);
302 const int mode = RNA_enum_get(ptr, "random_mode");
303 if (mode != GP_NOISE_RANDOM_KEYFRAME) {
304 random_col->prop(ptr, "step", UI_ITEM_NONE, std::nullopt, ICON_NONE);
305 }
306 }
307
308 if (uiLayout *influence_panel = layout->panel_prop(
309 C, ptr, "open_influence_panel", IFACE_("Influence")))
310 {
315 }
316
318}
319
324
325} // namespace blender
326
328 /*idname*/ "GreasePencilNoiseModifier",
329 /*name*/ N_("Noise"),
330 /*struct_name*/ "GreasePencilNoiseModifierData",
331 /*struct_size*/ sizeof(GreasePencilNoiseModifierData),
332 /*srna*/ &RNA_GreasePencilNoiseModifier,
334 /*flags*/
337 /*icon*/ ICON_GREASEPENCIL,
338
339 /*copy_data*/ blender::copy_data,
340
341 /*deform_verts*/ nullptr,
342 /*deform_matrices*/ nullptr,
343 /*deform_verts_EM*/ nullptr,
344 /*deform_matrices_EM*/ nullptr,
345 /*modify_mesh*/ nullptr,
346 /*modify_geometry_set*/ blender::modify_geometry_set,
347
348 /*init_data*/ blender::init_data,
349 /*required_data_mask*/ nullptr,
350 /*free_data*/ blender::free_data,
351 /*is_disabled*/ nullptr,
352 /*update_depsgraph*/ nullptr,
353 /*depends_on_time*/ blender::depends_on_time,
354 /*depends_on_normals*/ nullptr,
355 /*foreach_ID_link*/ blender::foreach_ID_link,
356 /*foreach_tex_link*/ nullptr,
357 /*free_runtime_data*/ nullptr,
358 /*panel_register*/ blender::panel_register,
359 /*blend_write*/ blender::blend_write,
360 /*blend_read*/ blender::blend_read,
361};
float BKE_curvemapping_evaluateF(const CurveMapping *cumap, int cur, float value)
Low-level operations for curves.
Low-level operations for grease pencil.
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_AcceptsGreasePencil
@ eModifierTypeFlag_EnableInEditmode
@ eModifierTypeFlag_SupportsEditmode
#define BLI_assert(a)
Definition BLI_assert.h:46
BLI_INLINE float BLI_hash_int_01(unsigned int k)
Definition BLI_hash.h:92
BLI_INLINE unsigned int BLI_hash_string(const char *str)
Definition BLI_hash.h:67
BLI_INLINE unsigned int BLI_hash_int_2d(unsigned int kx, unsigned int ky)
Definition BLI_hash.h:51
#define M_PI_2
#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 BLT_I18NCONTEXT_ID_GPENCIL
#define CTX_IFACE_(context, msgid)
#define IFACE_(msgid)
#define DNA_struct_default_get(struct_name)
@ GP_NOISE_RANDOM_KEYFRAME
@ GP_NOISE_USE_RANDOM
@ GREASE_PENCIL_INFLUENCE_USE_CUSTOM_CURVE
@ eModifierType_GreasePencilNoise
Object is a sort of wrapper for general info.
ModifierTypeInfo modifierType_GreasePencilNoise
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
#define UI_ITEM_NONE
static unsigned long seed
Definition btSoftBody.h:39
constexpr int64_t size() const
constexpr IndexRange index_range() const
MutableSpan< float3 > positions_for_write()
OffsetIndices< int > points_by_curve() const
MutableAttributeAccessor attributes_for_write()
Span< float3 > evaluated_tangents() const
GSpanAttributeWriter lookup_or_add_for_write_span(StringRef attribute_id, AttrDomain domain, AttrType data_type, const AttributeInit &initializer=AttributeInitDefaultValue())
Span< float3 > curve_plane_normals() const
MutableSpan< float > opacities_for_write()
MutableSpan< float > radii_for_write()
bke::CurvesGeometry & strokes_for_write()
void foreach_index(Fn &&fn) const
nullptr float
uint col
T clamp(const T &a, const T &min, const T &max)
T floor(const T &a)
AxisSigned cross(const AxisSigned a, const AxisSigned b)
T interpolate(const T &a, const T &b, const FactorT &t)
T fract(const T &a)
MatBase< T, NumCol, NumRow > normalize(const MatBase< T, NumCol, NumRow > &a)
T ceil(const T &a)
T max(const T &a, const T &b)
void read_influence_data(BlendDataReader *reader, GreasePencilModifierInfluenceData *influence_data)
void init_influence_data(GreasePencilModifierInfluenceData *influence_data, const bool has_custom_curve)
Vector< FrameDrawingInfo > get_drawing_infos_by_frame(GreasePencil &grease_pencil, const IndexMask &layer_mask, const int frame)
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)
void draw_vertex_group_settings(const bContext *, uiLayout *layout, PointerRNA *ptr)
VArray< float > get_influence_vertex_weights(const bke::CurvesGeometry &curves, 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)
void draw_material_filter_settings(const bContext *, uiLayout *layout, PointerRNA *ptr)
void draw_layer_filter_settings(const bContext *, uiLayout *layout, PointerRNA *ptr)
void draw_custom_curve_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 ensure_no_bezier_curves(Drawing &drawing)
void parallel_for_each(Range &&range, const Function &function)
Definition BLI_task.hh:56
static bool depends_on_time(Scene *, ModifierData *md)
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)
static void modify_geometry_set(ModifierData *md, const ModifierEvalContext *ctx, bke::GeometrySet *geometry_set)
static void deform_drawing(const ModifierData &md, const Object &ob, bke::greasepencil::Drawing &drawing)
static void free_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
VecBase< float, 3 > float3
static void blend_read(BlendDataReader *reader, ModifierData *md)
static Array< float > noise_table(int len, int offset, int seed)
bool RNA_boolean_get(PointerRNA *ptr, const char *name)
int RNA_enum_get(PointerRNA *ptr, const char *name)
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:414
char name[258]
Definition DNA_ID.h:432
struct uiLayout * layout
GreasePencil * get_grease_pencil_for_write()
PanelLayout panel_prop(const bContext *C, PointerRNA *open_prop_owner, blender::StringRefNull open_prop_name)
PanelLayout panel_prop_with_bool_header(const bContext *C, PointerRNA *open_prop_owner, blender::StringRefNull open_prop_name, PointerRNA *bool_prop_owner, blender::StringRefNull bool_prop_name, std::optional< blender::StringRef > label)
uiLayout & column(bool align)
void active_set(bool active)
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)
i
Definition text_draw.cc:230
uint len
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4238
uint8_t flag
Definition wm_window.cc:145