Blender V4.5
MOD_grease_pencil_hook.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_index_mask.hh"
10
11#include "BLT_translation.hh"
12
13#include "BLO_read_write.hh"
14
15#include "DNA_defaults.h"
16#include "DNA_modifier_types.h"
17#include "DNA_screen_types.h"
18
19#include "RNA_access.hh"
20
21#include "BKE_action.hh"
22#include "BKE_colortools.hh"
23#include "BKE_curves.hh"
24#include "BKE_geometry_set.hh"
25#include "BKE_grease_pencil.hh"
26#include "BKE_lib_query.hh"
27#include "BKE_modifier.hh"
28#include "BKE_object_types.hh"
29
30#include "UI_interface.hh"
31#include "UI_resources.hh"
32
34#include "MOD_modifiertypes.hh"
35#include "MOD_ui_common.hh"
36
37#include "RNA_prototypes.hh"
38
39namespace blender {
40
50
51static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
52{
53 const auto *gmd = reinterpret_cast<const GreasePencilHookModifierData *>(md);
54 auto *tgmd = reinterpret_cast<GreasePencilHookModifierData *>(target);
55
57 modifier::greasepencil::copy_influence_data(&gmd->influence, &tgmd->influence, flag);
58}
59
60static void free_data(ModifierData *md)
61{
62 auto *mmd = reinterpret_cast<GreasePencilHookModifierData *>(md);
63
65}
66
67static bool is_disabled(const Scene * /*scene*/, ModifierData *md, bool /*use_render_params*/)
68{
69 auto *mmd = reinterpret_cast<GreasePencilHookModifierData *>(md);
70
71 return (mmd->object == nullptr);
72}
73
75{
76 auto *mmd = reinterpret_cast<GreasePencilHookModifierData *>(md);
77 if (mmd->object != nullptr) {
78 if (mmd->subtarget[0]) {
80 ctx->node, mmd->object, mmd->subtarget, DEG_OB_COMP_BONE, "Hook Modifier");
81 }
82 DEG_add_object_relation(ctx->node, mmd->object, DEG_OB_COMP_TRANSFORM, "Hook Modifier");
83 }
84 DEG_add_object_relation(ctx->node, ctx->object, DEG_OB_COMP_TRANSFORM, "Hook Modifier");
85}
86
87static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
88{
89 auto *mmd = reinterpret_cast<GreasePencilHookModifierData *>(md);
90
91 modifier::greasepencil::foreach_influence_ID_link(&mmd->influence, ob, walk, user_data);
92
93 walk(user_data, ob, (ID **)&mmd->object, IDWALK_CB_NOP);
94}
95
96static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
97{
98 const auto *mmd = reinterpret_cast<const GreasePencilHookModifierData *>(md);
99
101 modifier::greasepencil::write_influence_data(writer, &mmd->influence);
102}
103
104static void blend_read(BlendDataReader *reader, ModifierData *md)
105{
106 auto *mmd = reinterpret_cast<GreasePencilHookModifierData *>(md);
107 modifier::greasepencil::read_influence_data(reader, &mmd->influence);
108}
109
110/* Calculate the factor of falloff. */
111static float hook_falloff(const float falloff,
112 const int falloff_type,
113 const float falloff_sq,
114 const float fac_orig,
115 const CurveMapping *curfalloff,
116 const float len_sq)
117{
118 BLI_assert(falloff_sq);
119 if (len_sq > falloff_sq) {
120 return 0.0f;
121 }
122 if (len_sq <= 0.0f) {
123 return fac_orig;
124 }
125 if (falloff_type == MOD_GREASE_PENCIL_HOOK_Falloff_Const) {
126 return fac_orig;
127 }
128 if (falloff_type == MOD_GREASE_PENCIL_HOOK_Falloff_InvSquare) {
129 /* Avoid sqrt below. */
130 return (1.0f - (len_sq / falloff_sq)) * fac_orig;
131 }
132
133 float fac = 1.0f - (math::sqrt(len_sq) / falloff);
134
135 switch (falloff_type) {
137 return BKE_curvemapping_evaluateF(curfalloff, 0, fac) * fac_orig;
139 return fac * fac * fac_orig;
141 return (3.0f * fac * fac - 2.0f * fac * fac * fac) * fac_orig;
143 return math::sqrt(fac) * fac_orig;
144 break;
146 return math::sqrt(2 * fac - fac * fac) * fac_orig;
148 ATTR_FALLTHROUGH; /* Pass. */
149 default:
150 return fac * fac_orig;
151 }
152}
153
154static void deform_drawing(const ModifierData &md,
155 const Object &ob,
157{
158 const auto &mmd = reinterpret_cast<const GreasePencilHookModifierData &>(md);
160 bke::CurvesGeometry &curves = drawing.strokes_for_write();
161
162 if (curves.is_empty()) {
163 return;
164 }
165 IndexMaskMemory memory;
167 &ob, curves, mmd.influence, memory);
168 if (strokes.is_empty()) {
169 return;
170 }
171
173 curves, mmd.influence);
174
175 const int falloff_type = mmd.falloff_type;
176 const float falloff = (mmd.falloff_type == eHook_Falloff_None) ? 0.0f : mmd.falloff;
177 const float falloff_sq = square_f(falloff);
178 const float fac_orig = mmd.force;
179 const bool use_falloff = falloff_sq != 0.0f;
180 const bool use_uniform = (mmd.flag & MOD_GREASE_PENCIL_HOOK_UNIFORM_SPACE) != 0;
181
182 const float3x3 mat_uniform = use_uniform ? float3x3(float4x4(mmd.parentinv)) :
184 const float3 cent = use_uniform ? math::transform_point(mat_uniform, float3(mmd.cent)) :
185 float3(mmd.cent);
186
187 float4x4 dmat;
188 /* Get world-space matrix of target, corrected for the space the verts are in. */
189 if (mmd.subtarget[0]) {
190 bPoseChannel *pchan = BKE_pose_channel_find_name(mmd.object->pose, mmd.subtarget);
191 if (pchan) {
192 /* Bone target if there's a matching pose-channel. */
193 dmat = mmd.object->object_to_world() * float4x4(pchan->pose_mat);
194 }
195 }
196 else {
197 /* Just object target. */
198 dmat = mmd.object->object_to_world();
199 }
200 float4x4 use_mat = ob.world_to_object() * dmat * float4x4(mmd.parentinv);
201
202 const OffsetIndices<int> points_by_curve = curves.points_by_curve();
203 MutableSpan<float3> positions = curves.positions_for_write();
204
205 strokes.foreach_index(blender::GrainSize(128), [&](const int stroke) {
206 const IndexRange points_range = points_by_curve[stroke].index_range();
207 for (const int point_i : points_range) {
208 const int point = point_i + points_by_curve[stroke].first();
209 const float weight = input_weights[point];
210 if (weight < 0.0f) {
211 continue;
212 }
213
214 float fac;
215 if (use_falloff) {
216 float len_sq;
217 if (use_uniform) {
218 const float3 co_uniform = math::transform_point(mat_uniform, positions[point]);
219 len_sq = math::distance(cent, co_uniform);
220 }
221 else {
222 len_sq = math::distance(cent, positions[point]);
223 }
224 fac = hook_falloff(
225 falloff, falloff_type, falloff_sq, fac_orig, mmd.influence.custom_curve, len_sq);
226 }
227 else {
228 fac = fac_orig;
229 }
230
231 if (fac != 0.0f) {
232 const float3 co_tmp = math::transform_point(use_mat, positions[point]);
233 positions[point] = math::interpolate(positions[point], co_tmp, fac * weight);
234 }
235 }
236 });
237
238 drawing.tag_positions_changed();
239}
240
242 const ModifierEvalContext *ctx,
243 bke::GeometrySet *geometry_set)
244{
245 const GreasePencilHookModifierData *mmd = reinterpret_cast<GreasePencilHookModifierData *>(md);
246
247 if (!geometry_set->has_grease_pencil()) {
248 return;
249 }
250
251 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
252
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_drawings_for_write(grease_pencil, layer_mask, current_frame);
260
262 deform_drawing(*md, *ctx->object, *drawing);
263 });
264}
265
266static void panel_draw(const bContext *C, Panel *panel)
267{
268 uiLayout *layout = panel->layout;
269
270 PointerRNA ob_ptr;
272
273 PointerRNA hook_object_ptr = RNA_pointer_get(ptr, "object");
274
275 uiLayoutSetPropSep(layout, true);
276
277 uiLayout *col = &layout->column(false);
278 col->prop(ptr, "object", UI_ITEM_NONE, std::nullopt, ICON_NONE);
279 if (!RNA_pointer_is_null(&hook_object_ptr) &&
280 RNA_enum_get(&hook_object_ptr, "type") == OB_ARMATURE)
281 {
282 PointerRNA hook_object_data_ptr = RNA_pointer_get(&hook_object_ptr, "data");
284 col, ptr, "subtarget", &hook_object_data_ptr, "bones", IFACE_("Bone"), ICON_NONE);
285 }
286
287 layout->prop(ptr, "strength", UI_ITEM_R_SLIDER, std::nullopt, ICON_NONE);
288
289 if (uiLayout *sub = layout->panel_prop(C, ptr, "open_falloff_panel", IFACE_("Falloff"))) {
290 uiLayoutSetPropSep(sub, true);
291
292 sub->prop(ptr, "falloff_type", UI_ITEM_NONE, IFACE_("Type"), ICON_NONE);
293
294 bool use_falloff = RNA_enum_get(ptr, "falloff_type") != eWarp_Falloff_None;
295
296 uiLayout *row = &sub->row(false);
297 uiLayoutSetActive(row, use_falloff);
298 row->prop(ptr, "falloff_radius", UI_ITEM_NONE, std::nullopt, ICON_NONE);
299
300 sub->prop(ptr, "use_falloff_uniform", UI_ITEM_NONE, std::nullopt, ICON_NONE);
301
302 if (RNA_enum_get(ptr, "falloff_type") == eWarp_Falloff_Curve) {
303 uiTemplateCurveMapping(sub, ptr, "custom_curve", 0, false, false, false, false);
304 }
305 }
306
307 if (uiLayout *influence_panel = layout->panel_prop(
308 C, ptr, "open_influence_panel", IFACE_("Influence")))
309 {
313 }
314
316}
317
322
323} // namespace blender
324
326 /*idname*/ "GreasePencilHookModifier",
327 /*name*/ N_("Hook"),
328 /*struct_name*/ "GreasePencilHookModifierData",
329 /*struct_size*/ sizeof(GreasePencilHookModifierData),
330 /*srna*/ &RNA_GreasePencilHookModifier,
332 /*flags*/
335 /*icon*/ ICON_HOOK,
336
337 /*copy_data*/ blender::copy_data,
338
339 /*deform_verts*/ nullptr,
340 /*deform_matrices*/ nullptr,
341 /*deform_verts_EM*/ nullptr,
342 /*deform_matrices_EM*/ nullptr,
343 /*modify_mesh*/ nullptr,
344 /*modify_geometry_set*/ blender::modify_geometry_set,
345
346 /*init_data*/ blender::init_data,
347 /*required_data_mask*/ nullptr,
348 /*free_data*/ blender::free_data,
349 /*is_disabled*/ blender::is_disabled,
350 /*update_depsgraph*/ blender::update_depsgraph,
351 /*depends_on_time*/ nullptr,
352 /*depends_on_normals*/ nullptr,
353 /*foreach_ID_link*/ blender::foreach_ID_link,
354 /*foreach_tex_link*/ nullptr,
355 /*free_runtime_data*/ nullptr,
356 /*panel_register*/ blender::panel_register,
357 /*blend_write*/ blender::blend_write,
358 /*blend_read*/ blender::blend_read,
359};
Blender kernel action and pose functionality.
bPoseChannel * BKE_pose_channel_find_name(const bPose *pose, const char *name)
float BKE_curvemapping_evaluateF(const CurveMapping *cumap, int cur, float value)
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 ATTR_FALLTHROUGH
MINLINE float square_f(float a)
#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_bone_relation(DepsNodeHandle *handle, Object *object, const char *bone_name, eDepsObjectComponentType component, const char *description)
void DEG_add_object_relation(DepsNodeHandle *node_handle, Object *object, eDepsObjectComponentType component, const char *description)
@ DEG_OB_COMP_TRANSFORM
@ DEG_OB_COMP_BONE
#define DNA_struct_default_get(struct_name)
@ eWarp_Falloff_Curve
@ eWarp_Falloff_None
@ MOD_GREASE_PENCIL_HOOK_UNIFORM_SPACE
@ eModifierType_GreasePencilHook
@ MOD_GREASE_PENCIL_HOOK_Falloff_Root
@ MOD_GREASE_PENCIL_HOOK_Falloff_InvSquare
@ MOD_GREASE_PENCIL_HOOK_Falloff_Const
@ MOD_GREASE_PENCIL_HOOK_Falloff_Linear
@ MOD_GREASE_PENCIL_HOOK_Falloff_Sphere
@ MOD_GREASE_PENCIL_HOOK_Falloff_Smooth
@ MOD_GREASE_PENCIL_HOOK_Falloff_Curve
@ MOD_GREASE_PENCIL_HOOK_Falloff_Sharp
@ eHook_Falloff_None
@ OB_ARMATURE
static bool is_disabled
ModifierTypeInfo modifierType_GreasePencilHook
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
void uiTemplateCurveMapping(uiLayout *layout, PointerRNA *ptr, blender::StringRefNull propname, int type, bool levels, bool brush, bool neg_slope, bool tone)
@ UI_ITEM_R_SLIDER
void uiLayoutSetActive(uiLayout *layout, bool active)
void uiItemPointerR(uiLayout *layout, PointerRNA *ptr, blender::StringRefNull propname, PointerRNA *searchptr, blender::StringRefNull searchpropname, std::optional< blender::StringRefNull > name, int icon)
void uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
#define UI_ITEM_NONE
MutableSpan< float3 > positions_for_write()
OffsetIndices< int > points_by_curve() const
bke::CurvesGeometry & strokes_for_write()
void foreach_index(Fn &&fn) const
uint col
T sqrt(const T &a)
T distance(const T &a, const T &b)
T interpolate(const T &a, const T &b, const FactorT &t)
VecBase< T, 3 > transform_point(const CartesianBasis &basis, const VecBase< T, 3 > &v)
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)
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)
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 ensure_no_bezier_curves(Drawing &drawing)
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 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 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)
MatBase< float, 3, 3 > float3x3
static float hook_falloff(const float falloff, const int falloff_type, const float falloff_sq, const float fac_orig, const CurveMapping *curfalloff, const float len_sq)
static void update_depsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
VecBase< float, 3 > float3
static bool is_disabled(const Scene *, ModifierData *md, bool)
static void blend_read(BlendDataReader *reader, ModifierData *md)
PointerRNA RNA_pointer_get(PointerRNA *ptr, const char *name)
bool RNA_pointer_is_null(const PointerRNA *ptr)
int RNA_enum_get(PointerRNA *ptr, const char *name)
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:404
struct uiLayout * layout
float pose_mat[4][4]
GreasePencil * get_grease_pencil_for_write()
PanelLayout panel_prop(const bContext *C, PointerRNA *open_prop_owner, blender::StringRefNull open_prop_name)
uiLayout & column(bool align)
uiLayout & row(bool align)
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:4227
uint8_t flag
Definition wm_window.cc:139