Blender V4.3
MOD_grease_pencil_color.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
9#include "MEM_guardedalloc.h"
10
11#include "BLI_math_color.hh"
12
13#include "DNA_defaults.h"
14#include "DNA_material_types.h"
15#include "DNA_modifier_types.h"
16#include "DNA_scene_types.h"
17
18#include "BKE_colortools.hh"
19#include "BKE_curves.hh"
20#include "BKE_geometry_set.hh"
21#include "BKE_grease_pencil.hh"
22#include "BKE_material.h"
23#include "BKE_modifier.hh"
24#include "BKE_screen.hh"
25
26#include "BLO_read_write.hh"
27
29
30#include "UI_interface.hh"
31#include "UI_resources.hh"
32
33#include "BLT_translation.hh"
34
35#include "WM_types.hh"
36
37#include "RNA_access.hh"
38#include "RNA_enum_types.hh"
39#include "RNA_prototypes.hh"
40
42#include "MOD_modifiertypes.hh"
43#include "MOD_ui_common.hh"
44
45namespace blender {
46
47using bke::greasepencil::Drawing;
48
49static void init_data(ModifierData *md)
50{
51 auto *cmd = reinterpret_cast<GreasePencilColorModifierData *>(md);
52
54
57}
58
59static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
60{
61 const auto *cmd = reinterpret_cast<const GreasePencilColorModifierData *>(md);
62 auto *tcmd = reinterpret_cast<GreasePencilColorModifierData *>(target);
63
65
67 modifier::greasepencil::copy_influence_data(&cmd->influence, &tcmd->influence, flag);
68}
69
70static void free_data(ModifierData *md)
71{
72 auto *cmd = reinterpret_cast<GreasePencilColorModifierData *>(md);
74}
75
76static void foreach_ID_link(ModifierData *md, Object *ob, IDWalkFunc walk, void *user_data)
77{
78 auto *cmd = reinterpret_cast<GreasePencilColorModifierData *>(md);
79 modifier::greasepencil::foreach_influence_ID_link(&cmd->influence, ob, walk, user_data);
80}
81
83 const ColorGeometry4f &material_color,
84 const float3 factor)
85{
86 float3 hsv;
87 /* When input alpha is zero, replace with material color. */
88 if (color.a == 0.0f && material_color.a > 0.0f) {
89 color.a = 1.0f;
90 rgb_to_hsv_v(material_color, hsv);
91 }
92 else {
93 rgb_to_hsv_v(color, hsv);
94 }
95 hsv[0] = fractf(hsv[0] + factor[0] + 0.5f);
96 hsv[1] = clamp_f(hsv[1] * factor[1], 0.0f, 1.0f);
97 hsv[2] = hsv[2] * factor[2];
98 hsv_to_rgb_v(hsv, color);
99}
100
103 bke::CurvesGeometry &curves,
104 const IndexMask &curves_mask,
105 const MutableSpan<ColorGeometry4f> vertex_colors)
106{
107 const bool use_curve = (cmd.influence.flag & GREASE_PENCIL_INFLUENCE_USE_CUSTOM_CURVE);
108 const OffsetIndices<int> points_by_curve = curves.points_by_curve();
109
110 bke::AttributeAccessor attributes = curves.attributes();
111 const VArray<int> stroke_materials = *attributes.lookup_or_default<int>(
112 "material_index", bke::AttrDomain::Curve, 0);
113
114 curves_mask.foreach_index(GrainSize(512), [&](const int64_t curve_i) {
115 const Material *ma = BKE_object_material_get(&ob, stroke_materials[curve_i] + 1);
116 const MaterialGPencilStyle *gp_style = ma ? ma->gp_style : nullptr;
117 const ColorGeometry4f material_color = (gp_style ? ColorGeometry4f(gp_style->stroke_rgba) :
118 ColorGeometry4f(0.0f, 0.0f, 0.0f, 0.0f));
119
120 const IndexRange points = points_by_curve[curve_i];
121 for (const int64_t i : points.index_range()) {
122 const int64_t point_i = points[i];
123 float3 factor = cmd.hsv;
124 if (use_curve) {
125 const float curve_input = points.size() >= 2 ? (float(i) / float(points.size() - 1)) :
126 0.0f;
127 const float curve_factor = BKE_curvemapping_evaluateF(
128 cmd.influence.custom_curve, 0, curve_input);
129 factor *= curve_factor;
130 }
131
132 apply_color_factor(vertex_colors[point_i], material_color, factor);
133 }
134 });
135}
136
137static void modify_fill_color(Object &ob,
139 Drawing &drawing,
140 const IndexMask &curves_mask)
141{
142 const bke::CurvesGeometry &curves = drawing.strokes();
143 const bke::AttributeAccessor attributes = curves.attributes();
144 /* Fill color per stroke. */
146 const VArray<int> stroke_materials = *attributes.lookup_or_default<int>(
147 "material_index", bke::AttrDomain::Curve, 0);
148
149 curves_mask.foreach_index(GrainSize(512), [&](int64_t curve_i) {
150 const Material *ma = BKE_object_material_get(&ob, stroke_materials[curve_i] + 1);
151 const MaterialGPencilStyle *gp_style = ma ? ma->gp_style : nullptr;
152 const ColorGeometry4f material_color = (gp_style ? ColorGeometry4f(gp_style->fill_rgba) :
153 ColorGeometry4f(0.0f, 0.0f, 0.0f, 0.0f));
154
155 apply_color_factor(fill_colors[curve_i], material_color, cmd.hsv);
156 });
157}
158
159static void modify_drawing(ModifierData &md, const ModifierEvalContext &ctx, Drawing &drawing)
160{
161 auto &cmd = reinterpret_cast<GreasePencilColorModifierData &>(md);
162
163 bke::CurvesGeometry &curves = drawing.strokes_for_write();
164 IndexMaskMemory mask_memory;
166 ctx.object, curves, cmd.influence, mask_memory);
167
168 switch (cmd.color_mode) {
171 *ctx.object, cmd, curves, curves_mask, drawing.vertex_colors_for_write());
172 break;
174 modify_fill_color(*ctx.object, cmd, drawing, curves_mask);
175 break;
178 *ctx.object, cmd, curves, curves_mask, drawing.vertex_colors_for_write());
179 modify_fill_color(*ctx.object, cmd, drawing, curves_mask);
180 break;
183 break;
184 }
185}
186
188 const ModifierEvalContext *ctx,
189 bke::GeometrySet *geometry_set)
190{
191 auto *cmd = reinterpret_cast<GreasePencilColorModifierData *>(md);
192
193 if (!geometry_set->has_grease_pencil()) {
194 return;
195 }
196 GreasePencil &grease_pencil = *geometry_set->get_grease_pencil_for_write();
197 const int frame = grease_pencil.runtime->eval_frame;
198
199 IndexMaskMemory mask_memory;
201 grease_pencil, cmd->influence, mask_memory);
203 grease_pencil, layer_mask, frame);
205 [&](Drawing *drawing) { modify_drawing(*md, *ctx, *drawing); });
206}
207
208static void panel_draw(const bContext *C, Panel *panel)
209{
210 uiLayout *layout = panel->layout;
211
212 PointerRNA ob_ptr;
214
215 uiLayoutSetPropSep(layout, true);
216
217 uiItemR(layout, ptr, "color_mode", UI_ITEM_NONE, nullptr, ICON_NONE);
218
219 uiItemR(layout, ptr, "hue", UI_ITEM_R_SLIDER, nullptr, ICON_NONE);
220 uiItemR(layout, ptr, "saturation", UI_ITEM_R_SLIDER, nullptr, ICON_NONE);
221 uiItemR(layout, ptr, "value", UI_ITEM_R_SLIDER, nullptr, ICON_NONE);
222
223 if (uiLayout *influence_panel = uiLayoutPanelProp(
224 C, layout, ptr, "open_influence_panel", IFACE_("Influence")))
225 {
229 }
230
231 modifier_panel_end(layout, ptr);
232}
233
238
239static void blend_write(BlendWriter *writer, const ID * /*id_owner*/, const ModifierData *md)
240{
241 const auto *cmd = reinterpret_cast<const GreasePencilColorModifierData *>(md);
242
244 modifier::greasepencil::write_influence_data(writer, &cmd->influence);
245}
246
247static void blend_read(BlendDataReader *reader, ModifierData *md)
248{
249 auto *cmd = reinterpret_cast<GreasePencilColorModifierData *>(md);
250
251 modifier::greasepencil::read_influence_data(reader, &cmd->influence);
252}
253
254} // namespace blender
255
257 /*idname*/ "GreasePencilColor",
258 /*name*/ N_("Color"),
259 /*struct_name*/ "GreasePencilColorModifierData",
260 /*struct_size*/ sizeof(GreasePencilColorModifierData),
261 /*srna*/ &RNA_GreasePencilColorModifier,
265 /*icon*/ ICON_MOD_HUE_SATURATION,
266
267 /*copy_data*/ blender::copy_data,
268
269 /*deform_verts*/ nullptr,
270 /*deform_matrices*/ nullptr,
271 /*deform_verts_EM*/ nullptr,
272 /*deform_matrices_EM*/ nullptr,
273 /*modify_mesh*/ nullptr,
274 /*modify_geometry_set*/ blender::modify_geometry_set,
275
276 /*init_data*/ blender::init_data,
277 /*required_data_mask*/ nullptr,
278 /*free_data*/ blender::free_data,
279 /*is_disabled*/ nullptr,
280 /*update_depsgraph*/ nullptr,
281 /*depends_on_time*/ nullptr,
282 /*depends_on_normals*/ nullptr,
283 /*foreach_ID_link*/ blender::foreach_ID_link,
284 /*foreach_tex_link*/ nullptr,
285 /*free_runtime_data*/ nullptr,
286 /*panel_register*/ blender::panel_register,
287 /*blend_write*/ blender::blend_write,
288 /*blend_read*/ blender::blend_read,
289};
float BKE_curvemapping_evaluateF(const CurveMapping *cumap, int cur, float value)
Low-level operations for curves.
Low-level operations for grease pencil.
General operations, lookup, etc. for materials.
struct Material * BKE_object_material_get(struct Object *ob, short act)
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_unreachable()
Definition BLI_assert.h:97
#define BLI_assert(a)
Definition BLI_assert.h:50
MINLINE float clamp_f(float value, float min, float max)
void hsv_to_rgb_v(const float hsv[3], float r_rgb[3])
Definition math_color.cc:57
void rgb_to_hsv_v(const float rgb[3], float r_hsv[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 IFACE_(msgid)
#define DNA_struct_default_get(struct_name)
@ GREASE_PENCIL_INFLUENCE_USE_CUSTOM_CURVE
@ MOD_GREASE_PENCIL_COLOR_FILL
@ MOD_GREASE_PENCIL_COLOR_STROKE
@ MOD_GREASE_PENCIL_COLOR_BOTH
@ MOD_GREASE_PENCIL_COLOR_HARDNESS
struct GreasePencilColorModifierData GreasePencilColorModifierData
@ eModifierType_GreasePencilColor
Read Guarded memory(de)allocation.
ModifierTypeInfo modifierType_GreasePencilColor
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 uiLayoutSetPropSep(uiLayout *layout, bool is_sep)
#define UI_ITEM_NONE
PanelLayout uiLayoutPanelProp(const bContext *C, uiLayout *layout, PointerRNA *open_prop_owner, const char *open_prop_name)
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
@ UI_ITEM_R_SLIDER
ChannelStorageType a
Definition BLI_color.hh:88
bke::CurvesGeometry & strokes_for_write()
const bke::CurvesGeometry & strokes() const
MutableSpan< ColorGeometry4f > fill_colors_for_write()
MutableSpan< ColorGeometry4f > vertex_colors_for_write()
void foreach_index(Fn &&fn) const
draw_view in_light_buf[] float
MINLINE float fractf(float a)
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_layer_mask(const GreasePencil &grease_pencil, const std::optional< StringRef > layer_name_filter, const std::optional< int > layer_pass_filter, const bool layer_filter_invert, const bool layer_pass_filter_invert, IndexMaskMemory &memory)
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)
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:58
static void copy_data(const ModifierData *md, ModifierData *target, const int flag)
static void apply_color_factor(ColorGeometry4f &color, const ColorGeometry4f &material_color, const float3 factor)
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 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_drawing(const GreasePencilArrayModifierData &mmd, const ModifierEvalContext &ctx, bke::greasepencil::Drawing &drawing)
static void modify_stroke_color(Object &ob, const GreasePencilColorModifierData &cmd, bke::CurvesGeometry &curves, const IndexMask &curves_mask, const MutableSpan< ColorGeometry4f > vertex_colors)
ColorSceneLinear4f< eAlpha::Premultiplied > ColorGeometry4f
Definition BLI_color.hh:337
static void blend_read(BlendDataReader *reader, ModifierData *md)
__int64 int64_t
Definition stdint.h:89
GreasePencilModifierInfluenceData influence
GreasePencilRuntimeHandle * runtime
Definition DNA_ID.h:413
struct MaterialGPencilStyle * gp_style
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