26 const bNode *node =
b.node_or_null();
33 if (node !=
nullptr) {
35 b.add_input(data_type,
"Value").hide_value().field_on_all();
38 "Which element to retrieve a value from on the geometry");
40 if (node !=
nullptr) {
42 b.add_output(data_type,
"Value").dependent_field({2});
57 data->domain =
int8_t(AttrDomain::Point);
72 bNode &node =
params.add_node(
"GeometryNodeSampleIndex");
73 node_storage(node).data_type = *type;
74 params.update_and_connect_available_socket(node,
"Value");
81 const AttrDomain domain)
83 if (!geometry.has(type)) {
91 const AttrDomain domain)
103 return geometry.get_component(src_type);
118 mask.foreach_index(
GrainSize(4096), [&](
const int i) {
119 const int index = indices[i];
120 dst[i] = src[std::clamp(index, 0, last_index)];
138 std::optional<bke::GeometryFieldContext> geometry_context_;
139 std::unique_ptr<FieldEvaluator> evaluator_;
140 const GVArray *src_data_ =
nullptr;
145 const AttrDomain domain,
147 : src_geometry_(std::move(geometry)),
148 src_field_(std::move(src_field)),
155 builder.single_input<
int>(
"Index");
156 builder.single_output(
"Value", src_field_.
cpp_type());
165 if (component ==
nullptr) {
170 evaluator_ = std::make_unique<FieldEvaluator>(*geometry_context_, domain_num);
171 evaluator_->add(src_field_);
172 evaluator_->evaluate();
173 src_data_ = &evaluator_->get_evaluated(0);
181 const CPPType &type = dst.type();
182 if (src_data_ ==
nullptr) {
189 using T =
decltype(dummy);
203 const AttrDomain domain = AttrDomain(storage.domain);
204 const bool use_clamp = bool(storage.clamp);
210 if (index_value_variant.is_context_dependent_field()) {
212 auto fn = std::make_shared<SampleIndexFunction>(
213 std::move(geometry), std::move(value_field), domain, use_clamp);
220 const int domain_size = component->attribute_domain_size(domain);
221 int index = index_value_variant.extract<
int>();
223 index = std::clamp(index, 0, domain_size - 1);
225 if (index >= 0 && index < domain_size) {
229 evaluator.
add(value_field);
233 data.get_to_uninitialized(index, buffer);
235 cpp_type.destruct(buffer);
#define NODE_STORAGE_FUNCS(StorageT)
#define NODE_CLASS_GEOMETRY
#define BUFFER_FOR_CPP_TYPE_VALUE(type, variable_name)
#define NOD_REGISTER_NODE(REGISTER_FUNC)
void uiItemR(uiLayout *layout, PointerRNA *ptr, const char *propname, eUI_Item_Flag flag, const char *name, int icon)
void value_initialize_indices(void *ptr, const IndexMask &mask) const
VArray< T > typed() const
constexpr int64_t last(const int64_t n=0) const
IndexRange index_range() const
int attribute_domain_size(AttrDomain domain) const
int add(GField field, GVArray *varray_ptr)
const GVArray & get_evaluated(const int field_index) const
static std::shared_ptr< FieldOperation > Create(std::shared_ptr< const mf::MultiFunction > function, Vector< GField > inputs={})
const CPPType & cpp_type() const
void set_signature(const Signature *signature)
Vector< SocketDeclaration * > inputs
SampleIndexFunction(GeometrySet geometry, GField src_field, const AttrDomain domain, const bool clamp)
void call(const IndexMask &mask, mf::Params params, mf::Context) const override
local_group_size(16, 16) .push_constant(Type b
ccl_device_inline float4 mask(const int4 mask, const float4 a)
void convert_to_static_type(const CPPType &cpp_type, const Func &func)
void copy_with_checked_indices(const GVArray &src, const VArray< int > &indices, const IndexMask &mask, GMutableSpan dst)
void node_type_storage(bNodeType *ntype, const char *storagename, void(*freefunc)(bNode *node), void(*copyfunc)(bNodeTree *dest_ntree, bNode *dest_node, const bNode *src_node))
void node_register_type(bNodeType *ntype)
std::optional< eCustomDataType > socket_type_to_custom_data_type(eNodeSocketDatatype type)
GField make_constant_field(const CPPType &type, const void *value)
static void node_layout(uiLayout *layout, bContext *, PointerRNA *ptr)
static const GeometryComponent * find_source_component(const GeometrySet &geometry, const AttrDomain domain)
static void node_geo_exec(GeoNodeExecParams params)
static void node_declare(NodeDeclarationBuilder &b)
static void node_register()
static void node_gather_link_searches(GatherLinkSearchOpParams ¶ms)
void copy_with_clamped_indices(const VArray< T > &src, const VArray< int > &indices, const IndexMask &mask, MutableSpan< T > dst)
static bool component_is_available(const GeometrySet &geometry, const GeometryComponent::Type type, const AttrDomain domain)
static void node_init(bNodeTree *, bNode *node)
void search_link_ops_for_declarations(GatherLinkSearchOpParams ¶ms, Span< SocketDeclaration * > declarations)
void devirtualize_varray2(const VArray< T1 > &varray1, const VArray< T2 > &varray2, const Func &func, bool enable=true)
void geo_node_type_base(blender::bke::bNodeType *ntype, int type, const char *name, short nclass)
void node_free_standard_storage(bNode *node)
void node_copy_standard_storage(bNodeTree *, bNode *dest_node, const bNode *src_node)
void ensure_owns_direct_data()
void(* initfunc)(bNodeTree *ntree, bNode *node)
NodeGeometryExecFunction geometry_node_execute
void(* draw_buttons)(uiLayout *, bContext *C, PointerRNA *ptr)
NodeGatherSocketLinkOperationsFunction gather_link_search_ops
NodeDeclareFunction declare