Files
test/source/blender/editors/object/object_collection.cc
Damien Picard 81d9e94218 UI: Fix and improve a few messages
- "Parameters for custom (OSL-based) Cameras" -> "cameras": lower case
  in tooltips.
- "Connect two nodes ... (automatically determined": missing
  parenthesis.
- "Join curve... control points are detected(if disabled...": add
  missing space.
- "Add Selected to Active Objects Collection" -> "Active Object's":
  typo.
- "Duplicate the acive shape key" -> "active": typo.
- "Copy selected points ": remove trailing space.
- "Move cursor" -> "Cursor": title case for operator.
- "Paste text to clipboard" -> "from clipboard": typo.
- "An empty Action considered as both a 'layered' and a 'layered'
  Action." -> "is considered as both a 'legacy' and a 'layered'
  Action": likely copy-paste error.
- "Target's Z axis will constraint..." -> "will constrain": typo.
- "The layer groups is expanded in the UI" -> "layer group": typo.
- Deprecation warnings: add missing parentheses.
- "... on low poly geometry.Offset rays...": add missing space after
  period.
- "... relative to the files directory" -> "... to the file's
  directory": typo.
- "The unit multiplier for pixels per meter" -> "The base unit": this
  property description was copy and pasted.
- "... beyond the faces UVs..." -> "the faces' UVs: typo.
- "Is tracking data contains ..." -> "Whether the tracking data
  contains": grammar.
- "Selected text" -> "Text": title case for prop.
- "The user has been shown the "Online Access" prompt and make a
  choice" -> "made a choice": grammar.
- "Glare ": remove trailing space.
- "Don't collapse a curves" -> "Do not collapse curves": grammar.

Some issues reported by Tamar Mebonia.

Pull Request: https://projects.blender.org/blender/blender/pulls/139118
2025-05-19 22:12:17 +02:00

1124 lines
33 KiB
C++

