Files
test2/source/blender/makesrna/intern/rna_cachefile.cc

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

405 lines
16 KiB
C++
Raw Normal View History

/* SPDX-FileCopyrightText: 2016 Blender Authors
*
* SPDX-License-Identifier: GPL-2.0-or-later */
/** \file
* \ingroup RNA
2018-08-30 01:56:08 +10:00
*/
#include "DNA_cachefile_types.h"
#include "DNA_scene_types.h"
#include "BLT_translation.hh"
#include "RNA_define.hh"
#include "RNA_enum_types.hh"
#include "rna_internal.hh"
const EnumPropertyItem rna_enum_velocity_unit_items[] = {
{CACHEFILE_VELOCITY_UNIT_SECOND, "SECOND", 0, "Second", ""},
{CACHEFILE_VELOCITY_UNIT_FRAME, "FRAME", 0, "Frame", ""},
{0, nullptr, 0, nullptr, nullptr},
};
#ifdef RNA_RUNTIME
# include "BLI_math_base.h"
# include "BKE_cachefile.hh"
# include "BKE_context.hh"
# include "BKE_report.hh"
# include "DEG_depsgraph.hh"
# include "DEG_depsgraph_build.hh"
# include "WM_api.hh"
# include "WM_types.hh"
# ifdef WITH_ALEMBIC
# include "ABC_alembic.h"
# endif
static void rna_CacheFile_update(Main * /*bmain*/, Scene * /*scene*/, PointerRNA *ptr)
{
CacheFile *cache_file = (CacheFile *)ptr->data;
DEG_id_tag_update(&cache_file->id, ID_RECALC_SYNC_TO_EVAL);
WM_main_add_notifier(NC_OBJECT | ND_DRAW, nullptr);
}
static void rna_CacheFileLayer_update(Main * /*bmain*/, Scene * /*scene*/, PointerRNA *ptr)
{
CacheFile *cache_file = (CacheFile *)ptr->owner_id;
DEG_id_tag_update(&cache_file->id, ID_RECALC_SYNC_TO_EVAL);
WM_main_add_notifier(NC_OBJECT | ND_DRAW, nullptr);
}
Cycles: experimental integration of Alembic procedural in viewport rendering This patch exposes the Cycles Alembic Procedural through the MeshSequenceCache modifier in order to use and test it from Blender. To enable it, one has to switch the render feature set to experimental and activate the Procedural in the modifier. An Alembic Procedural is then created for each CacheFile from Blender set to use the Procedural, and each Blender object having a MeshSequenceCache modifier is added to list of objects of the right procedural. The procedural's parameters derive from the CacheFile's properties which are already exposed in the UI through the modifier, although more Cycles specific options might be added in the future. As there is currently no cache controls and since we load all the data at the beginning of the render session, the procedural is only available during viewport renders at the moment. When an Alembic procedural is rendered, data from the archive are not read on the Blender side. If a Cycles render is not active and the CacheFile is set to use the Cycles Procedural, bounding boxes are used to display the objects in the scene as a signal that the objects are not processed by Blender anymore. This is standard in other DCCs. However this does not reduce the memory usage from Blender as the Alembic data was already loaded either during an import or during a .blend file read. This is mostly a hack to test the Cycles Alembic procedural until we have a better Blender side mechanism for letting renderers load their own geometry, which will be based on import and export settings on Collections (T68933). Ref T79174, D3089 Reviewed By: brecht, sybren Maniphest Tasks: T79174 Differential Revision: https://developer.blender.org/D10197
2021-08-19 14:34:01 +02:00
static void rna_CacheFile_dependency_update(Main *bmain, Scene *scene, PointerRNA *ptr)
{
rna_CacheFile_update(bmain, scene, ptr);
DEG_relations_tag_update(bmain);
}
static void rna_CacheFile_object_paths_begin(CollectionPropertyIterator *iter, PointerRNA *ptr)
{
CacheFile *cache_file = (CacheFile *)ptr->data;
Core: Add info about chain of ancestors (owner data) of a PointerRNA. The general idea is to store an array of (type, data) pointers of all PointerRNA ancestors of the current one. This will help solving cases in our code where the owner (or sometimes even the owner of the owner) of a random PointerRNA needs to be accessed. Current solution mainly relies on linear search from the owner ID, which is sub-optimal at best, and may not even be possible in case a same data is shared between different owners. This lead to refactoring quite a bit of existing PointerRNA creation code. At a high level (i.e. expected usages outside of RNA internals): * Add `RNA_pointer_create_with_parent` and `RNA_pointer_create_id_subdata` to create RNA pointers with ancestors info. * `RNA_id_pointer_create` and `RNA_main_pointer_create` remain unchanged, as they should never have ancestors currently. * Add `RNA_pointer_create_from_ancestor` to re-create a RNA pointer from the nth ancestor of another PointerRNA. * Add basic python API to access this new ancestors data. * Update internal RNA/bpy code to handle ancestors generation in most common generic cases. - The most verbose change here is for collection code, as the owner of the collection property is now passed around, to allow collection items to get a valid ancestors chain. Internally: * `PointerRNA` now has an array of `AncestorPointerRNA` data to store the ancestors. * `PointerRNA` now has constructors that take care of setting its data for most usual cases, including handling of the ancestor array data. * Pointer type refining has been fully factorized into a small utils, `rna_pointer_refine`, that is now used from all code doing that operation. * `rna_pointer_inherit_refine` has been replaced by `rna_pointer_create_with_ancestors` as the core function taking care of creating pointers with valid ancestors info. - Its usage outside of `rna_access` has been essentially reduced to custom collection lookup callbacks. Implements #122431. -------------- Some notes: * The goal of this commit is _not_ to fully cover all cases creating PointerRNA that should also store the ancestors' chain info. It only tackles the most generic code paths (in bpyrna and RNA itself mainly). The remaining 'missing cases' can be tackle later, as needs be. * Performances seem to be only marginally affected currently. * Currently `AncestorPointerRNA` only stores PointerRNA-like data. This will help `StructPathFunc` callbacks to more efficiently generate an RNA paths when calling e.g. `RNA_path_from_ID_to_property`, but will not be enough info to build these paths without these callbacks. And some cases may still remain fuzzy. We'd have to add thinks like a `PropertyRNA` pointer, and for RNA collection ones, an index and string identifier, to store a complete unambiguous 'RNA path' info. This is probably not needed, nor worth the extra processing and memory footprint, for now. Pull Request: https://projects.blender.org/blender/blender/pulls/122427
2025-02-05 15:45:04 +01:00
rna_iterator_listbase_begin(iter, ptr, &cache_file->object_paths, nullptr);
}
static PointerRNA rna_CacheFile_active_layer_get(PointerRNA *ptr)
{
CacheFile *cache_file = (CacheFile *)ptr->owner_id;
Core: Add info about chain of ancestors (owner data) of a PointerRNA. The general idea is to store an array of (type, data) pointers of all PointerRNA ancestors of the current one. This will help solving cases in our code where the owner (or sometimes even the owner of the owner) of a random PointerRNA needs to be accessed. Current solution mainly relies on linear search from the owner ID, which is sub-optimal at best, and may not even be possible in case a same data is shared between different owners. This lead to refactoring quite a bit of existing PointerRNA creation code. At a high level (i.e. expected usages outside of RNA internals): * Add `RNA_pointer_create_with_parent` and `RNA_pointer_create_id_subdata` to create RNA pointers with ancestors info. * `RNA_id_pointer_create` and `RNA_main_pointer_create` remain unchanged, as they should never have ancestors currently. * Add `RNA_pointer_create_from_ancestor` to re-create a RNA pointer from the nth ancestor of another PointerRNA. * Add basic python API to access this new ancestors data. * Update internal RNA/bpy code to handle ancestors generation in most common generic cases. - The most verbose change here is for collection code, as the owner of the collection property is now passed around, to allow collection items to get a valid ancestors chain. Internally: * `PointerRNA` now has an array of `AncestorPointerRNA` data to store the ancestors. * `PointerRNA` now has constructors that take care of setting its data for most usual cases, including handling of the ancestor array data. * Pointer type refining has been fully factorized into a small utils, `rna_pointer_refine`, that is now used from all code doing that operation. * `rna_pointer_inherit_refine` has been replaced by `rna_pointer_create_with_ancestors` as the core function taking care of creating pointers with valid ancestors info. - Its usage outside of `rna_access` has been essentially reduced to custom collection lookup callbacks. Implements #122431. -------------- Some notes: * The goal of this commit is _not_ to fully cover all cases creating PointerRNA that should also store the ancestors' chain info. It only tackles the most generic code paths (in bpyrna and RNA itself mainly). The remaining 'missing cases' can be tackle later, as needs be. * Performances seem to be only marginally affected currently. * Currently `AncestorPointerRNA` only stores PointerRNA-like data. This will help `StructPathFunc` callbacks to more efficiently generate an RNA paths when calling e.g. `RNA_path_from_ID_to_property`, but will not be enough info to build these paths without these callbacks. And some cases may still remain fuzzy. We'd have to add thinks like a `PropertyRNA` pointer, and for RNA collection ones, an index and string identifier, to store a complete unambiguous 'RNA path' info. This is probably not needed, nor worth the extra processing and memory footprint, for now. Pull Request: https://projects.blender.org/blender/blender/pulls/122427
2025-02-05 15:45:04 +01:00
return RNA_pointer_create_with_parent(
*ptr, &RNA_CacheFileLayer, BKE_cachefile_get_active_layer(cache_file));
}
static void rna_CacheFile_active_layer_set(PointerRNA *ptr, PointerRNA value, ReportList *reports)
{
CacheFile *cache_file = (CacheFile *)ptr->owner_id;
int index = BLI_findindex(&cache_file->layers, value.data);
if (index == -1) {
BKE_reportf(reports,
RPT_ERROR,
"Layer '%s' not found in object '%s'",
((CacheFileLayer *)value.data)->filepath,
cache_file->id.name + 2);
return;
}
cache_file->active_layer = index + 1;
}
static int rna_CacheFile_active_layer_index_get(PointerRNA *ptr)
{
CacheFile *cache_file = (CacheFile *)ptr->owner_id;
return cache_file->active_layer - 1;
}
static void rna_CacheFile_active_layer_index_set(PointerRNA *ptr, int value)
{
CacheFile *cache_file = (CacheFile *)ptr->owner_id;
cache_file->active_layer = value + 1;
}
static void rna_CacheFile_active_layer_index_range(
PointerRNA *ptr, int *min, int *max, int * /*softmin*/, int * /*softmax*/)
{
CacheFile *cache_file = (CacheFile *)ptr->owner_id;
*min = 0;
*max = max_ii(0, BLI_listbase_count(&cache_file->layers) - 1);
}
static void rna_CacheFileLayer_hidden_flag_set(PointerRNA *ptr, const bool value)
{
CacheFileLayer *layer = (CacheFileLayer *)ptr->data;
if (value) {
layer->flag |= CACHEFILE_LAYER_HIDDEN;
}
else {
layer->flag &= ~CACHEFILE_LAYER_HIDDEN;
}
}
static CacheFileLayer *rna_CacheFile_layer_new(CacheFile *cache_file,
bContext *C,
ReportList *reports,
const char *filepath)
{
CacheFileLayer *layer = BKE_cachefile_add_layer(cache_file, filepath);
if (layer == nullptr) {
BKE_reportf(
reports, RPT_ERROR, "Cannot add a layer to CacheFile '%s'", cache_file->id.name + 2);
return nullptr;
}
Depsgraph *depsgraph = CTX_data_ensure_evaluated_depsgraph(C);
BKE_cachefile_reload(depsgraph, cache_file);
WM_main_add_notifier(NC_OBJECT | ND_DRAW, nullptr);
return layer;
}
static void rna_CacheFile_layer_remove(CacheFile *cache_file, bContext *C, PointerRNA *layer_ptr)
{
CacheFileLayer *layer = static_cast<CacheFileLayer *>(layer_ptr->data);
BKE_cachefile_remove_layer(cache_file, layer);
Depsgraph *depsgraph = CTX_data_ensure_evaluated_depsgraph(C);
BKE_cachefile_reload(depsgraph, cache_file);
WM_main_add_notifier(NC_OBJECT | ND_DRAW, nullptr);
}
#else
/* cachefile.object_paths */
static void rna_def_alembic_object_path(BlenderRNA *brna)
{
StructRNA *srna = RNA_def_struct(brna, "CacheObjectPath", nullptr);
RNA_def_struct_sdna(srna, "CacheObjectPath");
RNA_def_struct_ui_text(srna, "Object Path", "Path of an object inside of an Alembic archive");
RNA_def_struct_ui_icon(srna, ICON_NONE);
RNA_define_lib_overridable(true);
PropertyRNA *prop = RNA_def_property(srna, "path", PROP_STRING, PROP_NONE);
RNA_def_property_ui_text(prop, "Path", "Object path");
RNA_def_property_translation_context(prop, BLT_I18NCONTEXT_EDITOR_FILEBROWSER);
RNA_def_struct_name_property(srna, prop);
RNA_define_lib_overridable(false);
}
/* cachefile.object_paths */
static void rna_def_cachefile_object_paths(BlenderRNA *brna, PropertyRNA *cprop)
{
RNA_def_property_srna(cprop, "CacheObjectPaths");
StructRNA *srna = RNA_def_struct(brna, "CacheObjectPaths", nullptr);
RNA_def_struct_sdna(srna, "CacheFile");
RNA_def_struct_ui_text(srna, "Object Paths", "Collection of object paths");
}
static void rna_def_cachefile_layer(BlenderRNA *brna)
{
StructRNA *srna = RNA_def_struct(brna, "CacheFileLayer", nullptr);
RNA_def_struct_sdna(srna, "CacheFileLayer");
RNA_def_struct_ui_text(
srna,
"Cache Layer",
"Layer of the cache, used to load or override data from the first the first layer");
PropertyRNA *prop = RNA_def_property(srna, "filepath", PROP_STRING, PROP_FILEPATH);
RNA_def_property_ui_text(prop, "File Path", "Path to the archive");
RNA_def_property_update(prop, 0, "rna_CacheFileLayer_update");
prop = RNA_def_property(srna, "hide_layer", PROP_BOOLEAN, PROP_NONE);
RNA_def_property_boolean_sdna(prop, nullptr, "flag", CACHEFILE_LAYER_HIDDEN);
RNA_def_property_boolean_funcs(prop, nullptr, "rna_CacheFileLayer_hidden_flag_set");
RNA_def_property_ui_icon(prop, ICON_HIDE_OFF, -1);
RNA_def_property_ui_text(prop, "Hide Layer", "Do not load data from this layer");
RNA_def_property_update(prop, 0, "rna_CacheFileLayer_update");
}
static void rna_def_cachefile_layers(BlenderRNA *brna, PropertyRNA *cprop)
{
RNA_def_property_srna(cprop, "CacheFileLayers");
StructRNA *srna = RNA_def_struct(brna, "CacheFileLayers", nullptr);
RNA_def_struct_sdna(srna, "CacheFile");
RNA_def_struct_ui_text(srna, "Cache Layers", "Collection of cache layers");
PropertyRNA *prop = RNA_def_property(srna, "active", PROP_POINTER, PROP_NONE);
RNA_def_property_struct_type(prop, "CacheFileLayer");
RNA_def_property_pointer_funcs(
prop, "rna_CacheFile_active_layer_get", "rna_CacheFile_active_layer_set", nullptr, nullptr);
RNA_def_property_flag(prop, PROP_EDITABLE);
RNA_def_property_ui_text(prop, "Active Layer", "Active layer of the CacheFile");
/* Add a layer. */
FunctionRNA *func = RNA_def_function(srna, "new", "rna_CacheFile_layer_new");
RNA_def_function_flag(func, FUNC_USE_REPORTS | FUNC_USE_CONTEXT);
RNA_def_function_ui_description(func, "Add a new layer");
PropertyRNA *parm = RNA_def_string(
func, "filepath", "File Path", 0, "", "File path to the archive used as a layer");
RNA_def_parameter_flags(parm, PropertyFlag(0), PARM_REQUIRED);
/* Return type. */
parm = RNA_def_pointer(func, "layer", "CacheFileLayer", "", "Newly created layer");
RNA_def_function_return(func, parm);
/* Remove a layer. */
func = RNA_def_function(srna, "remove", "rna_CacheFile_layer_remove");
RNA_def_function_flag(func, FUNC_USE_CONTEXT);
RNA_def_function_ui_description(func, "Remove an existing layer from the cache file");
parm = RNA_def_pointer(func, "layer", "CacheFileLayer", "", "Layer to remove");
RNA_def_parameter_flags(parm, PROP_NEVER_NULL, PARM_REQUIRED | PARM_RNAPTR);
RNA_def_parameter_clear_flags(parm, PROP_THICK_WRAP, ParameterFlag(0));
}
static void rna_def_cachefile(BlenderRNA *brna)
{
StructRNA *srna = RNA_def_struct(brna, "CacheFile", "ID");
RNA_def_struct_sdna(srna, "CacheFile");
RNA_def_struct_ui_text(srna, "CacheFile", "");
RNA_def_struct_ui_icon(srna, ICON_FILE);
RNA_define_lib_overridable(true);
PropertyRNA *prop = RNA_def_property(srna, "filepath", PROP_STRING, PROP_FILEPATH);
RNA_def_property_ui_text(prop, "File Path", "Path to external displacements file");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
prop = RNA_def_property(srna, "is_sequence", PROP_BOOLEAN, PROP_NONE);
RNA_def_property_ui_text(
prop, "Sequence", "Whether the cache is separated in a series of files");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
Cycles: experimental integration of Alembic procedural in viewport rendering This patch exposes the Cycles Alembic Procedural through the MeshSequenceCache modifier in order to use and test it from Blender. To enable it, one has to switch the render feature set to experimental and activate the Procedural in the modifier. An Alembic Procedural is then created for each CacheFile from Blender set to use the Procedural, and each Blender object having a MeshSequenceCache modifier is added to list of objects of the right procedural. The procedural's parameters derive from the CacheFile's properties which are already exposed in the UI through the modifier, although more Cycles specific options might be added in the future. As there is currently no cache controls and since we load all the data at the beginning of the render session, the procedural is only available during viewport renders at the moment. When an Alembic procedural is rendered, data from the archive are not read on the Blender side. If a Cycles render is not active and the CacheFile is set to use the Cycles Procedural, bounding boxes are used to display the objects in the scene as a signal that the objects are not processed by Blender anymore. This is standard in other DCCs. However this does not reduce the memory usage from Blender as the Alembic data was already loaded either during an import or during a .blend file read. This is mostly a hack to test the Cycles Alembic procedural until we have a better Blender side mechanism for letting renderers load their own geometry, which will be based on import and export settings on Collections (T68933). Ref T79174, D3089 Reviewed By: brecht, sybren Maniphest Tasks: T79174 Differential Revision: https://developer.blender.org/D10197
2021-08-19 14:34:01 +02:00
prop = RNA_def_property(srna, "use_render_procedural", PROP_BOOLEAN, PROP_NONE);
RNA_def_property_ui_text(
prop,
"Use Render Engine Procedural",
"Display boxes in the viewport as placeholders for the objects, Cycles will use a "
"procedural to load the objects during viewport rendering in experimental mode, "
"other render engines will also receive a placeholder and should take care of loading the "
"Alembic data themselves if possible");
RNA_def_property_update(prop, 0, "rna_CacheFile_dependency_update");
/* ----------------- For Scene time ------------------- */
prop = RNA_def_property(srna, "override_frame", PROP_BOOLEAN, PROP_NONE);
RNA_def_property_ui_text(prop,
"Override Frame",
"Whether to use a custom frame for looking up data in the cache file,"
" instead of using the current scene frame");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
prop = RNA_def_property(srna, "frame", PROP_FLOAT, PROP_NONE);
RNA_def_property_float_sdna(prop, nullptr, "frame");
RNA_def_property_range(prop, -MAXFRAME, MAXFRAME);
RNA_def_property_ui_text(prop,
"Frame",
"The time to use for looking up the data in the cache file,"
" or to determine which file to use in a file sequence");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
prop = RNA_def_property(srna, "frame_offset", PROP_FLOAT, PROP_NONE);
RNA_def_property_float_sdna(prop, nullptr, "frame_offset");
RNA_def_property_range(prop, -MAXFRAME, MAXFRAME);
RNA_def_property_ui_text(prop,
"Frame Offset",
"Subtracted from the current frame to use for "
"looking up the data in the cache file, or to "
"determine which file to use in a file sequence");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
/* ----------------- Cache controls ----------------- */
prop = RNA_def_property(srna, "use_prefetch", PROP_BOOLEAN, PROP_NONE);
RNA_def_property_ui_text(
prop,
"Use Prefetch",
"When enabled, the Cycles Procedural will preload animation data for faster updates");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
prop = RNA_def_property(srna, "prefetch_cache_size", PROP_INT, PROP_UNSIGNED);
RNA_def_property_ui_text(
prop,
"Prefetch Cache Size",
"Memory usage limit in megabytes for the Cycles Procedural cache, if the data does not "
"fit within the limit, rendering is aborted");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
/* ----------------- Axis Conversion ----------------- */
prop = RNA_def_property(srna, "forward_axis", PROP_ENUM, PROP_NONE);
RNA_def_property_enum_sdna(prop, nullptr, "forward_axis");
RNA_def_property_enum_items(prop, rna_enum_object_axis_items);
RNA_def_property_ui_text(prop, "Forward", "");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
prop = RNA_def_property(srna, "up_axis", PROP_ENUM, PROP_NONE);
RNA_def_property_enum_sdna(prop, nullptr, "up_axis");
RNA_def_property_enum_items(prop, rna_enum_object_axis_items);
RNA_def_property_ui_text(prop, "Up", "");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
prop = RNA_def_property(srna, "scale", PROP_FLOAT, PROP_NONE);
RNA_def_property_float_sdna(prop, nullptr, "scale");
RNA_def_property_range(prop, 0.0001f, 1000.0f);
RNA_def_property_ui_text(
prop,
"Scale",
"Value by which to enlarge or shrink the object with respect to the world's origin"
" (only applicable through a Transform Cache constraint)");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
/* object paths */
prop = RNA_def_property(srna, "object_paths", PROP_COLLECTION, PROP_NONE);
RNA_def_property_collection_sdna(prop, nullptr, "object_paths", nullptr);
RNA_def_property_collection_funcs(prop,
"rna_CacheFile_object_paths_begin",
"rna_iterator_listbase_next",
"rna_iterator_listbase_end",
"rna_iterator_listbase_get",
nullptr,
nullptr,
nullptr,
nullptr);
RNA_def_property_struct_type(prop, "CacheObjectPath");
RNA_def_property_srna(prop, "CacheObjectPaths");
RNA_def_property_ui_text(
prop, "Object Paths", "Paths of the objects inside the Alembic archive");
/* ----------------- Alembic Velocity Attribute ----------------- */
prop = RNA_def_property(srna, "velocity_name", PROP_STRING, PROP_NONE);
RNA_def_property_ui_text(prop,
"Velocity Attribute",
"Name of the Alembic attribute used for generating motion blur data");
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
RNA_def_property_clear_flag(prop, PROP_ANIMATABLE);
prop = RNA_def_property(srna, "velocity_unit", PROP_ENUM, PROP_NONE);
RNA_def_property_enum_sdna(prop, nullptr, "velocity_unit");
RNA_def_property_enum_items(prop, rna_enum_velocity_unit_items);
RNA_def_property_ui_text(
prop,
"Velocity Unit",
"Define how the velocity vectors are interpreted with regard to time, 'frame' means "
"the delta time is 1 frame, 'second' means the delta time is 1 / FPS");
RNA_def_property_translation_context(prop, BLT_I18NCONTEXT_UNIT);
RNA_def_property_update(prop, 0, "rna_CacheFile_update");
RNA_def_property_clear_flag(prop, PROP_ANIMATABLE);
/* ----------------- Alembic Layers ----------------- */
prop = RNA_def_property(srna, "layers", PROP_COLLECTION, PROP_NONE);
RNA_def_property_collection_sdna(prop, nullptr, "layers", nullptr);
RNA_def_property_struct_type(prop, "CacheFileLayer");
RNA_def_property_override_clear_flag(prop, PROPOVERRIDE_OVERRIDABLE_LIBRARY);
RNA_def_property_ui_text(prop, "Cache Layers", "Layers of the cache");
rna_def_cachefile_layers(brna, prop);
prop = RNA_def_property(srna, "active_index", PROP_INT, PROP_UNSIGNED);
RNA_def_property_clear_flag(prop, PROP_ANIMATABLE);
RNA_def_property_int_sdna(prop, nullptr, "active_layer");
RNA_def_property_int_funcs(prop,
"rna_CacheFile_active_layer_index_get",
"rna_CacheFile_active_layer_index_set",
"rna_CacheFile_active_layer_index_range");
RNA_define_lib_overridable(false);
rna_def_cachefile_object_paths(brna, prop);
rna_def_animdata_common(srna);
}
void RNA_def_cachefile(BlenderRNA *brna)
{
rna_def_cachefile(brna);
rna_def_alembic_object_path(brna);
rna_def_cachefile_layer(brna);
}
#endif