Blender V4.3
MOD_grease_pencil_weight_proximity.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_index_mask.hh"
10#include "BLI_math_matrix.hh"
11#include "BLI_string.h" /* For #STRNCPY. */
12
13#include "BLT_translation.hh"
14
15#include "BLO_read_write.hh"
16
17#include "DNA_defaults.h"
18#include "DNA_modifier_types.h"
19#include "DNA_screen_types.h"
20
21#include "RNA_access.hh"
22
23#include "BKE_curves.hh"
24#include "BKE_deform.hh"
25#include "BKE_geometry_set.hh"
26#include "BKE_grease_pencil.hh"
27#include "BKE_lib_query.hh"
28#include "BKE_modifier.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
41static void init_data(ModifierData *md)
42{
43 auto *gpmd = reinterpret_cast<GreasePencilWeightProximityModifierData *>(md);
44
46
49 modifier::greasepencil::init_influence_data(&gpmd->influence, false);
50}
51
52static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
53{
54 const auto *gmd = reinterpret_cast<const GreasePencilWeightProximityModifierData *>(md);
55 auto *tgmd = reinterpret_cast<GreasePencilWeightProximityModifierData *>(target);
56
58 modifier::greasepencil::copy_influence_data(&gmd->influence, &tgmd->influence, flag);
59}
60
61static void free_data(ModifierData *md)
62{
63 auto *mmd = reinterpret_cast<GreasePencilWeightProximityModifierData *>(md);
64
66}
67
68static bool is_disabled(const Scene * /*scene*/, ModifierData *md, bool /*use_render_params*/)
69{
70 auto *mmd = reinterpret_cast<GreasePencilWeightProximityModifierData *>(md);
71
72 return (mmd->target_vgname[0] == '\0' || mmd->object == nullptr);
73}
74
75static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
76{
78 reinterpret_cast<GreasePencilWeightProximityModifierData *>(md);
79
81
82 walk(user_data, ob, (ID **)&mmd->object, IDWALK_CB_NOP);
83}
84
86{
87 auto *mmd = reinterpret_cast<GreasePencilWeightProximityModifierData *>(md);
88 if (mmd->object != nullptr) {
90 ctx->node, mmd->object, DEG_OB_COMP_TRANSFORM, "Grease Pencil Proximity Modifier");
91 DEG_add_depends_on_transform_relation(ctx->node, "Grease Pencil Proximity Modifier");
92 }
93}
94
95static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
96{
98 reinterpret_cast<const GreasePencilWeightProximityModifierData *>(md);
99
102}
103
110
111static float get_distance_factor(float3 target_pos,
112 const float4x4 &obmat,
113 float3 pos,
114 const float dist_min,
115 const float dist_max)
116{
117 const float3 gvert = math::transform_point(obmat, pos);
118 const float dist = math::distance(target_pos, gvert);
119
120 if (dist > dist_max) {
121 return 1.0f;
122 }
123 if (dist <= dist_max && dist > dist_min) {
124 return 1.0f - ((dist_max - dist) / math::max((dist_max - dist_min), 0.0001f));
125 }
126 return 0.0f;
127}
128
129static int ensure_vertex_group(const StringRefNull name, ListBase &vertex_group_names)
130{
131 int def_nr = BKE_defgroup_name_index(&vertex_group_names, name);
132 if (def_nr < 0) {
133 bDeformGroup *defgroup = MEM_cnew<bDeformGroup>(__func__);
134 STRNCPY(defgroup->name, name.c_str());
135 BLI_addtail(&vertex_group_names, defgroup);
136 def_nr = BLI_listbase_count(&vertex_group_names) - 1;
137 BLI_assert(def_nr >= 0);
138 }
139 return def_nr;
140}
141
143 const ListBase &vertex_group_names)
144{
145 const int def_nr = BKE_defgroup_name_index(&vertex_group_names, name);
146 if (def_nr < 0) {
147 return false;
148 }
149 return true;
150}
151
153 const Object &ob,
155{
156 const auto &mmd = reinterpret_cast<const GreasePencilWeightProximityModifierData &>(md);
157 bke::CurvesGeometry &curves = drawing.strokes_for_write();
158 if (curves.points_num() == 0) {
159 return;
160 }
161 IndexMaskMemory memory;
163 &ob, curves, mmd.influence, memory);
164 if (strokes.is_empty()) {
165 return;
166 }
167
168 /* Make sure that the target vertex group is added to this drawing so we can write to it. */
169 ensure_vertex_group(mmd.target_vgname, curves.vertex_group_names);
170
171 bke::MutableAttributeAccessor attributes = curves.attributes_for_write();
172 bke::SpanAttributeWriter<float> dst_weights = attributes.lookup_for_write_span<float>(
173 mmd.target_vgname);
174
175 BLI_assert(!dst_weights.span.is_empty());
176
178 curves, mmd.influence);
179
180 const Span<float3> positions = curves.positions();
181 const float4x4 &obmat = ob.object_to_world();
182 const float3 target_pos = mmd.object->object_to_world().location();
183 const bool invert = (mmd.flag & MOD_GREASE_PENCIL_WEIGHT_PROXIMITY_INVERT_OUTPUT) != 0;
184 const bool do_multiply = (mmd.flag & MOD_GREASE_PENCIL_WEIGHT_PROXIMITY_MULTIPLY_DATA) != 0;
185
186 threading::parallel_for(positions.index_range(), 1024, [&](const IndexRange range) {
187 for (const int point_i : range) {
188 const float weight = vgroup_weights[point_i];
189 if (weight < 0.0f) {
190 continue;
191 }
192
193 float dist_fac = get_distance_factor(
194 target_pos, obmat, positions[point_i], mmd.dist_start, mmd.dist_end);
195
196 if (invert) {
197 dist_fac = 1.0f - dist_fac;
198 }
199
200 dst_weights.span[point_i] = do_multiply ? dst_weights.span[point_i] * dist_fac : dist_fac;
201
202 dst_weights.span[point_i] = math::clamp(
203 dst_weights.span[point_i],
207 math::max(mmd.min_weight, 1e-5f),
208 1.0f);
209 }
210 });
211
212 dst_weights.finish();
213}
214
216 const ModifierEvalContext *ctx,
217 bke::GeometrySet *geometry_set)
218{
220 reinterpret_cast<GreasePencilWeightProximityModifierData *>(md);
221
222 if (!geometry_set->has_grease_pencil()) {
223 return;
224 }
225
226 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
227
229 return;
230 }
231
232 const int current_frame = grease_pencil.runtime->eval_frame;
233
234 IndexMaskMemory mask_memory;
235 const IndexMask layer_mask = modifier::greasepencil::get_filtered_layer_mask(
236 grease_pencil, mmd->influence, mask_memory);
238 modifier::greasepencil::get_drawings_for_write(grease_pencil, layer_mask, current_frame);
239
240 threading::parallel_for_each(drawings, [&](bke::greasepencil::Drawing *drawing) {
241 write_weights_for_drawing(*md, *ctx->object, *drawing);
242 });
243}
244
245static void panel_draw(const bContext *C, Panel *panel)
246{
247 uiLayout *row, *sub;
248 uiLayout *layout = panel->layout;
249
250 PointerRNA ob_ptr;
252
253 uiLayoutSetPropSep(layout, true);
254 row = uiLayoutRow(layout, true);
255 uiItemPointerR(row, ptr, "target_vertex_group", &ob_ptr, "vertex_groups", nullptr, ICON_NONE);
256 sub = uiLayoutRow(row, true);
257 bool has_output = RNA_string_length(ptr, "target_vertex_group") != 0;
258 uiLayoutSetPropDecorate(sub, false);
259 uiLayoutSetActive(sub, has_output);
260 uiItemR(sub, ptr, "use_invert_output", UI_ITEM_NONE, "", ICON_ARROW_LEFTRIGHT);
261
262 uiItemR(layout, ptr, "object", UI_ITEM_NONE, nullptr, ICON_NONE);
263
264 sub = uiLayoutColumn(layout, true);
265 uiItemR(sub, ptr, "distance_start", UI_ITEM_NONE, nullptr, ICON_NONE);
266 uiItemR(sub, ptr, "distance_end", UI_ITEM_NONE, nullptr, ICON_NONE);
267
268 uiItemR(layout, ptr, "minimum_weight", UI_ITEM_NONE, nullptr, ICON_NONE);
269 uiItemR(layout, ptr, "use_multiply", UI_ITEM_NONE, nullptr, ICON_NONE);
270
271 if (uiLayout *influence_panel = uiLayoutPanelProp(
272 C, layout, ptr, "open_influence_panel", IFACE_("Influence")))
273 {
274 modifier::greasepencil::draw_layer_filter_settings(C, influence_panel, ptr);
275 modifier::greasepencil::draw_material_filter_settings(C, influence_panel, ptr);
276 modifier::greasepencil::draw_vertex_group_settings(C, influence_panel, ptr);
277 }
278
279 modifier_panel_end(layout, ptr);
280}
281
286
287} // namespace blender
288
290 /*idname*/ "GreasePencilWeightProximityModifier",
291 /*name*/ N_("Weight Proximity"),
292 /*struct_name*/ "GreasePencilWeightProximityModifierData",
293 /*struct_size*/ sizeof(GreasePencilWeightProximityModifierData),
294 /*srna*/ &RNA_GreasePencilWeightProximityModifier,
296 /*flags*/
299 /*icon*/ ICON_MOD_VERTEX_WEIGHT,
300
301 /*copy_data*/ blender::copy_data,
302
303 /*deform_verts*/ nullptr,
304 /*deform_matrices*/ nullptr,
305 /*deform_verts_EM*/ nullptr,
306 /*deform_matrices_EM*/ nullptr,
307 /*modify_mesh*/ nullptr,
308 /*modify_geometry_set*/ blender::modify_geometry_set,
309
310 /*init_data*/ blender::init_data,
311 /*required_data_mask*/ nullptr,
312 /*free_data*/ blender::free_data,
313 /*is_disabled*/ blender::is_disabled,
314 /*update_depsgraph*/ blender::update_depsgraph,
315 /*depends_on_time*/ nullptr,
316 /*depends_on_normals*/ nullptr,
317 /*foreach_ID_link*/ blender::foreach_ID_link,
318 /*foreach_tex_link*/ nullptr,
319 /*free_runtime_data*/ nullptr,
320 /*panel_register*/ blender::panel_register,
321 /*blend_write*/ blender::blend_write,
322 /*blend_read*/ blender::blend_read,
323};
Low-level operations for curves.
support for deformation groups and hooks.
int BKE_defgroup_name_index(const ListBase *defbase, blender::StringRef name)
Definition deform.cc:534
Low-level operations for grease pencil.
@ IDWALK_CB_NOP
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
void BLI_addtail(struct ListBase *listbase, void *vlink) ATTR_NONNULL(1)
Definition listbase.cc:110
int BLI_listbase_count(const struct ListBase *listbase) ATTR_WARN_UNUSED_RESULT ATTR_NONNULL(1)
#define STRNCPY(dst, src)
Definition BLI_string.h:593
#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)
@ eModifierType_GreasePencilWeightProximity
@ MOD_GREASE_PENCIL_WEIGHT_PROXIMITY_INVERT_OUTPUT
@ MOD_GREASE_PENCIL_WEIGHT_PROXIMITY_MULTIPLY_DATA
struct GreasePencilWeightProximityModifierData GreasePencilWeightProximityModifierData
static bool is_disabled
static void panel_register(ARegionType *region_type)
static void panel_draw(const bContext *, Panel *panel)
ModifierTypeInfo modifierType_GreasePencilWeightProximity
static void modify_geometry_set(ModifierData *md, const ModifierEvalContext *ctx, blender::bke::GeometrySet *geometry_set)
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 uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
#define UI_ITEM_NONE
void uiItemPointerR(uiLayout *layout, PointerRNA *ptr, const char *propname, PointerRNA *searchptr, const char *searchpropname, const char *name, int icon)
PanelLayout uiLayoutPanelProp(const bContext *C, uiLayout *layout, PointerRNA *open_prop_owner, const char *open_prop_name)
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)
bke::CurvesGeometry & strokes_for_write()
CCL_NAMESPACE_BEGIN ccl_device float invert(float color, float factor)
Definition invert.h:9
T distance(const T &a, const T &b)
T max(const T &a, const T &b)
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)
VArray< float > get_influence_vertex_weights(const bke::CurvesGeometry &curves, const GreasePencilModifierInfluenceData &influence_data)
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(const IndexRange range, const int64_t grain_size, const Function &function, const TaskSizeHints &size_hints=detail::TaskSizeHints_Static(1))
Definition BLI_task.hh:95
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 write_weights_for_drawing(const ModifierData &md, const Object &ob, bke::greasepencil::Drawing &drawing)
static void init_data(ModifierData *md)
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
static float get_distance_factor(float3 target_pos, const float4x4 &obmat, float3 pos, const float dist_min, const float dist_max)
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 int ensure_vertex_group(const StringRefNull name, ListBase &vertex_group_names)
static bool target_vertex_group_available(const StringRefNull name, const ListBase &vertex_group_names)
static void update_depsgraph(ModifierData *md, const ModifierUpdateDepsgraphContext *ctx)
static bool is_disabled(const Scene *, ModifierData *md, bool)
static void blend_read(BlendDataReader *reader, ModifierData *md)
int RNA_string_length(PointerRNA *ptr, const char *name)
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:413
struct uiLayout * layout
GreasePencil * get_grease_pencil_for_write()
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4126
uint8_t flag
Definition wm_window.cc:138