Blender V5.0
node_geo_merge_layers.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
6
8#include "GEO_merge_layers.hh"
9
10#include "BKE_grease_pencil.hh"
11
12#include "NOD_rna_define.hh"
13
15#include "UI_resources.hh"
16
18
20
21enum class MergeLayerMode {
22 ByName = 0,
23 ByID = 1,
24};
25
27{
28 b.use_custom_socket_order();
29 b.allow_any_socket_order();
30 b.add_default_layout();
31 b.add_input<decl::Geometry>("Grease Pencil")
32 .supported_type(GeometryComponent::Type::GreasePencil)
33 .description("Grease Pencil data to merge layers of");
34 b.add_output<decl::Geometry>("Grease Pencil").propagate_all().align_with_previous();
35 b.add_input<decl::Bool>("Selection").default_value(true).hide_value().field_on_all();
36 auto &group_id = b.add_input<decl::Int>("Group ID")
37 .hide_value()
38 .field_on_all()
39 .make_available([](bNode &node) {
40 node_storage(node).mode = int8_t(MergeLayerMode::ByID);
41 });
42
43 const bNode *node = b.node_or_null();
44 if (node) {
45 const NodeGeometryMergeLayers &storage = node_storage(*node);
46 const MergeLayerMode mode = MergeLayerMode(storage.mode);
47 group_id.available(mode == MergeLayerMode::ByID);
48 }
49}
50
51static void node_init(bNodeTree * /*tree*/, bNode *node)
52{
54 data->mode = int8_t(MergeLayerMode::ByName);
55 node->storage = data;
56}
57
58static void node_layout(uiLayout *layout, bContext * /*C*/, PointerRNA *ptr)
59{
60 layout->prop(ptr, "mode", UI_ITEM_NONE, "", ICON_NONE);
61}
62
65{
66 using namespace bke::greasepencil;
67
68 const int old_layers_num = src_grease_pencil.layers().size();
69
70 const Field<bool> selection_field = params.get_input<Field<bool>>("Selection");
71
72 bke::GreasePencilFieldContext field_context{src_grease_pencil};
73 FieldEvaluator field_evaluator{field_context, old_layers_num};
74 field_evaluator.add(selection_field);
75 field_evaluator.evaluate();
76 const VArray<bool> selection = field_evaluator.get_evaluated<bool>(0);
77
78 Vector<Vector<int>> layers_map;
79 Map<StringRef, int> new_layer_index_by_name;
80
81 for (const int layer_i : IndexRange(old_layers_num)) {
82 const bool is_selected = selection[layer_i];
83 if (!is_selected) {
84 layers_map.append({layer_i});
85 continue;
86 }
87
88 const Layer &layer = src_grease_pencil.layer(layer_i);
89 const int new_layer_index = new_layer_index_by_name.lookup_or_add_cb(
90 layer.name(), [&]() { return layers_map.append_and_get_index_as(); });
91 layers_map[new_layer_index].append(layer_i);
92 }
93 return layers_map;
94}
95
96static Vector<Vector<int>> get_layers_map_by_id(const GreasePencil &src_grease_pencil,
98{
99 using namespace bke::greasepencil;
100
101 const int old_layers_num = src_grease_pencil.layers().size();
102
103 const Field<bool> selection_field = params.get_input<Field<bool>>("Selection");
104 const Field<int> group_id_field = params.get_input<Field<int>>("Group ID");
105
106 bke::GreasePencilFieldContext field_context{src_grease_pencil};
107 FieldEvaluator field_evaluator{field_context, old_layers_num};
108 field_evaluator.add(selection_field);
109 field_evaluator.add(group_id_field);
110 field_evaluator.evaluate();
111 const VArray<bool> selection = field_evaluator.get_evaluated<bool>(0);
112 const VArray<int> group_ids = field_evaluator.get_evaluated<int>(1);
113
114 Vector<Vector<int>> layers_map;
115 Map<int, int> new_layer_index_by_id;
116
117 for (const int layer_i : IndexRange(old_layers_num)) {
118 const bool is_selected = selection[layer_i];
119 if (!is_selected) {
120 layers_map.append({layer_i});
121 continue;
122 }
123 const int group_id = group_ids[layer_i];
124 const int new_layer_index = new_layer_index_by_id.lookup_or_add_cb(
125 group_id, [&]() { return layers_map.append_and_get_index_as(); });
126 layers_map[new_layer_index].append(layer_i);
127 }
128 return layers_map;
129}
130
132 const NodeGeometryMergeLayers &storage,
134 const AttributeFilter &attribute_filter)
135{
136 using namespace bke::greasepencil;
137
138 const GreasePencil *src_grease_pencil = geometry.get_grease_pencil();
139 if (!src_grease_pencil) {
140 return;
141 }
142 const int old_layers_num = src_grease_pencil->layers().size();
143
144 Vector<Vector<int>> layers_map;
145 switch (MergeLayerMode(storage.mode)) {
147 layers_map = get_layers_map_by_name(*src_grease_pencil, params);
148 break;
149 }
151 layers_map = get_layers_map_by_id(*src_grease_pencil, params);
152 break;
153 }
154 }
155
156 const int new_layers_num = layers_map.size();
157 if (old_layers_num == new_layers_num) {
158 return;
159 }
160
161 GreasePencil *new_grease_pencil = geometry::merge_layers(
162 *src_grease_pencil, layers_map, attribute_filter);
163 geometry.replace_grease_pencil(new_grease_pencil);
164}
165
167{
168 GeometrySet main_geometry = params.extract_input<GeometrySet>("Grease Pencil");
169 const bNode &node = params.node();
170 const NodeGeometryMergeLayers &storage = node_storage(node);
171
172 const NodeAttributeFilter attribute_filter = params.get_attribute_filter("Grease Pencil");
173
175 merge_layers(geometry, storage, params, attribute_filter);
176 });
177
178 params.set_output("Grease Pencil", std::move(main_geometry));
179}
180
181static void node_rna(StructRNA *srna)
182{
183 static const EnumPropertyItem mode_items[] = {
185 "MERGE_BY_NAME",
186 0,
187 "By Name",
188 "Combine all layers which have the same name"},
190 "MERGE_BY_ID",
191 0,
192 "By Group ID",
193 "Provide a custom group ID for each layer and all layers with the same ID will be merged "
194 "into one"},
195 {0, nullptr, 0, nullptr, nullptr},
196 };
197
199 "mode",
200 "Mode",
201 "Determines how to choose which layers are merged",
202 mode_items,
205 nullptr);
206}
207
208static void node_register()
209{
210 static blender::bke::bNodeType ntype;
211
212 geo_node_type_base(&ntype, "GeometryNodeMergeLayers", GEO_NODE_MERGE_LAYERS);
213 ntype.ui_name = "Merge Layers";
214 ntype.ui_description = "Join groups of Grease Pencil layers into one";
215 ntype.enum_name_legacy = "MERGE_LAYERS";
217 ntype.declare = node_declare;
218 ntype.initfunc = node_init;
222 ntype, "NodeGeometryMergeLayers", node_free_standard_storage, node_copy_standard_storage);
224
225 node_rna(ntype.rna_ext.srna);
226}
228
229} // namespace blender::nodes::node_geo_merge_layers_cc
Low-level operations for grease pencil.
#define NODE_STORAGE_FUNCS(StorageT)
Definition BKE_node.hh:1240
#define NODE_CLASS_GEOMETRY
Definition BKE_node.hh:461
#define GEO_NODE_MERGE_LAYERS
#define NOD_REGISTER_NODE(REGISTER_FUNC)
#define NOD_storage_enum_accessors(member)
#define UI_ITEM_NONE
BMesh const char void * data
Value & lookup_or_add_cb(const Key &key, const CreateValueF &create_value)
Definition BLI_map.hh:620
int64_t size() const
int add(GField field, GVArray *varray_ptr)
Definition field.cc:751
const GVArray & get_evaluated(const int field_index) const
Definition FN_field.hh:448
void make_available(bNode &node) const
uiWidgetBaseParameters params[MAX_WIDGET_BASE_BATCH]
void * MEM_callocN(size_t len, const char *str)
Definition mallocn.cc:118
void node_register_type(bNodeType &ntype)
Definition node.cc:2416
void node_type_storage(bNodeType &ntype, std::optional< StringRefNull > storagename, void(*freefunc)(bNode *node), void(*copyfunc)(bNodeTree *dest_ntree, bNode *dest_node, const bNode *src_node))
Definition node.cc:5414
GreasePencil * merge_layers(const GreasePencil &src_grease_pencil, Span< Vector< int > > layers_to_merge, const bke::AttributeFilter &attribute_filter)
void foreach_real_geometry(bke::GeometrySet &geometry, FunctionRef< void(bke::GeometrySet &geometry_set)> fn)
static Vector< Vector< int > > get_layers_map_by_name(const GreasePencil &src_grease_pencil, const GeoNodeExecParams &params)
static void merge_layers(GeometrySet &geometry, const NodeGeometryMergeLayers &storage, const GeoNodeExecParams &params, const AttributeFilter &attribute_filter)
static void node_init(bNodeTree *, bNode *node)
static void node_geo_exec(GeoNodeExecParams params)
static void node_declare(NodeDeclarationBuilder &b)
static Vector< Vector< int > > get_layers_map_by_id(const GreasePencil &src_grease_pencil, const GeoNodeExecParams &params)
static void node_layout(uiLayout *layout, bContext *, PointerRNA *ptr)
PropertyRNA * RNA_def_node_enum(StructRNA *srna, const char *identifier, const char *ui_name, const char *ui_description, const EnumPropertyItem *static_items, const EnumRNAAccessors accessors, std::optional< int > default_value, const EnumPropertyItemFunc item_func, const bool allow_animation)
void geo_node_type_base(blender::bke::bNodeType *ntype, std::string idname, const std::optional< int16_t > legacy_type)
void node_free_standard_storage(bNode *node)
Definition node_util.cc:42
void node_copy_standard_storage(bNodeTree *, bNode *dest_node, const bNode *src_node)
Definition node_util.cc:54
StructRNA * srna
void * storage
Defines a node type.
Definition BKE_node.hh:238
std::string ui_description
Definition BKE_node.hh:244
void(* initfunc)(bNodeTree *ntree, bNode *node)
Definition BKE_node.hh:289
NodeGeometryExecFunction geometry_node_execute
Definition BKE_node.hh:354
const char * enum_name_legacy
Definition BKE_node.hh:247
void(* draw_buttons)(uiLayout *, bContext *C, PointerRNA *ptr)
Definition BKE_node.hh:259
NodeDeclareFunction declare
Definition BKE_node.hh:362
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)
PointerRNA * ptr
Definition wm_files.cc:4238