Blender V5.0
MOD_grease_pencil_opacity.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 "DNA_defaults.h"
10#include "DNA_modifier_types.h"
11
12#include "BKE_colortools.hh"
13#include "BKE_curves.hh"
14#include "BKE_geometry_set.hh"
15#include "BKE_grease_pencil.hh"
16#include "BKE_modifier.hh"
17
18#include "BLO_read_write.hh"
19
21#include "UI_resources.hh"
22
23#include "BLT_translation.hh"
24
25#include "WM_types.hh"
26
27#include "RNA_access.hh"
28#include "RNA_prototypes.hh"
29
31#include "MOD_ui_common.hh"
32
33namespace blender {
34
38
48
49static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
50{
51 const auto *omd = reinterpret_cast<const GreasePencilOpacityModifierData *>(md);
52 auto *tomd = reinterpret_cast<GreasePencilOpacityModifierData *>(target);
53
55
57 modifier::greasepencil::copy_influence_data(&omd->influence, &tomd->influence, flag);
58}
59
60static void free_data(ModifierData *md)
61{
62 auto *omd = reinterpret_cast<GreasePencilOpacityModifierData *>(md);
64}
65
66static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
67{
68 auto *omd = reinterpret_cast<GreasePencilOpacityModifierData *>(md);
69 modifier::greasepencil::foreach_influence_ID_link(&omd->influence, ob, walk, user_data);
70}
71
73 bke::CurvesGeometry &curves,
74 const IndexMask &curves_mask)
75{
76 const bool use_uniform_opacity = (omd.flag & MOD_GREASE_PENCIL_OPACITY_USE_UNIFORM_OPACITY);
77 const bool use_weight_as_factor = (omd.flag & MOD_GREASE_PENCIL_OPACITY_USE_WEIGHT_AS_FACTOR);
78 const bool use_curve = (omd.influence.flag & GREASE_PENCIL_INFLUENCE_USE_CUSTOM_CURVE);
79 const OffsetIndices<int> points_by_curve = curves.points_by_curve();
80
83 "opacity", bke::AttrDomain::Point);
84 if (!opacities) {
85 return;
86 }
88 curves, omd.influence);
89
90 curves_mask.foreach_index(GrainSize(512), [&](const int64_t curve_i) {
91 const IndexRange points = points_by_curve[curve_i];
92 for (const int64_t point_i : points) {
93 const float vgroup_weight = vgroup_weights[point_i];
94 if (vgroup_weight <= 0.0f) {
95 continue;
96 }
97
98 const float curve_input = points.size() >= 2 ?
99 (float(point_i - points.first()) / float(points.size() - 1)) :
100 0.0f;
101 const float curve_factor = use_curve ? BKE_curvemapping_evaluateF(
102 omd.influence.custom_curve, 0, curve_input) :
103 1.0f;
104
105 if (use_uniform_opacity) {
106 opacities.span[point_i] = std::clamp(omd.color_factor * curve_factor, 0.0f, 1.0f);
107 }
108 else if (use_weight_as_factor) {
109 /* Use vertex group weights as opacity factors. */
110 opacities.span[point_i] = std::clamp(curve_factor * vgroup_weight, 0.0f, 1.0f);
111 }
112 else {
113 /* Use vertex group weights as influence factors. */
114 opacities.span[point_i] = std::clamp(
115 opacities.span[point_i] + (omd.color_factor * curve_factor - 1.0f) * vgroup_weight,
116 0.0f,
117 1.0f);
118 }
119 }
120 });
121
122 opacities.finish();
123}
124
126 bke::CurvesGeometry &curves,
127 const IndexMask &curves_mask)
128{
129 const bool use_vgroup_opacity = (omd.flag & MOD_GREASE_PENCIL_OPACITY_USE_WEIGHT_AS_FACTOR);
130 const OffsetIndices<int> points_by_curve = curves.points_by_curve();
131
133 /* Fill color opacity per stroke. */
134 bke::SpanAttributeWriter<float> fill_opacities = attributes.lookup_or_add_for_write_span<float>(
135 "fill_opacity",
139 curves, omd.influence);
140
141 curves_mask.foreach_index(GrainSize(512), [&](int64_t curve_i) {
142 /* Use the first stroke point as vertex weight. */
143 const IndexRange points = points_by_curve[curve_i];
144 const float vgroup_weight_first = vgroup_weights[points.first()];
145 float stroke_weight = vgroup_weight_first;
146 if (use_vgroup_opacity) {
147 if (points.is_empty() || (stroke_weight <= 0.0f)) {
148 stroke_weight = 1.0f;
149 }
150 fill_opacities.span[curve_i] = std::clamp(stroke_weight, 0.0f, 1.0f);
151 }
152 else {
153 if (!points.is_empty() && (stroke_weight > 0.0f)) {
154 fill_opacities.span[curve_i] = std::clamp(omd.color_factor * stroke_weight, 0.0f, 1.0f);
155 }
156 }
157 });
158
159 fill_opacities.finish();
160}
161
163 bke::CurvesGeometry &curves,
164 const IndexMask &curves_mask)
165{
168 "softness", bke::AttrDomain::Curve);
169
170 curves_mask.foreach_index(GrainSize(512), [&](int64_t curve_i) {
171 softness.span[curve_i] =
172 1.0f - std::clamp((1.0f - softness.span[curve_i]) * omd.hardness_factor, 0.0f, 1.0f);
173 });
174
175 softness.finish();
176}
177
179 const ModifierEvalContext *ctx,
180 bke::CurvesGeometry &curves)
181{
182 const auto *omd = reinterpret_cast<GreasePencilOpacityModifierData *>(md);
183
184 IndexMaskMemory mask_memory;
186 ctx->object, curves, omd->influence, mask_memory);
187
188 switch (omd->color_mode) {
190 modify_stroke_color(*omd, curves, curves_mask);
191 break;
193 modify_fill_color(*omd, curves, curves_mask);
194 break;
196 modify_stroke_color(*omd, curves, curves_mask);
197 modify_fill_color(*omd, curves, curves_mask);
198 break;
200 modify_softness(*omd, curves, curves_mask);
201 break;
202 }
203}
204
206 const ModifierEvalContext *ctx,
207 bke::GeometrySet *geometry_set)
208{
209 const auto *omd = reinterpret_cast<GreasePencilOpacityModifierData *>(md);
210
211 GreasePencil *grease_pencil = geometry_set->get_grease_pencil_for_write();
212 if (grease_pencil == nullptr) {
213 return;
214 }
215
216 IndexMaskMemory mask_memory;
218 *grease_pencil, omd->influence, mask_memory);
219 const int frame = grease_pencil->runtime->eval_frame;
221 *grease_pencil, layer_mask, frame);
223 drawings, [&](Drawing *drawing) { modify_curves(md, ctx, drawing->strokes_for_write()); });
224}
225
226static void panel_draw(const bContext *C, Panel *panel)
227{
228 uiLayout *layout = panel->layout;
229
230 PointerRNA ob_ptr;
232
233 layout->use_property_split_set(true);
234
236 RNA_enum_get(ptr, "color_mode"));
237
238 layout->prop(ptr, "color_mode", UI_ITEM_NONE, std::nullopt, ICON_NONE);
239
240 if (color_mode == MOD_GREASE_PENCIL_COLOR_HARDNESS) {
241 layout->prop(ptr, "hardness_factor", UI_ITEM_NONE, std::nullopt, ICON_NONE);
242 }
243 else {
244 const bool use_uniform_opacity = RNA_boolean_get(ptr, "use_uniform_opacity");
245 const bool use_weight_as_factor = RNA_boolean_get(ptr, "use_weight_as_factor");
246
247 layout->prop(ptr, "use_uniform_opacity", UI_ITEM_NONE, std::nullopt, ICON_NONE);
248 const char *text = (use_uniform_opacity) ? IFACE_("Opacity") : IFACE_("Opacity Factor");
249
250 uiLayout *row = &layout->row(true);
251 row->active_set(!use_weight_as_factor || use_uniform_opacity);
252 row->prop(ptr, "color_factor", UI_ITEM_NONE, text, ICON_NONE);
253 if (!use_uniform_opacity) {
254 uiLayout *sub = &row->row(true);
255 sub->active_set(true);
256 row->prop(ptr, "use_weight_as_factor", UI_ITEM_NONE, "", ICON_MOD_VERTEX_WEIGHT);
257 }
258 }
259
260 if (uiLayout *influence_panel = layout->panel_prop(
261 C, ptr, "open_influence_panel", IFACE_("Influence")))
262 {
267 }
268
270}
271
276
277static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
278{
279 const auto *omd = reinterpret_cast<const GreasePencilOpacityModifierData *>(md);
280
282 modifier::greasepencil::write_influence_data(writer, &omd->influence);
283}
284
285static void blend_read(BlendDataReader *reader, ModifierData *md)
286{
287 auto *omd = reinterpret_cast<GreasePencilOpacityModifierData *>(md);
288
289 modifier::greasepencil::read_influence_data(reader, &omd->influence);
290}
291
292} // namespace blender
293
295 /*idname*/ "GreasePencilOpacity",
296 /*name*/ N_("Opacity"),
297 /*struct_name*/ "GreasePencilOpacityModifierData",
298 /*struct_size*/ sizeof(GreasePencilOpacityModifierData),
299 /*srna*/ &RNA_GreasePencilOpacityModifier,
303 /*icon*/ ICON_MOD_OPACITY,
304
305 /*copy_data*/ blender::copy_data,
306
307 /*deform_verts*/ nullptr,
308 /*deform_matrices*/ nullptr,
309 /*deform_verts_EM*/ nullptr,
310 /*deform_matrices_EM*/ nullptr,
311 /*modify_mesh*/ nullptr,
312 /*modify_geometry_set*/ blender::modify_geometry_set,
313
314 /*init_data*/ blender::init_data,
315 /*required_data_mask*/ nullptr,
316 /*free_data*/ blender::free_data,
317 /*is_disabled*/ nullptr,
318 /*update_depsgraph*/ nullptr,
319 /*depends_on_time*/ nullptr,
320 /*depends_on_normals*/ nullptr,
321 /*foreach_ID_link*/ blender::foreach_ID_link,
322 /*foreach_tex_link*/ nullptr,
323 /*free_runtime_data*/ nullptr,
324 /*panel_register*/ blender::panel_register,
325 /*blend_write*/ blender::blend_write,
326 /*blend_read*/ blender::blend_read,
327 /*foreach_cache*/ nullptr,
328 /*foreach_working_space_color*/ nullptr,
329};
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_SupportsMapping
@ 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)
@ GREASE_PENCIL_INFLUENCE_USE_CUSTOM_CURVE
@ MOD_GREASE_PENCIL_OPACITY_USE_WEIGHT_AS_FACTOR
@ MOD_GREASE_PENCIL_OPACITY_USE_UNIFORM_OPACITY
GreasePencilModifierColorMode
@ MOD_GREASE_PENCIL_COLOR_FILL
@ MOD_GREASE_PENCIL_COLOR_STROKE
@ MOD_GREASE_PENCIL_COLOR_BOTH
@ MOD_GREASE_PENCIL_COLOR_HARDNESS
@ eModifierType_GreasePencilOpacity
ModifierTypeInfo modifierType_GreasePencilOpacity
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 first() const
constexpr int64_t size() const
constexpr bool is_empty() const
static VArray from_single(T value, const int64_t size)
OffsetIndices< int > points_by_curve() const
MutableAttributeAccessor attributes_for_write()
GSpanAttributeWriter lookup_or_add_for_write_span(StringRef attribute_id, AttrDomain domain, AttrType data_type, const AttributeInit &initializer=AttributeInitDefaultValue())
bke::CurvesGeometry & strokes_for_write()
void foreach_index(Fn &&fn) const
nullptr float
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 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 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 modify_curves(ModifierData &md, const ModifierEvalContext &ctx, Drawing &drawing, bke::GreasePencilDrawingEditHints *edit_hints)
static void modify_softness(const GreasePencilOpacityModifierData &omd, bke::CurvesGeometry &curves, const IndexMask &curves_mask)
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 modify_fill_color(Object &ob, const GreasePencilColorModifierData &cmd, Drawing &drawing, const IndexMask &curves_mask)
static void free_data(ModifierData *md)
static void panel_register(ARegionType *region_type)
static void modify_stroke_color(Object &ob, const GreasePencilColorModifierData &cmd, bke::CurvesGeometry &curves, const IndexMask &curves_mask, const MutableSpan< ColorGeometry4f > vertex_colors)
static void blend_read(BlendDataReader *reader, ModifierData *md)
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
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 active_set(bool active)
uiLayout & row(bool align)
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