Blender V5.0
MOD_grease_pencil_subdiv.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 "BKE_geometry_set.hh"
20#include "BKE_grease_pencil.hh"
21#include "BKE_modifier.hh"
22
24
26#include "UI_resources.hh"
27
29#include "MOD_ui_common.hh"
30
31#include "RNA_prototypes.hh"
32
33namespace blender {
34
44
51
52static void copy_data(const ModifierData *md, ModifierData *target, int flag)
53{
55 reinterpret_cast<const GreasePencilSubdivModifierData *>(md);
57 target);
58
61}
62
63static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
64{
66 reinterpret_cast<const GreasePencilSubdivModifierData *>(md);
67
70}
71
78
80{
82 const bool use_catmull_clark = mmd.type == MOD_GREASE_PENCIL_SUBDIV_CATMULL;
83
84 IndexMaskMemory memory;
86 &ob, drawing.strokes_for_write(), mmd.influence, memory);
87
88 if (use_catmull_clark) {
90 bke::CurvesGeometry subdivided_curves = drawing.strokes();
91 const VArray<bool> cyclic = subdivided_curves.cyclic();
92 for ([[maybe_unused]] const int level_i : IndexRange(mmd.level)) {
93 VArray<int> one_cut = VArray<int>::from_single(1, subdivided_curves.points_num());
94 subdivided_curves = geometry::subdivide_curves(
95 subdivided_curves, strokes, std::move(one_cut), {});
96
97 offset_indices::OffsetIndices<int> points_by_curve = subdivided_curves.points_by_curve();
98 const Array<float3> src_positions = subdivided_curves.positions();
99 MutableSpan<float3> dst_positions = subdivided_curves.positions_for_write();
100 threading::parallel_for(subdivided_curves.curves_range(), 1024, [&](const IndexRange range) {
101 for (const int curve_i : range) {
102 const IndexRange points = points_by_curve[curve_i];
103 for (const int point_i : points.drop_front(1).drop_back(1)) {
104 dst_positions[point_i] = math::interpolate(
105 src_positions[point_i],
106 math::interpolate(src_positions[point_i - 1], src_positions[point_i + 1], 0.5f),
107 0.5f);
108 }
109
110 if (cyclic[curve_i] && points.size() > 1) {
111 const float3 &first_pos = src_positions[points.first()];
112 const float3 &last_pos = src_positions[points.last()];
113 const float3 &after_first_pos = src_positions[points.first() + 1];
114 const float3 &before_last_pos = src_positions[points.last() - 1];
115 dst_positions[points.first()] = math::interpolate(
116 first_pos, math::interpolate(last_pos, after_first_pos, 0.5f), 0.5f);
117 dst_positions[points.last()] = math::interpolate(
118 last_pos, math::interpolate(before_last_pos, first_pos, 0.5f), 0.5f);
119 }
120 }
121 });
122 }
123 drawing.strokes_for_write() = subdivided_curves;
124 }
125 else {
127 drawing.strokes().points_num());
128 drawing.strokes_for_write() = geometry::subdivide_curves(drawing.strokes(), strokes, cuts, {});
129 }
130
131 drawing.tag_topology_changed();
132}
133
135 const ModifierEvalContext *ctx,
136 bke::GeometrySet *geometry_set)
137{
139
140 if (mmd->level < 1 || !geometry_set->has_grease_pencil()) {
141 return;
142 }
143
144 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
145 const int current_frame = grease_pencil.runtime->eval_frame;
146
147 IndexMaskMemory mask_memory;
149 grease_pencil, mmd->influence, mask_memory);
151 modifier::greasepencil::get_drawings_for_write(grease_pencil, layer_mask, current_frame);
152
154 subdivide_drawing(*md, *ctx->object, *drawing);
155 });
156}
157
158static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
159{
161
163}
164
165static void panel_draw(const bContext *C, Panel *panel)
166{
167 uiLayout *layout = panel->layout;
168
170
171 layout->use_property_split_set(true);
172
173 layout->prop(ptr, "subdivision_type", UI_ITEM_NONE, std::nullopt, ICON_NONE);
174 layout->prop(ptr, "level", UI_ITEM_NONE, IFACE_("Subdivisions"), ICON_NONE);
175
176 if (uiLayout *influence_panel = layout->panel_prop(
177 C, ptr, "open_influence_panel", IFACE_("Influence")))
178 {
181 }
182
184}
185
190
191} // namespace blender
192
194 /*idname*/ "GreasePencilSubdivModifier",
195 /*name*/ N_("Subdivide"),
196 /*struct_name*/ "GreasePencilSubdivModifierData",
197 /*struct_size*/ sizeof(GreasePencilSubdivModifierData),
198 /*srna*/ &RNA_GreasePencilSubdivModifier,
200 /*flags*/
203 /*icon*/ ICON_MOD_SUBSURF,
204
205 /*copy_data*/ blender::copy_data,
206
207 /*deform_verts*/ nullptr,
208 /*deform_matrices*/ nullptr,
209 /*deform_verts_EM*/ nullptr,
210 /*deform_matrices_EM*/ nullptr,
211 /*modify_mesh*/ nullptr,
212 /*modify_geometry_set*/ blender::modify_geometry_set,
213
214 /*init_data*/ blender::init_data,
215 /*required_data_mask*/ nullptr,
216 /*free_data*/ blender::free_data,
217 /*is_disabled*/ nullptr,
218 /*update_depsgraph*/ nullptr,
219 /*depends_on_time*/ nullptr,
220 /*depends_on_normals*/ nullptr,
221 /*foreach_ID_link*/ blender::foreach_ID_link,
222 /*foreach_tex_link*/ nullptr,
223 /*free_runtime_data*/ nullptr,
224 /*panel_register*/ blender::panel_register,
225 /*blend_write*/ blender::blend_write,
226 /*blend_read*/ blender::blend_read,
227 /*foreach_cache*/ nullptr,
228 /*foreach_working_space_color*/ nullptr,
229};
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
#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)
#define DNA_struct_default_get(struct_name)
@ MOD_GREASE_PENCIL_SUBDIV_CATMULL
@ eModifierType_GreasePencilSubdiv
static void panel_register(ARegionType *region_type)
static void panel_draw(const bContext *, Panel *panel)
static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
ModifierTypeInfo modifierType_GreasePencilSubdiv
static void modify_geometry_set(ModifierData *md, const ModifierEvalContext *ctx, blender::bke::GeometrySet *geometry_set)
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 VArray from_single(T value, const int64_t size)
MutableSpan< float3 > positions_for_write()
OffsetIndices< int > points_by_curve() const
IndexRange curves_range() const
Span< float3 > positions() const
VArray< bool > cyclic() const
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
bke::CurvesGeometry subdivide_curves(const bke::CurvesGeometry &src_curves, const IndexMask &selection, const VArray< int > &cuts, const bke::AttributeFilter &attribute_filter={})
T pow(const T &x, const T &power)
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)
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
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:93
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 free_data(ModifierData *md)
static void subdivide_drawing(ModifierData &md, Object &ob, bke::greasepencil::Drawing &drawing)
static void panel_register(ARegionType *region_type)
static void blend_read(BlendDataReader *reader, ModifierData *md)
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:414
struct uiLayout * layout
GreasePencil * get_grease_pencil_for_write()
PanelLayout panel_prop(const bContext *C, PointerRNA *open_prop_owner, blender::StringRefNull open_prop_name)
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)
#define N_(msgid)
PointerRNA * ptr
Definition wm_files.cc:4238
uint8_t flag
Definition wm_window.cc:145