Painting after converting active color attribute in sculpt mode causes crash.
Fix is same as 3641b4b884 with some cleanup
Pull Request: https://projects.blender.org/blender/blender/pulls/119780
975 lines
30 KiB
C++
975 lines
30 KiB
C++
/* SPDX-FileCopyrightText: 2006 Blender Authors
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
/** \file
|
|
* \ingroup bke
|
|
* Implementation of generic geometry attributes management. This is built
|
|
* on top of CustomData, which manages individual domains.
|
|
*/
|
|
|
|
#include <cstring>
|
|
#include <optional>
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "DNA_ID.h"
|
|
#include "DNA_curves_types.h"
|
|
#include "DNA_customdata_types.h"
|
|
#include "DNA_mesh_types.h"
|
|
#include "DNA_pointcloud_types.h"
|
|
|
|
#include "BLI_index_range.hh"
|
|
#include "BLI_string.h"
|
|
#include "BLI_string_utf8.h"
|
|
#include "BLI_string_utils.hh"
|
|
|
|
#include "BLT_translation.hh"
|
|
|
|
#include "BKE_attribute.hh"
|
|
#include "BKE_curves.hh"
|
|
#include "BKE_customdata.hh"
|
|
#include "BKE_editmesh.hh"
|
|
#include "BKE_grease_pencil.hh"
|
|
#include "BKE_mesh.hh"
|
|
#include "BKE_pointcloud.hh"
|
|
#include "BKE_report.hh"
|
|
|
|
#include "RNA_access.hh"
|
|
|
|
using blender::IndexRange;
|
|
using blender::bke::AttrDomain;
|
|
|
|
struct DomainInfo {
|
|
CustomData *customdata;
|
|
int length;
|
|
};
|
|
|
|
static void get_domains(const ID *id, DomainInfo info[ATTR_DOMAIN_NUM])
|
|
{
|
|
memset(info, 0, sizeof(DomainInfo) * ATTR_DOMAIN_NUM);
|
|
|
|
switch (GS(id->name)) {
|
|
case ID_PT: {
|
|
PointCloud *pointcloud = (PointCloud *)id;
|
|
info[int(AttrDomain::Point)].customdata = &pointcloud->pdata;
|
|
info[int(AttrDomain::Point)].length = pointcloud->totpoint;
|
|
break;
|
|
}
|
|
case ID_ME: {
|
|
Mesh *mesh = (Mesh *)id;
|
|
BMEditMesh *em = mesh->runtime->edit_mesh;
|
|
if (em != nullptr) {
|
|
BMesh *bm = em->bm;
|
|
info[int(AttrDomain::Point)].customdata = &bm->vdata;
|
|
info[int(AttrDomain::Point)].length = bm->totvert;
|
|
info[int(AttrDomain::Edge)].customdata = &bm->edata;
|
|
info[int(AttrDomain::Edge)].length = bm->totedge;
|
|
info[int(AttrDomain::Corner)].customdata = &bm->ldata;
|
|
info[int(AttrDomain::Corner)].length = bm->totloop;
|
|
info[int(AttrDomain::Face)].customdata = &bm->pdata;
|
|
info[int(AttrDomain::Face)].length = bm->totface;
|
|
}
|
|
else {
|
|
info[int(AttrDomain::Point)].customdata = &mesh->vert_data;
|
|
info[int(AttrDomain::Point)].length = mesh->verts_num;
|
|
info[int(AttrDomain::Edge)].customdata = &mesh->edge_data;
|
|
info[int(AttrDomain::Edge)].length = mesh->edges_num;
|
|
info[int(AttrDomain::Corner)].customdata = &mesh->corner_data;
|
|
info[int(AttrDomain::Corner)].length = mesh->corners_num;
|
|
info[int(AttrDomain::Face)].customdata = &mesh->face_data;
|
|
info[int(AttrDomain::Face)].length = mesh->faces_num;
|
|
}
|
|
break;
|
|
}
|
|
case ID_CV: {
|
|
Curves *curves = (Curves *)id;
|
|
info[int(AttrDomain::Point)].customdata = &curves->geometry.point_data;
|
|
info[int(AttrDomain::Point)].length = curves->geometry.point_num;
|
|
info[int(AttrDomain::Curve)].customdata = &curves->geometry.curve_data;
|
|
info[int(AttrDomain::Curve)].length = curves->geometry.curve_num;
|
|
break;
|
|
}
|
|
case ID_GP: {
|
|
GreasePencil *grease_pencil = (GreasePencil *)id;
|
|
info[int(AttrDomain::Layer)].customdata = &grease_pencil->layers_data;
|
|
info[int(AttrDomain::Layer)].length = grease_pencil->layers().size();
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
namespace blender::bke {
|
|
|
|
static std::optional<blender::bke::MutableAttributeAccessor> get_attribute_accessor_for_write(
|
|
ID &id)
|
|
{
|
|
switch (GS(id.name)) {
|
|
case ID_ME: {
|
|
Mesh &mesh = reinterpret_cast<Mesh &>(id);
|
|
/* The attribute API isn't implemented for BMesh, so edit mode meshes are not supported. */
|
|
BLI_assert(mesh.runtime->edit_mesh == nullptr);
|
|
return mesh.attributes_for_write();
|
|
}
|
|
case ID_PT: {
|
|
PointCloud &pointcloud = reinterpret_cast<PointCloud &>(id);
|
|
return pointcloud.attributes_for_write();
|
|
}
|
|
case ID_CV: {
|
|
Curves &curves_id = reinterpret_cast<Curves &>(id);
|
|
CurvesGeometry &curves = curves_id.geometry.wrap();
|
|
return curves.attributes_for_write();
|
|
}
|
|
case ID_GP: {
|
|
GreasePencil &grease_pencil = reinterpret_cast<GreasePencil &>(id);
|
|
return grease_pencil.attributes_for_write();
|
|
}
|
|
default: {
|
|
BLI_assert_unreachable();
|
|
return {};
|
|
}
|
|
}
|
|
}
|
|
|
|
} // namespace blender::bke
|
|
|
|
bool BKE_id_attributes_supported(const ID *id)
|
|
{
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
if (info[domain].customdata) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
bool BKE_attribute_allow_procedural_access(const char *attribute_name)
|
|
{
|
|
return blender::bke::allow_procedural_attribute_access(attribute_name);
|
|
}
|
|
|
|
static bool bke_id_attribute_rename_if_exists(ID *id,
|
|
const char *old_name,
|
|
const char *new_name,
|
|
ReportList *reports)
|
|
{
|
|
CustomDataLayer *layer = BKE_id_attribute_search_for_write(
|
|
id, old_name, CD_MASK_PROP_ALL, ATTR_DOMAIN_MASK_ALL);
|
|
if (layer == nullptr) {
|
|
return false;
|
|
}
|
|
return BKE_id_attribute_rename(id, old_name, new_name, reports);
|
|
}
|
|
|
|
static bool mesh_edit_mode_attribute_valid(const blender::StringRef name,
|
|
const AttrDomain domain,
|
|
const eCustomDataType data_type,
|
|
ReportList *reports)
|
|
{
|
|
if (ELEM(name,
|
|
"position",
|
|
".edge_verts",
|
|
".corner_vert",
|
|
".corner_edge",
|
|
"sharp_edge",
|
|
"sharp_face",
|
|
"material_index"))
|
|
{
|
|
BKE_report(reports, RPT_ERROR, "Unable to create builtin attribute in edit mode");
|
|
return false;
|
|
}
|
|
if (name == "id") {
|
|
if (domain != AttrDomain::Point) {
|
|
BKE_report(reports, RPT_ERROR, "Domain unsupported for \"id\" attribute");
|
|
return false;
|
|
}
|
|
if (data_type != CD_PROP_INT32) {
|
|
BKE_report(reports, RPT_ERROR, "Type unsupported for \"id\" attribute");
|
|
return false;
|
|
}
|
|
}
|
|
return true;
|
|
}
|
|
|
|
bool BKE_id_attribute_rename(ID *id,
|
|
const char *old_name,
|
|
const char *new_name,
|
|
ReportList *reports)
|
|
{
|
|
using namespace blender;
|
|
if (BKE_id_attribute_required(id, old_name)) {
|
|
BLI_assert_msg(0, "Required attribute name is not editable");
|
|
return false;
|
|
}
|
|
if (STREQ(new_name, "")) {
|
|
BKE_report(reports, RPT_ERROR, "Attribute name cannot be empty");
|
|
return false;
|
|
}
|
|
|
|
/* NOTE: Checking if the new name matches the old name only makes sense when the name
|
|
* is clamped to it's maximum length, otherwise assigning an over-long name multiple times
|
|
* will add `.001` suffix unnecessarily. */
|
|
{
|
|
const int new_name_maxncpy = CustomData_name_maxncpy_calc(new_name);
|
|
/* NOTE: A function that performs a clamped comparison without copying would be handy here. */
|
|
char new_name_clamped[MAX_CUSTOMDATA_LAYER_NAME];
|
|
BLI_strncpy_utf8(new_name_clamped, new_name, new_name_maxncpy);
|
|
if (STREQ(old_name, new_name_clamped)) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
CustomDataLayer *layer = BKE_id_attribute_search_for_write(
|
|
id, old_name, CD_MASK_PROP_ALL, ATTR_DOMAIN_MASK_ALL);
|
|
if (layer == nullptr) {
|
|
BKE_report(reports, RPT_ERROR, "Attribute is not part of this geometry");
|
|
return false;
|
|
}
|
|
|
|
if (GS(id->name) == ID_ME) {
|
|
Mesh *mesh = reinterpret_cast<Mesh *>(id);
|
|
if (mesh->runtime->edit_mesh) {
|
|
if (!mesh_edit_mode_attribute_valid(
|
|
new_name, BKE_id_attribute_domain(id, layer), eCustomDataType(layer->type), reports))
|
|
{
|
|
return false;
|
|
}
|
|
}
|
|
}
|
|
|
|
std::string result_name = BKE_id_attribute_calc_unique_name(*id, new_name);
|
|
|
|
if (layer->type == CD_PROP_FLOAT2 && GS(id->name) == ID_ME) {
|
|
/* Rename UV sub-attributes. */
|
|
char buffer_src[MAX_CUSTOMDATA_LAYER_NAME];
|
|
char buffer_dst[MAX_CUSTOMDATA_LAYER_NAME];
|
|
|
|
bke_id_attribute_rename_if_exists(
|
|
id,
|
|
BKE_uv_map_vert_select_name_get(layer->name, buffer_src),
|
|
BKE_uv_map_vert_select_name_get(result_name.c_str(), buffer_dst),
|
|
reports);
|
|
bke_id_attribute_rename_if_exists(
|
|
id,
|
|
BKE_uv_map_edge_select_name_get(layer->name, buffer_src),
|
|
BKE_uv_map_edge_select_name_get(result_name.c_str(), buffer_dst),
|
|
reports);
|
|
bke_id_attribute_rename_if_exists(id,
|
|
BKE_uv_map_pin_name_get(layer->name, buffer_src),
|
|
BKE_uv_map_pin_name_get(result_name.c_str(), buffer_dst),
|
|
reports);
|
|
}
|
|
if (StringRef(old_name) == BKE_id_attributes_active_color_name(id)) {
|
|
BKE_id_attributes_active_color_set(id, result_name.c_str());
|
|
}
|
|
if (StringRef(old_name) == BKE_id_attributes_default_color_name(id)) {
|
|
BKE_id_attributes_default_color_set(id, result_name.c_str());
|
|
}
|
|
|
|
STRNCPY_UTF8(layer->name, result_name.c_str());
|
|
|
|
return true;
|
|
}
|
|
|
|
static bool attribute_name_exists(const ID &id, const blender::StringRef name)
|
|
{
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(&id, info);
|
|
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
if (!info[domain].customdata) {
|
|
continue;
|
|
}
|
|
|
|
const CustomData *cdata = info[domain].customdata;
|
|
for (int i = 0; i < cdata->totlayer; i++) {
|
|
const CustomDataLayer *layer = cdata->layers + i;
|
|
|
|
if (layer->name == name) {
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
std::string BKE_id_attribute_calc_unique_name(const ID &id, const blender::StringRef name)
|
|
{
|
|
return BLI_uniquename_cb(
|
|
[&](const blender::StringRef new_name) { return attribute_name_exists(id, new_name); },
|
|
'.',
|
|
name.is_empty() ? DATA_("Attribute") : name);
|
|
}
|
|
|
|
CustomDataLayer *BKE_id_attribute_new(ID *id,
|
|
const char *name,
|
|
const eCustomDataType type,
|
|
const AttrDomain domain,
|
|
ReportList *reports)
|
|
{
|
|
using namespace blender::bke;
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
CustomData *customdata = info[int(domain)].customdata;
|
|
if (customdata == nullptr) {
|
|
BKE_report(reports, RPT_ERROR, "Attribute domain not supported by this geometry type");
|
|
return nullptr;
|
|
}
|
|
|
|
std::string uniquename = BKE_id_attribute_calc_unique_name(*id, name);
|
|
|
|
if (GS(id->name) == ID_ME) {
|
|
Mesh *mesh = reinterpret_cast<Mesh *>(id);
|
|
if (BMEditMesh *em = mesh->runtime->edit_mesh) {
|
|
if (!mesh_edit_mode_attribute_valid(name, domain, type, reports)) {
|
|
return nullptr;
|
|
}
|
|
BM_data_layer_add_named(em->bm, customdata, type, uniquename.c_str());
|
|
const int index = CustomData_get_named_layer_index(customdata, type, uniquename);
|
|
return (index == -1) ? nullptr : &(customdata->layers[index]);
|
|
}
|
|
}
|
|
|
|
std::optional<MutableAttributeAccessor> attributes = get_attribute_accessor_for_write(*id);
|
|
if (!attributes) {
|
|
return nullptr;
|
|
}
|
|
|
|
attributes->add(uniquename, domain, eCustomDataType(type), AttributeInitDefaultValue());
|
|
|
|
const int index = CustomData_get_named_layer_index(customdata, type, uniquename);
|
|
if (index == -1) {
|
|
BKE_reportf(reports, RPT_WARNING, "Layer '%s' could not be created", uniquename.c_str());
|
|
}
|
|
|
|
return (index == -1) ? nullptr : &(customdata->layers[index]);
|
|
}
|
|
|
|
static void bke_id_attribute_copy_if_exists(ID *id, const char *srcname, const char *dstname)
|
|
{
|
|
using namespace blender::bke;
|
|
|
|
std::optional<MutableAttributeAccessor> attributes = get_attribute_accessor_for_write(*id);
|
|
if (!attributes) {
|
|
return;
|
|
}
|
|
|
|
GAttributeReader src = attributes->lookup(srcname);
|
|
if (!src) {
|
|
return;
|
|
}
|
|
|
|
const eCustomDataType type = cpp_type_to_custom_data_type(src.varray.type());
|
|
attributes->add(dstname, src.domain, type, AttributeInitVArray(src.varray));
|
|
}
|
|
|
|
CustomDataLayer *BKE_id_attribute_duplicate(ID *id, const char *name, ReportList *reports)
|
|
{
|
|
using namespace blender::bke;
|
|
std::string uniquename = BKE_id_attribute_calc_unique_name(*id, name);
|
|
|
|
if (GS(id->name) == ID_ME) {
|
|
Mesh *mesh = reinterpret_cast<Mesh *>(id);
|
|
if (BMEditMesh *em = mesh->runtime->edit_mesh) {
|
|
BLI_assert_unreachable();
|
|
UNUSED_VARS(em);
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
std::optional<MutableAttributeAccessor> attributes = get_attribute_accessor_for_write(*id);
|
|
if (!attributes) {
|
|
return nullptr;
|
|
}
|
|
|
|
GAttributeReader src = attributes->lookup(name);
|
|
if (!src) {
|
|
BKE_report(reports, RPT_ERROR, "Attribute is not part of this geometry");
|
|
return nullptr;
|
|
}
|
|
|
|
const eCustomDataType type = cpp_type_to_custom_data_type(src.varray.type());
|
|
attributes->add(uniquename, src.domain, type, AttributeInitVArray(src.varray));
|
|
|
|
if (GS(id->name) == ID_ME && type == CD_PROP_FLOAT2) {
|
|
/* Duplicate UV sub-attributes. */
|
|
char buffer_src[MAX_CUSTOMDATA_LAYER_NAME];
|
|
char buffer_dst[MAX_CUSTOMDATA_LAYER_NAME];
|
|
|
|
bke_id_attribute_copy_if_exists(
|
|
id,
|
|
BKE_uv_map_vert_select_name_get(name, buffer_src),
|
|
BKE_uv_map_vert_select_name_get(uniquename.c_str(), buffer_dst));
|
|
bke_id_attribute_copy_if_exists(
|
|
id,
|
|
BKE_uv_map_edge_select_name_get(name, buffer_src),
|
|
BKE_uv_map_edge_select_name_get(uniquename.c_str(), buffer_dst));
|
|
bke_id_attribute_copy_if_exists(id,
|
|
BKE_uv_map_pin_name_get(name, buffer_src),
|
|
BKE_uv_map_pin_name_get(uniquename.c_str(), buffer_dst));
|
|
}
|
|
|
|
return BKE_id_attribute_search_for_write(
|
|
id, uniquename.c_str(), CD_MASK_PROP_ALL, ATTR_DOMAIN_MASK_ALL);
|
|
}
|
|
|
|
static int color_name_to_index(ID *id, const char *name)
|
|
{
|
|
const CustomDataLayer *layer = BKE_id_attribute_search(
|
|
id, name, CD_MASK_COLOR_ALL, ATTR_DOMAIN_MASK_COLOR);
|
|
return BKE_id_attribute_to_index(id, layer, ATTR_DOMAIN_MASK_COLOR, CD_MASK_COLOR_ALL);
|
|
}
|
|
|
|
static int color_clamp_index(ID *id, int index)
|
|
{
|
|
const int length = BKE_id_attributes_length(id, ATTR_DOMAIN_MASK_COLOR, CD_MASK_COLOR_ALL);
|
|
return min_ii(index, length - 1);
|
|
}
|
|
|
|
static const char *color_name_from_index(ID *id, int index)
|
|
{
|
|
const CustomDataLayer *layer = BKE_id_attribute_from_index(
|
|
id, index, ATTR_DOMAIN_MASK_COLOR, CD_MASK_COLOR_ALL);
|
|
return layer ? layer->name : nullptr;
|
|
}
|
|
|
|
bool BKE_id_attribute_remove(ID *id, const char *name, ReportList *reports)
|
|
{
|
|
using namespace blender;
|
|
using namespace blender::bke;
|
|
if (!name || name[0] == '\0') {
|
|
BKE_report(reports, RPT_ERROR, "The attribute name must not be empty");
|
|
return false;
|
|
}
|
|
if (BKE_id_attribute_required(id, name)) {
|
|
BKE_report(reports, RPT_ERROR, "Attribute is required and can't be removed");
|
|
return false;
|
|
}
|
|
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
if (GS(id->name) == ID_ME) {
|
|
Mesh *mesh = reinterpret_cast<Mesh *>(id);
|
|
if (BMEditMesh *em = mesh->runtime->edit_mesh) {
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
if (CustomData *data = info[domain].customdata) {
|
|
const std::string name_copy = name;
|
|
const int layer_index = CustomData_get_named_layer_index_notype(data, name_copy);
|
|
if (layer_index == -1) {
|
|
continue;
|
|
}
|
|
|
|
const eCustomDataType type = eCustomDataType(data->layers[layer_index].type);
|
|
const bool is_active_color_attribute = name_copy.c_str() ==
|
|
StringRef(mesh->active_color_attribute);
|
|
const bool is_default_color_attribute = name_copy.c_str() ==
|
|
StringRef(mesh->default_color_attribute);
|
|
const int active_color_index = color_name_to_index(id, mesh->active_color_attribute);
|
|
const int default_color_index = color_name_to_index(id, mesh->default_color_attribute);
|
|
|
|
if (!BM_data_layer_free_named(em->bm, data, name_copy.c_str())) {
|
|
BLI_assert_unreachable();
|
|
}
|
|
|
|
if (is_active_color_attribute) {
|
|
BKE_id_attributes_active_color_set(
|
|
id, color_name_from_index(id, color_clamp_index(id, active_color_index)));
|
|
}
|
|
if (is_default_color_attribute) {
|
|
BKE_id_attributes_default_color_set(
|
|
id, color_name_from_index(id, color_clamp_index(id, default_color_index)));
|
|
}
|
|
|
|
if (type == CD_PROP_FLOAT2 && domain == int(AttrDomain::Corner)) {
|
|
char buffer[MAX_CUSTOMDATA_LAYER_NAME];
|
|
BM_data_layer_free_named(
|
|
em->bm, data, BKE_uv_map_vert_select_name_get(name_copy.c_str(), buffer));
|
|
BM_data_layer_free_named(
|
|
em->bm, data, BKE_uv_map_edge_select_name_get(name_copy.c_str(), buffer));
|
|
BM_data_layer_free_named(
|
|
em->bm, data, BKE_uv_map_pin_name_get(name_copy.c_str(), buffer));
|
|
}
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
|
|
std::optional<MutableAttributeAccessor> attributes = get_attribute_accessor_for_write(*id);
|
|
if (!attributes) {
|
|
return false;
|
|
}
|
|
|
|
if (GS(id->name) == ID_ME) {
|
|
const std::string name_copy = name;
|
|
std::optional<blender::bke::AttributeMetaData> metadata = attributes->lookup_meta_data(
|
|
name_copy);
|
|
if (!metadata) {
|
|
return false;
|
|
}
|
|
/* Update active and default color attributes. */
|
|
Mesh *mesh = reinterpret_cast<Mesh *>(id);
|
|
const bool is_active_color_attribute = name_copy == StringRef(mesh->active_color_attribute);
|
|
const bool is_default_color_attribute = name_copy == StringRef(mesh->default_color_attribute);
|
|
const int active_color_index = color_name_to_index(id, mesh->active_color_attribute);
|
|
const int default_color_index = color_name_to_index(id, mesh->default_color_attribute);
|
|
|
|
if (!attributes->remove(name_copy)) {
|
|
BLI_assert_unreachable();
|
|
}
|
|
|
|
if (is_active_color_attribute) {
|
|
BKE_id_attributes_active_color_set(
|
|
id, color_name_from_index(id, color_clamp_index(id, active_color_index)));
|
|
}
|
|
if (is_default_color_attribute) {
|
|
BKE_id_attributes_default_color_set(
|
|
id, color_name_from_index(id, color_clamp_index(id, default_color_index)));
|
|
}
|
|
|
|
if (metadata->data_type == CD_PROP_FLOAT2 && metadata->domain == AttrDomain::Corner) {
|
|
char buffer[MAX_CUSTOMDATA_LAYER_NAME];
|
|
attributes->remove(BKE_uv_map_vert_select_name_get(name_copy.c_str(), buffer));
|
|
attributes->remove(BKE_uv_map_edge_select_name_get(name_copy.c_str(), buffer));
|
|
attributes->remove(BKE_uv_map_pin_name_get(name_copy.c_str(), buffer));
|
|
}
|
|
return true;
|
|
}
|
|
|
|
return attributes->remove(name);
|
|
}
|
|
|
|
CustomDataLayer *BKE_id_attribute_find(const ID *id,
|
|
const char *name,
|
|
const eCustomDataType type,
|
|
const AttrDomain domain)
|
|
{
|
|
if (!name) {
|
|
return nullptr;
|
|
}
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
CustomData *customdata = info[int(domain)].customdata;
|
|
if (customdata == nullptr) {
|
|
return nullptr;
|
|
}
|
|
|
|
for (int i = 0; i < customdata->totlayer; i++) {
|
|
CustomDataLayer *layer = &customdata->layers[i];
|
|
if (layer->type == type && STREQ(layer->name, name)) {
|
|
return layer;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
const CustomDataLayer *BKE_id_attribute_search(const ID *id,
|
|
const char *name,
|
|
const eCustomDataMask type_mask,
|
|
const AttrDomainMask domain_mask)
|
|
{
|
|
if (!name) {
|
|
return nullptr;
|
|
}
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
for (AttrDomain domain = AttrDomain::Point; int(domain) < ATTR_DOMAIN_NUM;
|
|
domain = AttrDomain(int(domain) + 1))
|
|
{
|
|
if (!(domain_mask & ATTR_DOMAIN_AS_MASK(domain))) {
|
|
continue;
|
|
}
|
|
|
|
CustomData *customdata = info[int(domain)].customdata;
|
|
if (customdata == nullptr) {
|
|
continue;
|
|
}
|
|
|
|
for (int i = 0; i < customdata->totlayer; i++) {
|
|
CustomDataLayer *layer = &customdata->layers[i];
|
|
if ((CD_TYPE_AS_MASK(layer->type) & type_mask) && STREQ(layer->name, name)) {
|
|
return layer;
|
|
}
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
CustomDataLayer *BKE_id_attribute_search_for_write(ID *id,
|
|
const char *name,
|
|
const eCustomDataMask type_mask,
|
|
const AttrDomainMask domain_mask)
|
|
{
|
|
/* Reuse the implementation of the const version.
|
|
* Implicit sharing for the layer's data is handled below. */
|
|
CustomDataLayer *layer = const_cast<CustomDataLayer *>(
|
|
BKE_id_attribute_search(id, name, type_mask, domain_mask));
|
|
if (!layer) {
|
|
return nullptr;
|
|
}
|
|
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
const AttrDomain domain = BKE_id_attribute_domain(id, layer);
|
|
CustomData_ensure_data_is_mutable(layer, info[int(domain)].length);
|
|
|
|
return layer;
|
|
}
|
|
|
|
int BKE_id_attributes_length(const ID *id, AttrDomainMask domain_mask, eCustomDataMask mask)
|
|
{
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
int length = 0;
|
|
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
const CustomData *customdata = info[domain].customdata;
|
|
if (customdata == nullptr) {
|
|
continue;
|
|
}
|
|
|
|
if ((1 << int(domain)) & domain_mask) {
|
|
length += CustomData_number_of_layers_typemask(customdata, mask);
|
|
}
|
|
}
|
|
|
|
return length;
|
|
}
|
|
|
|
AttrDomain BKE_id_attribute_domain(const ID *id, const CustomDataLayer *layer)
|
|
{
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
const CustomData *customdata = info[domain].customdata;
|
|
if (customdata == nullptr) {
|
|
continue;
|
|
}
|
|
if (ARRAY_HAS_ITEM((CustomDataLayer *)layer, customdata->layers, customdata->totlayer)) {
|
|
return AttrDomain(domain);
|
|
}
|
|
}
|
|
|
|
BLI_assert_msg(0, "Custom data layer not found in geometry");
|
|
return AttrDomain(AttrDomain::Point);
|
|
}
|
|
|
|
int BKE_id_attribute_data_length(ID *id, CustomDataLayer *layer)
|
|
{
|
|
/* When in mesh editmode, attributes point to bmesh customdata layers, the attribute data is
|
|
* empty since custom data is stored per element instead of a single array there (same es UVs
|
|
* etc.), see D11998. */
|
|
switch (GS(id->name)) {
|
|
case ID_ME: {
|
|
Mesh *mesh = (Mesh *)id;
|
|
if (mesh->runtime->edit_mesh != nullptr) {
|
|
return 0;
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
const CustomData *customdata = info[domain].customdata;
|
|
if (customdata == nullptr) {
|
|
continue;
|
|
}
|
|
if (ARRAY_HAS_ITEM((CustomDataLayer *)layer, customdata->layers, customdata->totlayer)) {
|
|
return info[domain].length;
|
|
}
|
|
}
|
|
|
|
BLI_assert_msg(0, "Custom data layer not found in geometry");
|
|
return 0;
|
|
}
|
|
|
|
bool BKE_id_attribute_required(const ID *id, const char *name)
|
|
{
|
|
switch (GS(id->name)) {
|
|
case ID_PT:
|
|
return BKE_pointcloud_attribute_required((const PointCloud *)id, name);
|
|
case ID_CV:
|
|
return BKE_curves_attribute_required((const Curves *)id, name);
|
|
case ID_ME:
|
|
return BKE_mesh_attribute_required(name);
|
|
default:
|
|
return false;
|
|
}
|
|
}
|
|
|
|
CustomDataLayer *BKE_id_attributes_active_get(ID *id)
|
|
{
|
|
int active_index = *BKE_id_attributes_active_index_p(id);
|
|
if (active_index > BKE_id_attributes_length(id, ATTR_DOMAIN_MASK_ALL, CD_MASK_PROP_ALL)) {
|
|
active_index = 0;
|
|
}
|
|
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
int index = 0;
|
|
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
CustomData *customdata = info[domain].customdata;
|
|
if (customdata == nullptr) {
|
|
continue;
|
|
}
|
|
for (int i = 0; i < customdata->totlayer; i++) {
|
|
CustomDataLayer *layer = &customdata->layers[i];
|
|
if (CD_MASK_PROP_ALL & CD_TYPE_AS_MASK(layer->type)) {
|
|
if (index == active_index) {
|
|
if (BKE_attribute_allow_procedural_access(layer->name)) {
|
|
return layer;
|
|
}
|
|
return nullptr;
|
|
}
|
|
index++;
|
|
}
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
void BKE_id_attributes_active_set(ID *id, const char *name)
|
|
{
|
|
const CustomDataLayer *layer = BKE_id_attribute_search(
|
|
id, name, CD_MASK_PROP_ALL, ATTR_DOMAIN_MASK_ALL);
|
|
BLI_assert(layer != nullptr);
|
|
|
|
const int index = BKE_id_attribute_to_index(id, layer, ATTR_DOMAIN_MASK_ALL, CD_MASK_PROP_ALL);
|
|
*BKE_id_attributes_active_index_p(id) = index;
|
|
}
|
|
|
|
int *BKE_id_attributes_active_index_p(ID *id)
|
|
{
|
|
switch (GS(id->name)) {
|
|
case ID_PT: {
|
|
return &((PointCloud *)id)->attributes_active_index;
|
|
}
|
|
case ID_ME: {
|
|
return &((Mesh *)id)->attributes_active_index;
|
|
}
|
|
case ID_CV: {
|
|
return &((Curves *)id)->attributes_active_index;
|
|
}
|
|
case ID_GP: {
|
|
return &((GreasePencil *)id)->attributes_active_index;
|
|
}
|
|
default:
|
|
return nullptr;
|
|
}
|
|
}
|
|
|
|
CustomData *BKE_id_attributes_iterator_next_domain(ID *id, CustomDataLayer *layers)
|
|
{
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
bool use_next = (layers == nullptr);
|
|
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
CustomData *customdata = info[domain].customdata;
|
|
if (customdata == nullptr) {
|
|
continue;
|
|
}
|
|
if (customdata->layers && customdata->totlayer) {
|
|
if (customdata->layers == layers) {
|
|
use_next = true;
|
|
}
|
|
else if (use_next) {
|
|
return customdata;
|
|
}
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
CustomDataLayer *BKE_id_attribute_from_index(ID *id,
|
|
int lookup_index,
|
|
AttrDomainMask domain_mask,
|
|
eCustomDataMask layer_mask)
|
|
{
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
int index = 0;
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
CustomData *customdata = info[domain].customdata;
|
|
|
|
if (!customdata || !((1 << int(domain)) & domain_mask)) {
|
|
continue;
|
|
}
|
|
|
|
for (int i = 0; i < customdata->totlayer; i++) {
|
|
if (!(layer_mask & CD_TYPE_AS_MASK(customdata->layers[i].type)) ||
|
|
(customdata->layers[i].flag & CD_FLAG_TEMPORARY))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (index == lookup_index) {
|
|
return customdata->layers + i;
|
|
}
|
|
|
|
index++;
|
|
}
|
|
}
|
|
|
|
return nullptr;
|
|
}
|
|
|
|
int BKE_id_attribute_to_index(const ID *id,
|
|
const CustomDataLayer *layer,
|
|
AttrDomainMask domain_mask,
|
|
eCustomDataMask layer_mask)
|
|
{
|
|
if (!layer) {
|
|
return -1;
|
|
}
|
|
|
|
DomainInfo info[ATTR_DOMAIN_NUM];
|
|
get_domains(id, info);
|
|
|
|
int index = 0;
|
|
for (const int domain : IndexRange(ATTR_DOMAIN_NUM)) {
|
|
const CustomData *customdata = info[domain].customdata;
|
|
|
|
if (!customdata || !((1 << int(domain)) & domain_mask)) {
|
|
continue;
|
|
}
|
|
|
|
for (int i = 0; i < customdata->totlayer; i++) {
|
|
const CustomDataLayer *layer_iter = customdata->layers + i;
|
|
if (!(layer_mask & CD_TYPE_AS_MASK(layer_iter->type)) ||
|
|
(layer_iter->flag & CD_FLAG_TEMPORARY))
|
|
{
|
|
continue;
|
|
}
|
|
|
|
if (layer == layer_iter) {
|
|
return index;
|
|
}
|
|
|
|
index++;
|
|
}
|
|
}
|
|
|
|
return -1;
|
|
}
|
|
|
|
const char *BKE_id_attributes_active_color_name(const ID *id)
|
|
{
|
|
if (GS(id->name) == ID_ME) {
|
|
return reinterpret_cast<const Mesh *>(id)->active_color_attribute;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
const char *BKE_id_attributes_default_color_name(const ID *id)
|
|
{
|
|
if (GS(id->name) == ID_ME) {
|
|
return reinterpret_cast<const Mesh *>(id)->default_color_attribute;
|
|
}
|
|
return nullptr;
|
|
}
|
|
|
|
void BKE_id_attributes_active_color_set(ID *id, const char *name)
|
|
{
|
|
switch (GS(id->name)) {
|
|
case ID_ME: {
|
|
Mesh *mesh = reinterpret_cast<Mesh *>(id);
|
|
MEM_SAFE_FREE(mesh->active_color_attribute);
|
|
if (name) {
|
|
mesh->active_color_attribute = BLI_strdup(name);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void BKE_id_attributes_default_color_set(ID *id, const char *name)
|
|
{
|
|
switch (GS(id->name)) {
|
|
case ID_ME: {
|
|
Mesh *mesh = reinterpret_cast<Mesh *>(id);
|
|
MEM_SAFE_FREE(mesh->default_color_attribute);
|
|
if (name) {
|
|
mesh->default_color_attribute = BLI_strdup(name);
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
const CustomDataLayer *BKE_id_attributes_color_find(const ID *id, const char *name)
|
|
{
|
|
return BKE_id_attribute_search(
|
|
const_cast<ID *>(id), name, CD_MASK_COLOR_ALL, ATTR_DOMAIN_MASK_COLOR);
|
|
}
|
|
|
|
bool BKE_color_attribute_supported(const Mesh &mesh, const blender::StringRef name)
|
|
{
|
|
std::optional<blender::bke::AttributeMetaData> meta_data = mesh.attributes().lookup_meta_data(
|
|
name);
|
|
|
|
if (!meta_data) {
|
|
return false;
|
|
}
|
|
if (!(ATTR_DOMAIN_AS_MASK(meta_data->domain) & ATTR_DOMAIN_MASK_COLOR) ||
|
|
!(CD_TYPE_AS_MASK(meta_data->data_type) & CD_MASK_COLOR_ALL))
|
|
{
|
|
return false;
|
|
}
|
|
return true;
|
|
}
|
|
|
|
const char *BKE_uv_map_vert_select_name_get(const char *uv_map_name, char *buffer)
|
|
{
|
|
BLI_assert(strlen(UV_VERTSEL_NAME) == 2);
|
|
BLI_assert(strlen(uv_map_name) < MAX_CUSTOMDATA_LAYER_NAME - 4);
|
|
BLI_snprintf(buffer, MAX_CUSTOMDATA_LAYER_NAME, ".%s.%s", UV_VERTSEL_NAME, uv_map_name);
|
|
return buffer;
|
|
}
|
|
|
|
const char *BKE_uv_map_edge_select_name_get(const char *uv_map_name, char *buffer)
|
|
{
|
|
BLI_assert(strlen(UV_EDGESEL_NAME) == 2);
|
|
BLI_assert(strlen(uv_map_name) < MAX_CUSTOMDATA_LAYER_NAME - 4);
|
|
BLI_snprintf(buffer, MAX_CUSTOMDATA_LAYER_NAME, ".%s.%s", UV_EDGESEL_NAME, uv_map_name);
|
|
return buffer;
|
|
}
|
|
|
|
const char *BKE_uv_map_pin_name_get(const char *uv_map_name, char *buffer)
|
|
{
|
|
BLI_assert(strlen(UV_PINNED_NAME) == 2);
|
|
BLI_assert(strlen(uv_map_name) < MAX_CUSTOMDATA_LAYER_NAME - 4);
|
|
BLI_snprintf(buffer, MAX_CUSTOMDATA_LAYER_NAME, ".%s.%s", UV_PINNED_NAME, uv_map_name);
|
|
return buffer;
|
|
}
|