Blender V5.0
MOD_grease_pencil_outline.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 "BKE_attribute.hh"
10#include "BKE_material.hh"
11#include "BLI_array_utils.hh"
12#include "BLI_index_range.hh"
13#include "BLI_math_matrix.hh"
14#include "BLI_math_vector.hh"
15#include "BLI_offset_indices.hh"
16#include "BLI_span.hh"
17#include "BLI_virtual_array.hh"
18
19#include "DNA_defaults.h"
20#include "DNA_modifier_types.h"
21#include "DNA_scene_types.h"
22
23#include "BKE_curves.hh"
24#include "BKE_geometry_set.hh"
25#include "BKE_grease_pencil.hh"
26#include "BKE_instances.hh"
27#include "BKE_lib_query.hh"
28#include "BKE_modifier.hh"
29#include "BKE_screen.hh"
30
31#include "BLO_read_write.hh"
32
34
35#include "ED_grease_pencil.hh"
36
38
40#include "UI_resources.hh"
41
42#include "BLT_translation.hh"
43
44#include "WM_api.hh"
45#include "WM_types.hh"
46
47#include "RNA_access.hh"
48#include "RNA_prototypes.hh"
49
51#include "MOD_ui_common.hh"
52
53namespace blender {
54
64
65static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
66{
67 const auto *omd = reinterpret_cast<const GreasePencilOutlineModifierData *>(md);
68 auto *tmmd = reinterpret_cast<GreasePencilOutlineModifierData *>(target);
69
71
73 modifier::greasepencil::copy_influence_data(&omd->influence, &tmmd->influence, flag);
74}
75
76static void free_data(ModifierData *md)
77{
78 auto *omd = reinterpret_cast<GreasePencilOutlineModifierData *>(md);
80}
81
82static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
83{
84 auto *omd = reinterpret_cast<GreasePencilOutlineModifierData *>(md);
85 modifier::greasepencil::foreach_influence_ID_link(&omd->influence, ob, walk, user_data);
86 walk(user_data, ob, (ID **)&omd->outline_material, IDWALK_CB_USER);
87 walk(user_data, ob, (ID **)&omd->object, IDWALK_CB_NOP);
88}
89
91{
92 auto *omd = reinterpret_cast<GreasePencilOutlineModifierData *>(md);
93 if (ctx->scene->camera) {
95 ctx->node, ctx->scene->camera, DEG_OB_COMP_TRANSFORM, "Grease Pencil Outline Modifier");
97 ctx->node, ctx->scene->camera, DEG_OB_COMP_PARAMETERS, "Grease Pencil Outline Modifier");
98 }
99 if (omd->object != nullptr) {
101 ctx->node, omd->object, DEG_OB_COMP_TRANSFORM, "Grease Pencil Outline Modifier");
102 }
104 ctx->node, ctx->object, DEG_OB_COMP_TRANSFORM, "Grease Pencil Outline Modifier");
105}
106
113 const IndexMask &curve_selection,
114 const Span<int> curve_offsets)
115{
116 BLI_assert(curve_offsets.size() == src_curves.curves_num());
117
118 OffsetIndices<int> src_offsets = src_curves.points_by_curve();
119 bke::AttributeAccessor src_attributes = src_curves.attributes();
120
121 Array<int> indices(src_curves.points_num());
122 curve_selection.foreach_index(GrainSize(512), [&](const int64_t curve_i) {
123 const IndexRange points = src_offsets[curve_i];
124 const int point_num = points.size();
125 const int point_start = points.start();
126 MutableSpan<int> point_indices = indices.as_mutable_span().slice(points);
127 if (points.size() < 2) {
128 array_utils::fill_index_range(point_indices, point_start);
129 return;
130 }
131 /* Offset can be negative or larger than the buffer. Use modulo to get an
132 * equivalent offset within buffer size to simplify copying. */
133 const int offset_raw = curve_offsets[curve_i];
134 const int offset = offset_raw >= 0 ? offset_raw % points.size() :
135 points.size() - ((-offset_raw) % points.size());
136 BLI_assert(0 <= offset && offset < points.size());
137 if (offset == 0) {
138 array_utils::fill_index_range(point_indices, point_start);
139 return;
140 }
141
142 const int point_middle = point_start + offset;
143 array_utils::fill_index_range(point_indices.take_front(point_num - offset), point_middle);
144 array_utils::fill_index_range(point_indices.take_back(offset), point_start);
145 });
146
147 /* Have to make a copy of the input geometry, gather_attributes does not work in-place when the
148 * source indices are not ordered. */
149 bke::CurvesGeometry dst_curves(src_curves);
150 bke::MutableAttributeAccessor dst_attributes = dst_curves.attributes_for_write();
152 src_attributes, bke::AttrDomain::Point, bke::AttrDomain::Point, {}, indices, dst_attributes);
153
154 return dst_curves;
155}
156
157static int find_closest_point(const Span<float3> positions, const float3 &target)
158{
159 if (positions.is_empty()) {
160 return 0;
161 }
162
163 int closest_i = 0;
164 float min_dist_squared = math::distance_squared(positions.first(), target);
165 for (const int i : positions.index_range().drop_front(1)) {
166 const float dist_squared = math::distance_squared(positions[i], target);
167 if (dist_squared < min_dist_squared) {
168 closest_i = i;
169 min_dist_squared = dist_squared;
170 }
171 }
172 return closest_i;
173}
174
176 const ModifierEvalContext &ctx,
178 const float4x4 &viewmat)
179{
181
182 if (drawing.strokes().curve_num == 0) {
183 return;
184 }
185
186 /* Selected source curves. */
187 IndexMaskMemory curve_mask_memory;
189 ctx.object, drawing.strokes(), omd.influence, curve_mask_memory);
190
191 /* Unit object scale is applied to the stroke radius. */
192 const float object_scale = math::length(
193 math::transform_direction(ctx.object->object_to_world(), float3(M_SQRT1_3)));
194 /* Legacy thickness setting is diameter in pixels, divide by 2000 to get radius. */
195 const float radius = math::max(omd.thickness * object_scale, 1.0f) *
197 /* Offset the strokes by the radius so the outside aligns with the input stroke. */
198 const float outline_offset = (omd.flag & MOD_GREASE_PENCIL_OUTLINE_KEEP_SHAPE) != 0 ? -radius :
199 0.0f;
200 const int mat_nr = (omd.outline_material ?
202 -1);
203
205 drawing, curves_mask, viewmat, omd.subdiv, radius, outline_offset, mat_nr);
206
207 /* Cyclic curve reordering feature. */
208 if (omd.object) {
209 const OffsetIndices points_by_curve = curves.points_by_curve();
210
211 /* Computes the offset of the closest point to the object from the curve start. */
212 Array<int> offset_by_curve(curves.curves_num());
213 for (const int i : curves.curves_range()) {
214 const IndexRange points = points_by_curve[i];
215 /* Closest point index is already relative to the point range and can be used as offset. */
216 offset_by_curve[i] = find_closest_point(curves.positions().slice(points), omd.object->loc);
217 }
218
219 curves = reorder_cyclic_curve_points(curves, curves.curves_range(), offset_by_curve);
220 }
221
222 /* Resampling feature. */
223 if (omd.sample_length > 0.0f) {
225 curves.curves_num());
226 curves = geometry::resample_to_length(curves, curves.curves_range(), sample_lengths);
227 }
228
229 drawing.strokes_for_write() = std::move(curves);
230 drawing.tag_topology_changed();
231}
232
234 const ModifierEvalContext *ctx,
235 bke::GeometrySet *geometry_set)
236{
240
241 const auto &omd = *reinterpret_cast<const GreasePencilOutlineModifierData *>(md);
242
243 const Scene *scene = DEG_get_evaluated_scene(ctx->depsgraph);
244 if (!scene->camera) {
245 return;
246 }
247 const float4x4 viewinv = scene->camera->world_to_object();
248
249 if (!geometry_set->has_grease_pencil()) {
250 return;
251 }
252 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
253 const int frame = grease_pencil.runtime->eval_frame;
254
255 IndexMaskMemory mask_memory;
257 grease_pencil, omd.influence, mask_memory);
258
260 grease_pencil, layer_mask, frame);
261 threading::parallel_for_each(drawings, [&](const LayerDrawingInfo &info) {
262 const Layer &layer = grease_pencil.layer(info.layer_index);
263 const float4x4 viewmat = viewinv * layer.to_world_space(*ctx->object);
264 modify_drawing(omd, *ctx, *info.drawing, viewmat);
265 });
266}
267
268static void panel_draw(const bContext *C, Panel *panel)
269{
270 uiLayout *layout = panel->layout;
271
272 PointerRNA ob_ptr;
274
275 layout->use_property_split_set(true);
276
277 layout->prop(ptr, "thickness", UI_ITEM_NONE, std::nullopt, ICON_NONE);
278 layout->prop(ptr, "use_keep_shape", UI_ITEM_NONE, std::nullopt, ICON_NONE);
279 layout->prop(ptr, "subdivision", UI_ITEM_NONE, std::nullopt, ICON_NONE);
280 layout->prop(ptr, "sample_length", UI_ITEM_NONE, std::nullopt, ICON_NONE);
281 layout->prop(ptr, "outline_material", UI_ITEM_NONE, std::nullopt, ICON_NONE);
282 layout->prop(ptr, "object", UI_ITEM_NONE, std::nullopt, ICON_NONE);
283
284 Scene *scene = CTX_data_scene(C);
285 if (scene->camera == nullptr) {
286 layout->label(RPT_("Outline requires an active camera"), ICON_ERROR);
287 }
288
289 if (uiLayout *influence_panel = layout->panel_prop(
290 C, ptr, "open_influence_panel", IFACE_("Influence")))
291 {
294 }
295
297}
298
303
304static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
305{
306 const auto *omd = reinterpret_cast<const GreasePencilOutlineModifierData *>(md);
307
309 modifier::greasepencil::write_influence_data(writer, &omd->influence);
310}
311
312static void blend_read(BlendDataReader *reader, ModifierData *md)
313{
314 auto *omd = reinterpret_cast<GreasePencilOutlineModifierData *>(md);
315
316 modifier::greasepencil::read_influence_data(reader, &omd->influence);
317}
318
319} // namespace blender
320
322 /*idname*/ "GreasePencilOutline",
323 /*name*/ N_("Outline"),
324 /*struct_name*/ "GreasePencilOutlineModifierData",
325 /*struct_size*/ sizeof(GreasePencilOutlineModifierData),
326 /*srna*/ &RNA_GreasePencilOutlineModifier,
330 /*icon*/ ICON_MOD_OUTLINE,
331
332 /*copy_data*/ blender::copy_data,
333
334 /*deform_verts*/ nullptr,
335 /*deform_matrices*/ nullptr,
336 /*deform_verts_EM*/ nullptr,
337 /*deform_matrices_EM*/ nullptr,
338 /*modify_mesh*/ nullptr,
339 /*modify_geometry_set*/ blender::modify_geometry_set,
340
341 /*init_data*/ blender::init_data,
342 /*required_data_mask*/ nullptr,
343 /*free_data*/ blender::free_data,
344 /*is_disabled*/ nullptr,
345 /*update_depsgraph*/ blender::update_depsgraph,
346 /*depends_on_time*/ nullptr,
347 /*depends_on_normals*/ nullptr,
348 /*foreach_ID_link*/ blender::foreach_ID_link,
349 /*foreach_tex_link*/ nullptr,
350 /*free_runtime_data*/ nullptr,
351 /*panel_register*/ blender::panel_register,
352 /*blend_write*/ blender::blend_write,
353 /*blend_read*/ blender::blend_read,
354};
Scene * CTX_data_scene(const bContext *C)
Low-level operations for curves.
Low-level operations for grease pencil.
@ IDWALK_CB_USER
@ IDWALK_CB_NOP
General operations, lookup, etc. for materials.
int BKE_object_material_index_get(Object *ob, const Material *ma)
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 M_SQRT1_3
#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 RPT_(msgid)
#define IFACE_(msgid)
void DEG_add_object_relation(DepsNodeHandle *node_handle, Object *object, eDepsObjectComponentType component, const char *description)
@ DEG_OB_COMP_TRANSFORM
@ DEG_OB_COMP_PARAMETERS
Scene * DEG_get_evaluated_scene(const Depsgraph *graph)
#define DNA_struct_default_get(struct_name)
@ MOD_GREASE_PENCIL_OUTLINE_KEEP_SHAPE
@ eModifierType_GreasePencilOutline
ModifierTypeInfo modifierType_GreasePencilOutline
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
long long int int64_t
constexpr int64_t size() const
constexpr int64_t start() const
constexpr IndexRange drop_front(int64_t n) const
constexpr MutableSpan take_back(const int64_t n) const
Definition BLI_span.hh:640
constexpr MutableSpan take_front(const int64_t n) const
Definition BLI_span.hh:629
constexpr const T & first() const
Definition BLI_span.hh:315
constexpr int64_t size() const
Definition BLI_span.hh:252
constexpr IndexRange index_range() const
Definition BLI_span.hh:401
constexpr bool is_empty() const
Definition BLI_span.hh:260
static VArray from_single(T value, const int64_t size)
OffsetIndices< int > points_by_curve() const
IndexRange curves_range() const
MutableAttributeAccessor attributes_for_write()
Span< float3 > positions() const
AttributeAccessor attributes() const
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
float4x4 to_world_space(const Object &object) const
void foreach_index(Fn &&fn) const
static ushort indices[]
void fill_index_range(MutableSpan< T > span, const T start=0)
constexpr float LEGACY_RADIUS_CONVERSION_FACTOR
void gather_attributes(AttributeAccessor src_attributes, AttrDomain src_domain, AttrDomain dst_domain, const AttributeFilter &attribute_filter, const IndexMask &selection, MutableAttributeAccessor dst_attributes)
bke::CurvesGeometry create_curves_outline(const bke::greasepencil::Drawing &drawing, const IndexMask &strokes, const float4x4 &transform, const int corner_subdivisions, const float outline_radius, const float outline_offset, const int material_index)
CurvesGeometry resample_to_length(const CurvesGeometry &src_curves, const IndexMask &selection, const VArray< float > &sample_lengths, const ResampleCurvesOutputAttributeIDs &output_ids={}, bool keep_last_segment=false)
T length(const VecBase< T, Size > &a)
T distance_squared(const VecBase< T, Size > &a, const VecBase< T, Size > &b)
T max(const T &a, const T &b)
VecBase< T, 3 > transform_direction(const MatBase< T, 3, 3 > &mat, const VecBase< T, 3 > &direction)
void read_influence_data(BlendDataReader *reader, GreasePencilModifierInfluenceData *influence_data)
void init_influence_data(GreasePencilModifierInfluenceData *influence_data, const bool has_custom_curve)
Vector< LayerDrawingInfo > get_drawing_infos_by_layer(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)
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 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 int find_closest_point(const Span< float3 > positions, const float3 &target)
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 bke::CurvesGeometry reorder_cyclic_curve_points(const bke::CurvesGeometry &src_curves, const IndexMask &curve_selection, const Span< int > curve_offsets)
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 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)
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:414
float loc[3]
struct uiLayout * layout
struct Object * camera
GreasePencil * get_grease_pencil_for_write()
PanelLayout panel_prop(const bContext *C, PointerRNA *open_prop_owner, blender::StringRefNull open_prop_name)
void label(blender::StringRef name, int icon)
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
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4238
uint8_t flag
Definition wm_window.cc:145