/* SPDX-FileCopyrightText: Blender Authors
*
* SPDX-License-Identifier: GPL-2.0-or-later */
/** \file
* \ingroup edobj
*/
#include <cstring>
#include "BLI_listbase.h"
#include "BLI_path_utils.hh"
#include "BLI_string.h"
#include "BLI_utildefines.h"
#include "DNA_collection_types.h"
#include "DNA_object_types.h"
#include "DNA_scene_types.h"
#include "BKE_collection.hh"
#include "BKE_context.hh"
#include "BKE_file_handler.hh"
#include "BKE_idprop.hh"
#include "BKE_layer.hh"
#include "BKE_lib_id.hh"
#include "BKE_library.hh"
#include "BKE_main.hh"
#include "BKE_object.hh"
#include "BKE_report.hh"
#include "BKE_screen.hh"
#include "BLT_translation.hh"
#include "DEG_depsgraph.hh"
#include "DEG_depsgraph_build.hh"
#include "ED_object.hh"
#include "ED_screen.hh"
#include "WM_api.hh"
#include "WM_types.hh"
#include "RNA_access.hh"
#include "RNA_define.hh"
#include "RNA_enum_types.hh"
#include "RNA_prototypes.hh"
#include "UI_interface.hh"
#include "UI_interface_icons.hh"
#include "object_intern.hh"
namespace blender::ed::object {
/********************* 3d view operators ***********************/
/* can be called with C == nullptr */
static const EnumPropertyItem *collection_object_active_itemf(bContext *C,
PointerRNA * /*ptr*/,
PropertyRNA * /*prop*/,
bool *r_free)
{
Main *bmain = CTX_data_main(C);
Scene *scene = CTX_data_scene(C);
Object *ob;
EnumPropertyItem *item = nullptr, item_tmp = {0};
int totitem = 0;
if (C == nullptr) {
return rna_enum_dummy_NULL_items;
}
ob = context_object(C);
/* check that the object exists */
if (ob) {
Collection *collection;
int i = 0, count = 0;
/* if 2 or more collections, add option to add to all collections */
collection = nullptr;
while ((collection = BKE_collection_object_find(bmain, scene, collection, ob))) {
count++;
}
if (count >= 2) {
item_tmp.identifier = item_tmp.name = "All Collections";
item_tmp.value = INT_MAX; /* this will give nullptr on lookup */
RNA_enum_item_add(&item, &totitem, &item_tmp);
RNA_enum_item_add_separator(&item, &totitem);
}
/* add collections */
collection = nullptr;
while ((collection = BKE_collection_object_find(bmain, scene, collection, ob))) {
item_tmp.identifier = item_tmp.name = collection->id.name + 2;
item_tmp.icon = UI_icon_color_from_collection(collection);
item_tmp.value = i;
RNA_enum_item_add(&item, &totitem, &item_tmp);
i++;
}
}
RNA_enum_item_end(&item, &totitem);
*r_free = true;
return item;
}
/* get the collection back from the enum index, quite awkward and UI specific */
static Collection *collection_object_active_find_index(Main *bmain,
Scene *scene,
Object *ob,
const int collection_object_index)
{
Collection *collection = nullptr;
int i = 0;
while ((collection = BKE_collection_object_find(bmain, scene, collection, ob))) {
if (i == collection_object_index) {
break;
}
i++;
}
return collection;
}
static wmOperatorStatus objects_add_active_exec(bContext *C, wmOperator *op)
{
Object *ob = context_object(C);
Main *bmain = CTX_data_main(C);
Scene *scene = CTX_data_scene(C);
int single_collection_index = RNA_enum_get(op->ptr, "collection");
Collection *single_collection = collection_object_active_find_index(
bmain, scene, ob, single_collection_index);
bool is_cycle = false;
bool changed_multi = false;
if (ob == nullptr) {
return OPERATOR_CANCELLED;
}
/* now add all selected objects to the collection(s) */
FOREACH_COLLECTION_BEGIN (bmain, scene, Collection *, collection) {
if (single_collection && collection != single_collection) {
continue;
}
if (!BKE_collection_has_object(collection, ob)) {
continue;
}
bool changed = false;
CTX_DATA_BEGIN (C, Base *, base, selected_editable_bases) {
if (BKE_collection_has_object(collection, base->object)) {
continue;
}
if (!BKE_collection_object_cyclic_check(bmain, base->object, collection)) {
BKE_collection_object_add(bmain, collection, base->object);
changed = true;
}
else {
is_cycle = true;
}
}
CTX_DATA_END;
if (changed) {
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
changed_multi = true;
}
}
FOREACH_COLLECTION_END;
if (is_cycle) {
BKE_report(op->reports, RPT_WARNING, "Skipped some collections because of cycle detected");
}
if (!changed_multi) {
return OPERATOR_CANCELLED;
}
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_GROUP | NA_EDITED, nullptr);
return OPERATOR_FINISHED;
}
void COLLECTION_OT_objects_add_active(wmOperatorType *ot)
{
PropertyRNA *prop;
/* identifiers */
ot->name = "Add Selected to Active Object's Collection";
ot->description =
"Add selected objects to one of the collections the active-object is part of. "
"Optionally add to \"All Collections\" to ensure selected objects are included in "
"the same collections as the active object";
ot->idname = "COLLECTION_OT_objects_add_active";
/* API callbacks. */
ot->exec = objects_add_active_exec;
ot->invoke = WM_menu_invoke;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
/* properties */
prop = RNA_def_enum(ot->srna,
"collection",
rna_enum_dummy_NULL_items,
0,
"Collection",
"The collection to add other selected objects to");
RNA_def_enum_funcs(prop, collection_object_active_itemf);
RNA_def_property_flag(prop, PROP_ENUM_NO_TRANSLATE);
ot->prop = prop;
}
static wmOperatorStatus objects_remove_active_exec(bContext *C, wmOperator *op)
{
Main *bmain = CTX_data_main(C);
Scene *scene = CTX_data_scene(C);
ViewLayer *view_layer = CTX_data_view_layer(C);
BKE_view_layer_synced_ensure(scene, view_layer);
Object *ob = BKE_view_layer_active_object_get(view_layer);
int single_collection_index = RNA_enum_get(op->ptr, "collection");
Collection *single_collection = collection_object_active_find_index(
bmain, scene, ob, single_collection_index);
bool changed_multi = false;
if (ob == nullptr) {
return OPERATOR_CANCELLED;
}
/* Linking to same collection requires its own loop so we can avoid
* looking up the active objects collections each time. */
FOREACH_COLLECTION_BEGIN (bmain, scene, Collection *, collection) {
if (single_collection && collection != single_collection) {
continue;
}
if (BKE_collection_has_object(collection, ob)) {
/* Remove collections from selected objects */
bool changed = false;
CTX_DATA_BEGIN (C, Base *, base, selected_editable_bases) {
BKE_collection_object_remove(bmain, collection, base->object, false);
changed = true;
}
CTX_DATA_END;
if (changed) {
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
changed_multi = true;
}
}
}
FOREACH_COLLECTION_END;
if (!changed_multi) {
BKE_report(op->reports, RPT_ERROR, "Active object contains no collections");
}
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_GROUP | NA_EDITED, nullptr);
return OPERATOR_FINISHED;
}
void COLLECTION_OT_objects_remove_active(wmOperatorType *ot)
{
PropertyRNA *prop;
/* identifiers */
ot->name = "Remove Selected from Active Collection";
ot->description = "Remove the object from an object collection that contains the active object";
ot->idname = "COLLECTION_OT_objects_remove_active";
/* API callbacks. */
ot->exec = objects_remove_active_exec;
ot->invoke = WM_menu_invoke;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
/* properties */
prop = RNA_def_enum(ot->srna,
"collection",
rna_enum_dummy_NULL_items,
0,
"Collection",
"The collection to remove other selected objects from");
RNA_def_enum_funcs(prop, collection_object_active_itemf);
RNA_def_property_flag(prop, PROP_ENUM_NO_TRANSLATE);
ot->prop = prop;
}
static wmOperatorStatus collection_objects_remove_all_exec(bContext *C, wmOperator * /*op*/)
{
Main *bmain = CTX_data_main(C);
Scene *scene = CTX_data_scene(C);
CTX_DATA_BEGIN (C, Base *, base, selected_editable_bases) {
BKE_object_groups_clear(bmain, scene, base->object);
}
CTX_DATA_END;
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_GROUP | NA_EDITED, nullptr);
return OPERATOR_FINISHED;
}
void COLLECTION_OT_objects_remove_all(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Remove from All Collections";
ot->description = "Remove selected objects from all collections";
ot->idname = "COLLECTION_OT_objects_remove_all";
/* API callbacks. */
ot->exec = collection_objects_remove_all_exec;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
}
static wmOperatorStatus collection_objects_remove_exec(bContext *C, wmOperator *op)
{
Object *ob = context_object(C);
Main *bmain = CTX_data_main(C);
Scene *scene = CTX_data_scene(C);
int single_collection_index = RNA_enum_get(op->ptr, "collection");
Collection *single_collection = collection_object_active_find_index(
bmain, scene, ob, single_collection_index);
bool changed_multi = false;
if (ob == nullptr) {
return OPERATOR_CANCELLED;
}
FOREACH_COLLECTION_BEGIN (bmain, scene, Collection *, collection) {
if (single_collection && collection != single_collection) {
continue;
}
if (!BKE_collection_has_object(collection, ob)) {
continue;
}
/* now remove all selected objects from the collection */
bool changed = false;
CTX_DATA_BEGIN (C, Base *, base, selected_editable_bases) {
BKE_collection_object_remove(bmain, collection, base->object, false);
changed = true;
}
CTX_DATA_END;
if (changed) {
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
changed_multi = true;
}
}
FOREACH_COLLECTION_END;
if (!changed_multi) {
return OPERATOR_CANCELLED;
}
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_GROUP | NA_EDITED, nullptr);
return OPERATOR_FINISHED;
}
void COLLECTION_OT_objects_remove(wmOperatorType *ot)
{
PropertyRNA *prop;
/* identifiers */
ot->name = "Remove from Collection";
ot->description = "Remove selected objects from a collection";
ot->idname = "COLLECTION_OT_objects_remove";
/* API callbacks. */
ot->exec = collection_objects_remove_exec;
ot->invoke = WM_menu_invoke;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
/* properties */
prop = RNA_def_enum(ot->srna,
"collection",
rna_enum_dummy_NULL_items,
0,
"Collection",
"The collection to remove this object from");
RNA_def_enum_funcs(prop, collection_object_active_itemf);
RNA_def_property_flag(prop, PROP_ENUM_NO_TRANSLATE);
ot->prop = prop;
}
static wmOperatorStatus collection_create_exec(bContext *C, wmOperator *op)
{
Main *bmain = CTX_data_main(C);
char name[MAX_ID_NAME - 2]; /* id name */
bool changed = false;
RNA_string_get(op->ptr, "name", name);
Collection *collection = BKE_collection_add(bmain, nullptr, name);
id_fake_user_set(&collection->id);
CTX_DATA_BEGIN (C, Base *, base, selected_bases) {
BKE_collection_object_add(bmain, collection, base->object);
changed = true;
}
CTX_DATA_END;
if (changed) {
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
}
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_GROUP | NA_EDITED, nullptr);
return OPERATOR_FINISHED;
}
void COLLECTION_OT_create(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Create New Collection";
ot->description = "Create an object collection from selected objects";
ot->idname = "COLLECTION_OT_create";
/* API callbacks. */
ot->exec = collection_create_exec;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
RNA_def_string(
ot->srna, "name", "Collection", MAX_ID_NAME - 2, "Name", "Name of the new collection");
}
static bool collection_exporter_common_check(const Collection *collection)
{
return collection != nullptr &&
!(ID_IS_LINKED(&collection->id) || ID_IS_OVERRIDE_LIBRARY(&collection->id));
}
static bool collection_exporter_poll(bContext *C)
{
const Collection *collection = CTX_data_collection(C);
return collection_exporter_common_check(collection);
}
static bool collection_exporter_remove_poll(bContext *C)
{
const Collection *collection = CTX_data_collection(C);
return collection_exporter_common_check(collection) &&
!BLI_listbase_is_empty(&collection->exporters);
}
static bool collection_export_all_poll(bContext *C)
{
return CTX_data_view_layer(C) != nullptr;
}
static wmOperatorStatus collection_exporter_add_exec(bContext *C, wmOperator *op)
{
using namespace blender;
Collection *collection = CTX_data_collection(C);
ListBase *exporters = &collection->exporters;
char name[MAX_ID_NAME - 2]; /* id name */
RNA_string_get(op->ptr, "name", name);
bke::FileHandlerType *fh = bke::file_handler_find(name);
if (!fh) {
BKE_reportf(op->reports, RPT_ERROR, "File handler '%s' not found", name);
return OPERATOR_CANCELLED;
}
if (!WM_operatortype_find(fh->export_operator, true)) {
BKE_reportf(
op->reports, RPT_ERROR, "File handler operator '%s' not found", fh->export_operator);
return OPERATOR_CANCELLED;
}
/* Add a new #CollectionExport item to our handler list and fill it with #FileHandlerType
* information. Also load in the operator's properties now as well. */
CollectionExport *data = MEM_callocN<CollectionExport>("CollectionExport");
STRNCPY(data->fh_idname, fh->idname);
BKE_collection_exporter_name_set(exporters, data, fh->label);
IDPropertyTemplate val{};
data->export_properties = IDP_New(IDP_GROUP, &val, "export_properties");
data->flag |= IO_HANDLER_PANEL_OPEN;
BLI_addtail(exporters, data);
collection->active_exporter_index = BLI_listbase_count(exporters) - 1;
BKE_view_layer_need_resync_tag(CTX_data_view_layer(C));
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
WM_event_add_notifier(C, NC_SPACE | ND_SPACE_PROPERTIES, nullptr);
WM_event_add_notifier(C, NC_SPACE | ND_SPACE_OUTLINER, nullptr);
return OPERATOR_FINISHED;
}
static void COLLECTION_OT_exporter_add(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Add Exporter";
ot->description = "Add Exporter";
ot->idname = "COLLECTION_OT_exporter_add";
/* API callbacks. */
ot->exec = collection_exporter_add_exec;
ot->poll = collection_exporter_poll;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
RNA_def_string(ot->srna, "name", nullptr, MAX_ID_NAME - 2, "Name", "FileHandler idname");
}
static wmOperatorStatus collection_exporter_remove_exec(bContext *C, wmOperator *op)
{
Collection *collection = CTX_data_collection(C);
ListBase *exporters = &collection->exporters;
int index = RNA_int_get(op->ptr, "index");
CollectionExport *data = static_cast<CollectionExport *>(BLI_findlink(exporters, index));
if (!data) {
return OPERATOR_CANCELLED;
}
BLI_remlink(exporters, data);
BKE_collection_exporter_free_data(data);
MEM_freeN(data);
const int count = BLI_listbase_count(exporters);
const int new_index = count == 0 ? 0 : std::min(collection->active_exporter_index, count - 1);
collection->active_exporter_index = new_index;
BKE_view_layer_need_resync_tag(CTX_data_view_layer(C));
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
WM_event_add_notifier(C, NC_SPACE | ND_SPACE_PROPERTIES, nullptr);
WM_event_add_notifier(C, NC_SPACE | ND_SPACE_OUTLINER, nullptr);
return OPERATOR_FINISHED;
}
static wmOperatorStatus collection_exporter_remove_invoke(bContext *C,
wmOperator *op,
const wmEvent * /*event*/)
{
return WM_operator_confirm_ex(
C, op, IFACE_("Remove exporter?"), nullptr, IFACE_("Delete"), ALERT_ICON_NONE, false);
}
static void COLLECTION_OT_exporter_remove(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Remove Exporter";
ot->description = "Remove Exporter";
ot->idname = "COLLECTION_OT_exporter_remove";
/* API callbacks. */
ot->invoke = collection_exporter_remove_invoke;
ot->exec = collection_exporter_remove_exec;
ot->poll = collection_exporter_remove_poll;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
RNA_def_int(ot->srna, "index", 0, 0, INT_MAX, "Index", "Exporter index", 0, INT_MAX);
}
static wmOperatorStatus collection_exporter_export(bContext *C,
wmOperator *op,
CollectionExport *data,
Collection *collection,
const bool report_success)
{
using namespace blender;
bke::FileHandlerType *fh = bke::file_handler_find(data->fh_idname);
if (!fh) {
BKE_reportf(op->reports, RPT_ERROR, "File handler '%s' not found", data->fh_idname);
return OPERATOR_CANCELLED;
}
wmOperatorType *ot = WM_operatortype_find(fh->export_operator, false);
if (!ot) {
BKE_reportf(
op->reports, RPT_ERROR, "File handler operator '%s' not found", fh->export_operator);
return OPERATOR_CANCELLED;
}
/* Execute operator with our stored properties. */
/* TODO: Cascade settings down from parent collections(?) */
IDProperty *op_props = IDP_CopyProperty(data->export_properties);
PointerRNA properties = RNA_pointer_create_discrete(nullptr, ot->srna, op_props);
const char *collection_name = collection->id.name + 2;
/* Ensure we have a valid filepath set. Create one if the user has not specified anything yet. */
char filepath[FILE_MAX];
RNA_string_get(&properties, "filepath", filepath);
if (!filepath[0]) {
BLI_path_join(
filepath, sizeof(filepath), "//", fh->get_default_filename(collection_name).c_str());
}
else {
const char *filename = BLI_path_basename(filepath);
if (!filename[0] || !BLI_path_extension(filename)) {
BKE_reportf(op->reports, RPT_ERROR, "File path '%s' is not a valid file", filepath);
IDP_FreeProperty(op_props);
return OPERATOR_CANCELLED;
}
}
const Main *bmain = CTX_data_main(C);
BLI_path_abs(filepath, BKE_main_blendfile_path(bmain));
/* Ensure that any properties from when this operator was "last used" are cleared. Save them for
* restoration later. Otherwise properties from a regular File->Export may contaminate this
* collection export. */
IDProperty *last_properties = ot->last_properties;
ot->last_properties = nullptr;
RNA_string_set(&properties, "filepath", filepath);
RNA_string_set(&properties, "collection", collection_name);
wmOperatorStatus op_result = WM_operator_name_call_ptr(
C, ot, WM_OP_EXEC_DEFAULT, &properties, nullptr);
/* Free the "last used" properties that were just set from the collection export and restore the
* original "last used" properties. */
if (ot->last_properties) {
IDP_FreeProperty(ot->last_properties);
}
ot->last_properties = last_properties;
IDP_FreeProperty(op_props);
if (report_success && op_result == OPERATOR_FINISHED) {
BKE_reportf(op->reports, RPT_INFO, "Exported '%s'", filepath);
}
return op_result;
}
static wmOperatorStatus collection_exporter_export_exec(bContext *C, wmOperator *op)
{
Collection *collection = CTX_data_collection(C);
ListBase *exporters = &collection->exporters;
int index = RNA_int_get(op->ptr, "index");
CollectionExport *data = static_cast<CollectionExport *>(BLI_findlink(exporters, index));
if (!data) {
return OPERATOR_CANCELLED;
}
return collection_exporter_export(C, op, data, collection, true);
}
static void COLLECTION_OT_exporter_export(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Export";
ot->description = "Invoke the export operation";
ot->idname = "COLLECTION_OT_exporter_export";
/* API callbacks. */
ot->exec = collection_exporter_export_exec;
ot->poll = collection_exporter_poll;
/* flags */
ot->flag = 0;
RNA_def_int(ot->srna, "index", 0, 0, INT_MAX, "Index", "Exporter index", 0, INT_MAX);
}
struct CollectionExportStats {
int successful_exports_num = 0;
int collections_num = 0;
};
static wmOperatorStatus collection_export(bContext *C,
wmOperator *op,
Collection *collection,
CollectionExportStats &stats)
{
ListBase *exporters = &collection->exporters;
int files_num = 0;
LISTBASE_FOREACH (CollectionExport *, data, exporters) {
if (collection_exporter_export(C, op, data, collection, false) != OPERATOR_FINISHED) {
/* Do not continue calling exporters if we encounter one that fails. */
return OPERATOR_CANCELLED;
}
files_num++;
}
if (files_num) {
stats.successful_exports_num += files_num;
stats.collections_num++;
}
return OPERATOR_FINISHED;
}
static wmOperatorStatus collection_io_export_all_exec(bContext *C, wmOperator *op)
{
Collection *collection = CTX_data_collection(C);
CollectionExportStats stats;
wmOperatorStatus result = collection_export(C, op, collection, stats);
/* Only report if nothing was cancelled along the way. We don't want this UI report to happen
* over-top any reports from the actual failures. */
if (result == OPERATOR_FINISHED && stats.successful_exports_num > 0) {
BKE_reportf(op->reports,
RPT_INFO,
"Exported %d files from collection '%s'",
stats.successful_exports_num,
collection->id.name + 2);
}
return result;
}
static void COLLECTION_OT_export_all(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Export All";
ot->description = "Invoke all configured exporters on this collection";
ot->idname = "COLLECTION_OT_export_all";
/* API callbacks. */
ot->exec = collection_io_export_all_exec;
ot->poll = collection_exporter_poll;
/* flags */
ot->flag = 0;
}
static wmOperatorStatus collection_export_recursive(bContext *C,
wmOperator *op,
LayerCollection *layer_collection,
CollectionExportStats &stats)
{
/* Skip collections which have been Excluded in the View Layer. */
if (layer_collection->flag & LAYER_COLLECTION_EXCLUDE) {
return OPERATOR_FINISHED;
}
if (!collection_exporter_common_check(layer_collection->collection)) {
return OPERATOR_FINISHED;
}
if (collection_export(C, op, layer_collection->collection, stats) != OPERATOR_FINISHED) {
return OPERATOR_CANCELLED;
}
LISTBASE_FOREACH (LayerCollection *, child, &layer_collection->layer_collections) {
if (collection_export_recursive(C, op, child, stats) != OPERATOR_FINISHED) {
return OPERATOR_CANCELLED;
}
}
return OPERATOR_FINISHED;
}
static wmOperatorStatus wm_collection_export_all_exec(bContext *C, wmOperator *op)
{
ViewLayer *view_layer = CTX_data_view_layer(C);
CollectionExportStats stats;
LISTBASE_FOREACH (LayerCollection *, layer_collection, &view_layer->layer_collections) {
if (collection_export_recursive(C, op, layer_collection, stats) != OPERATOR_FINISHED) {
return OPERATOR_CANCELLED;
}
}
/* Only report if nothing was cancelled along the way. We don't want this UI report to happen
* over-top any reports from the actual failures. */
if (stats.successful_exports_num > 0) {
BKE_reportf(op->reports,
RPT_INFO,
"Exported %d files from %d collections",
stats.successful_exports_num,
stats.collections_num);
}
return OPERATOR_FINISHED;
}
static void WM_OT_collection_export_all(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Export All Collections";
ot->description = "Invoke all configured exporters for all collections";
ot->idname = "WM_OT_collection_export_all";
/* API callbacks. */
ot->exec = wm_collection_export_all_exec;
ot->poll = collection_export_all_poll;
/* flags */
ot->flag = 0;
}
static void collection_exporter_menu_draw(const bContext * /*C*/, Menu *menu)
{
using namespace blender;
uiLayout *layout = menu->layout;
/* Add all file handlers capable of being exported to the menu. */
bool at_least_one = false;
for (const auto &fh : bke::file_handlers()) {
if (WM_operatortype_find(fh->export_operator, true)) {
uiItemStringO(
layout, fh->label, ICON_NONE, "COLLECTION_OT_exporter_add", "name", fh->idname);
at_least_one = true;
}
}
if (!at_least_one) {
layout->label(IFACE_("No file handlers available"), ICON_NONE);
}
}
void collection_exporter_register()
{
MenuType *mt = MEM_callocN<MenuType>(__func__);
STRNCPY(mt->idname, "COLLECTION_MT_exporter_add");
STRNCPY(mt->label, N_("Add Exporter"));
mt->draw = collection_exporter_menu_draw;
WM_menutype_add(mt);
WM_operatortype_append(COLLECTION_OT_exporter_add);
WM_operatortype_append(COLLECTION_OT_exporter_remove);
WM_operatortype_append(COLLECTION_OT_exporter_export);
WM_operatortype_append(COLLECTION_OT_export_all);
WM_operatortype_append(WM_OT_collection_export_all);
}
/****************** properties window operators *********************/
static wmOperatorStatus collection_add_exec(bContext *C, wmOperator * /*op*/)
{
Object *ob = context_object(C);
Main *bmain = CTX_data_main(C);
if (ob == nullptr) {
return OPERATOR_CANCELLED;
}
Collection *collection = BKE_collection_add(bmain, nullptr, "Collection");
id_fake_user_set(&collection->id);
BKE_collection_object_add(bmain, collection, ob);
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_OBJECT | ND_DRAW, ob);
return OPERATOR_FINISHED;
}
void OBJECT_OT_collection_add(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Add to Collection";
ot->idname = "OBJECT_OT_collection_add";
ot->description = "Add an object to a new collection";
/* API callbacks. */
ot->exec = collection_add_exec;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
}
static wmOperatorStatus collection_link_exec(bContext *C, wmOperator *op)
{
Main *bmain = CTX_data_main(C);
Object *ob = context_object(C);
Collection *collection = static_cast<Collection *>(
BLI_findlink(&bmain->collections, RNA_enum_get(op->ptr, "collection")));
if (ELEM(nullptr, ob, collection)) {
return OPERATOR_CANCELLED;
}
/* Early return check, if the object is already in collection
* we could skip all the dependency check and just consider
* operator is finished.
*/
if (BKE_collection_has_object(collection, ob)) {
return OPERATOR_FINISHED;
}
/* Currently this should not be allowed (might be supported in the future though...). */
if (ID_IS_OVERRIDE_LIBRARY(&collection->id)) {
BKE_report(op->reports, RPT_ERROR, "Could not add the collection because it is overridden");
return OPERATOR_CANCELLED;
}
/* Linked collections are already checked for by using RNA_collection_local_itemf
* but operator can be called without invoke */
if (!ID_IS_EDITABLE(&collection->id)) {
BKE_report(op->reports, RPT_ERROR, "Could not add the collection because it is linked");
return OPERATOR_CANCELLED;
}
/* Adding object to collection which is used as dupli-collection for self is bad idea.
*
* It is also bad idea to add object to collection which is in collection which
* contains our current object.
*/
if (BKE_collection_object_cyclic_check(bmain, ob, collection)) {
BKE_report(op->reports,
RPT_ERROR,
"Could not add the collection because of dependency cycle detected");
return OPERATOR_CANCELLED;
}
BKE_collection_object_add(bmain, collection, ob);
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_OBJECT | ND_DRAW, ob);
return OPERATOR_FINISHED;
}
void OBJECT_OT_collection_link(wmOperatorType *ot)
{
PropertyRNA *prop;
/* identifiers */
ot->name = "Link to Collection";
ot->idname = "OBJECT_OT_collection_link";
ot->description = "Add an object to an existing collection";
/* API callbacks. */
ot->exec = collection_link_exec;
ot->invoke = WM_enum_search_invoke;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
/* properties */
prop = RNA_def_enum(ot->srna, "collection", rna_enum_dummy_NULL_items, 0, "Collection", "");
RNA_def_enum_funcs(prop, RNA_collection_local_itemf);
RNA_def_property_flag(prop, PROP_ENUM_NO_TRANSLATE);
ot->prop = prop;
}
static wmOperatorStatus collection_remove_exec(bContext *C, wmOperator *op)
{
Main *bmain = CTX_data_main(C);
Object *ob = context_object(C);
Collection *collection = static_cast<Collection *>(
CTX_data_pointer_get_type(C, "collection", &RNA_Collection).data);
if (!ob || !collection) {
return OPERATOR_CANCELLED;
}
if (!ID_IS_EDITABLE(collection) || ID_IS_OVERRIDE_LIBRARY(collection)) {
BKE_report(op->reports,
RPT_ERROR,
"Cannot remove an object from a linked or library override collection");
return OPERATOR_CANCELLED;
}
BKE_collection_object_remove(bmain, collection, ob, false);
DEG_id_tag_update(&collection->id, ID_RECALC_SYNC_TO_EVAL);
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_OBJECT | ND_DRAW, ob);
return OPERATOR_FINISHED;
}
void OBJECT_OT_collection_remove(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Remove Collection";
ot->idname = "OBJECT_OT_collection_remove";
ot->description = "Remove the active object from this collection";
/* API callbacks. */
ot->exec = collection_remove_exec;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
}
static wmOperatorStatus collection_unlink_exec(bContext *C, wmOperator *op)
{
Main *bmain = CTX_data_main(C);
Collection *collection = CTX_data_collection(C);
if (!collection) {
return OPERATOR_CANCELLED;
}
if (collection->flag & COLLECTION_IS_MASTER) {
return OPERATOR_CANCELLED;
}
BLI_assert((collection->id.flag & ID_FLAG_EMBEDDED_DATA) == 0);
if (ID_IS_OVERRIDE_LIBRARY(collection) &&
collection->id.override_library->hierarchy_root != &collection->id)
{
BKE_report(op->reports,
RPT_ERROR,
"Cannot unlink a library override collection which is not the root of its override "
"hierarchy");
return OPERATOR_CANCELLED;
}
BKE_id_delete(bmain, collection);
DEG_relations_tag_update(bmain);
WM_event_add_notifier(C, NC_OBJECT | ND_DRAW, nullptr);
return OPERATOR_FINISHED;
}
static bool collection_unlink_poll(bContext *C)
{
Collection *collection = CTX_data_collection(C);
if (!collection) {
return false;
}
if (collection->flag & COLLECTION_IS_MASTER) {
return false;
}
BLI_assert((collection->id.flag & ID_FLAG_EMBEDDED_DATA) == 0);
if (ID_IS_OVERRIDE_LIBRARY(collection) &&
collection->id.override_library->hierarchy_root != &collection->id)
{
return false;
}
return ED_operator_objectmode(C);
}
void OBJECT_OT_collection_unlink(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Unlink Collection";
ot->idname = "OBJECT_OT_collection_unlink";
ot->description = "Unlink the collection from all objects";
/* API callbacks. */
ot->exec = collection_unlink_exec;
ot->poll = collection_unlink_poll;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
}
/* Select objects in the same collection as the active */
static wmOperatorStatus select_grouped_exec(bContext *C, wmOperator * /*op*/)
{
Scene *scene = CTX_data_scene(C);
Collection *collection = static_cast<Collection *>(
CTX_data_pointer_get_type(C, "collection", &RNA_Collection).data);
if (!collection) {
return OPERATOR_CANCELLED;
}
CTX_DATA_BEGIN (C, Base *, base, visible_bases) {
if (((base->flag & BASE_SELECTED) == 0) && ((base->flag & BASE_SELECTABLE) != 0)) {
if (BKE_collection_has_object_recursive(collection, base->object)) {
base_select(base, BA_SELECT);
}
}
}
CTX_DATA_END;
DEG_id_tag_update(&scene->id, ID_RECALC_SELECT);
WM_main_add_notifier(NC_SCENE | ND_OB_SELECT, scene);
return OPERATOR_FINISHED;
}
void OBJECT_OT_collection_objects_select(wmOperatorType *ot)
{
/* identifiers */
ot->name = "Select Objects in Collection";
ot->idname = "OBJECT_OT_collection_objects_select";
ot->description = "Select all objects in collection";
/* API callbacks. */
ot->exec = select_grouped_exec;
ot->poll = ED_operator_objectmode;
/* flags */
ot->flag = OPTYPE_REGISTER | OPTYPE_UNDO;
}
} // namespace blender::ed::object