Allow keying properties with multiple nodes selected

This commit is contained in:
Mikael Hermansson 2024-06-06 17:47:15 +02:00
parent 7a4a6fbc03
commit 351f454a94
4 changed files with 47 additions and 109 deletions

View File

@ -41,6 +41,7 @@
#include "editor/gui/editor_spin_slider.h" #include "editor/gui/editor_spin_slider.h"
#include "editor/gui/scene_tree_editor.h" #include "editor/gui/scene_tree_editor.h"
#include "editor/inspector_dock.h" #include "editor/inspector_dock.h"
#include "editor/multi_node_edit.h"
#include "editor/plugins/animation_player_editor_plugin.h" #include "editor/plugins/animation_player_editor_plugin.h"
#include "editor/themes/editor_scale.h" #include "editor/themes/editor_scale.h"
#include "scene/3d/mesh_instance_3d.h" #include "scene/3d/mesh_instance_3d.h"
@ -3627,7 +3628,7 @@ void AnimationTrackEditor::update_keying() {
EditorSelectionHistory *editor_history = EditorNode::get_singleton()->get_editor_selection_history(); EditorSelectionHistory *editor_history = EditorNode::get_singleton()->get_editor_selection_history();
if (is_visible_in_tree() && animation.is_valid() && editor_history->get_path_size() > 0) { if (is_visible_in_tree() && animation.is_valid() && editor_history->get_path_size() > 0) {
Object *obj = ObjectDB::get_instance(editor_history->get_path_object(0)); Object *obj = ObjectDB::get_instance(editor_history->get_path_object(0));
keying_enabled = Object::cast_to<Node>(obj) != nullptr; keying_enabled = Object::cast_to<Node>(obj) != nullptr || Object::cast_to<MultiNodeEdit>(obj) != nullptr;
} }
if (keying_enabled == keying) { if (keying_enabled == keying) {
@ -4044,19 +4045,20 @@ void AnimationTrackEditor::_insert_animation_key(NodePath p_path, const Variant
_query_insert(id); _query_insert(id);
} }
void AnimationTrackEditor::insert_node_value_key(Node *p_node, const String &p_property, const Variant &p_value, bool p_only_if_exists) { void AnimationTrackEditor::insert_node_value_key(Node *p_node, const String &p_property, bool p_only_if_exists, bool p_advance) {
ERR_FAIL_NULL(root); ERR_FAIL_NULL(root);
// Let's build a node path. // Let's build a node path.
Node *node = p_node; String path = root->get_path_to(p_node, true);
String path = root->get_path_to(node, true);
if (Object::cast_to<AnimationPlayer>(node) && p_property == "current_animation") { Variant value = p_node->get(p_property);
if (node == AnimationPlayerEditor::get_singleton()->get_player()) {
if (Object::cast_to<AnimationPlayer>(p_node) && p_property == "current_animation") {
if (p_node == AnimationPlayerEditor::get_singleton()->get_player()) {
EditorNode::get_singleton()->show_warning(TTR("AnimationPlayer can't animate itself, only other players.")); EditorNode::get_singleton()->show_warning(TTR("AnimationPlayer can't animate itself, only other players."));
return; return;
} }
_insert_animation_key(path, p_value); _insert_animation_key(path, value);
return; return;
} }
@ -4084,26 +4086,26 @@ void AnimationTrackEditor::insert_node_value_key(Node *p_node, const String &p_p
InsertData id; InsertData id;
id.path = np; id.path = np;
id.track_idx = i; id.track_idx = i;
id.value = p_value; id.value = value;
id.type = Animation::TYPE_VALUE; id.type = Animation::TYPE_VALUE;
// TRANSLATORS: This describes the target of new animation track, will be inserted into another string. // TRANSLATORS: This describes the target of new animation track, will be inserted into another string.
id.query = vformat(TTR("property '%s'"), p_property); id.query = vformat(TTR("property '%s'"), p_property);
id.advance = false; id.advance = p_advance;
// Dialog insert. // Dialog insert.
_query_insert(id); _query_insert(id);
inserted = true; inserted = true;
} else if (animation->track_get_type(i) == Animation::TYPE_BEZIER) { } else if (animation->track_get_type(i) == Animation::TYPE_BEZIER) {
Variant value; Variant actual_value;
String track_path = animation->track_get_path(i); String track_path = animation->track_get_path(i);
if (track_path == np) { if (track_path == np) {
value = p_value; // All good. actual_value = value; // All good.
} else { } else {
int sep = track_path.rfind(":"); int sep = track_path.rfind(":");
if (sep != -1) { if (sep != -1) {
String base_path = track_path.substr(0, sep); String base_path = track_path.substr(0, sep);
if (base_path == np) { if (base_path == np) {
String value_name = track_path.substr(sep + 1); String value_name = track_path.substr(sep + 1);
value = p_value.get(value_name); actual_value = value.get(value_name);
} else { } else {
continue; continue;
} }
@ -4115,10 +4117,10 @@ void AnimationTrackEditor::insert_node_value_key(Node *p_node, const String &p_p
InsertData id; InsertData id;
id.path = animation->track_get_path(i); id.path = animation->track_get_path(i);
id.track_idx = i; id.track_idx = i;
id.value = value; id.value = actual_value;
id.type = Animation::TYPE_BEZIER; id.type = Animation::TYPE_BEZIER;
id.query = vformat(TTR("property '%s'"), p_property); id.query = vformat(TTR("property '%s'"), p_property);
id.advance = false; id.advance = p_advance;
// Dialog insert. // Dialog insert.
_query_insert(id); _query_insert(id);
inserted = true; inserted = true;
@ -4131,105 +4133,41 @@ void AnimationTrackEditor::insert_node_value_key(Node *p_node, const String &p_p
InsertData id; InsertData id;
id.path = np; id.path = np;
id.track_idx = -1; id.track_idx = -1;
id.value = p_value; id.value = value;
id.type = Animation::TYPE_VALUE; id.type = Animation::TYPE_VALUE;
id.query = vformat(TTR("property '%s'"), p_property); id.query = vformat(TTR("property '%s'"), p_property);
id.advance = false; id.advance = p_advance;
// Dialog insert. // Dialog insert.
_query_insert(id); _query_insert(id);
} }
void AnimationTrackEditor::insert_value_key(const String &p_property, const Variant &p_value, bool p_advance) { void AnimationTrackEditor::insert_value_key(const String &p_property, bool p_advance) {
EditorSelectionHistory *history = EditorNode::get_singleton()->get_editor_selection_history(); EditorSelectionHistory *history = EditorNode::get_singleton()->get_editor_selection_history();
ERR_FAIL_NULL(root); ERR_FAIL_NULL(root);
ERR_FAIL_COND(history->get_path_size() == 0); ERR_FAIL_COND(history->get_path_size() == 0);
Object *obj = ObjectDB::get_instance(history->get_path_object(0)); Object *obj = ObjectDB::get_instance(history->get_path_object(0));
ERR_FAIL_NULL(Object::cast_to<Node>(obj));
// Let's build a node path. Ref<MultiNodeEdit> multi_node_edit(obj);
Node *node = Object::cast_to<Node>(obj); if (multi_node_edit.is_valid()) {
String path = root->get_path_to(node, true); Node *edited_scene = EditorNode::get_singleton()->get_edited_scene();
ERR_FAIL_NULL(edited_scene);
if (Object::cast_to<AnimationPlayer>(node) && p_property == "current_animation") {
if (node == AnimationPlayerEditor::get_singleton()->get_player()) {
EditorNode::get_singleton()->show_warning(TTR("AnimationPlayer can't animate itself, only other players."));
return;
}
_insert_animation_key(path, p_value);
return;
}
for (int i = 1; i < history->get_path_size(); i++) {
String prop = history->get_path_property(i);
ERR_FAIL_COND(prop.is_empty());
path += ":" + prop;
}
path += ":" + p_property;
NodePath np = path;
// Locate track.
bool inserted = false;
make_insert_queue(); make_insert_queue();
for (int i = 0; i < animation->get_track_count(); i++) {
if (animation->track_get_type(i) == Animation::TYPE_VALUE) { for (int i = 0; i < multi_node_edit->get_node_count(); ++i) {
if (animation->track_get_path(i) != np) { Node *node = edited_scene->get_node(multi_node_edit->get_node(i));
continue; insert_node_value_key(node, p_property, false, p_advance);
} }
InsertData id;
id.path = np;
id.track_idx = i;
id.value = p_value;
id.type = Animation::TYPE_VALUE;
id.query = vformat(TTR("property '%s'"), p_property);
id.advance = p_advance;
// Dialog insert.
_query_insert(id);
inserted = true;
} else if (animation->track_get_type(i) == Animation::TYPE_BEZIER) {
Variant value;
if (animation->track_get_path(i) == np) {
value = p_value; // All good.
} else {
String tpath = animation->track_get_path(i);
int index = tpath.rfind(":");
if (NodePath(tpath.substr(0, index + 1)) == np) {
String subindex = tpath.substr(index + 1, tpath.length() - index);
value = p_value.get(subindex);
} else {
continue;
}
}
InsertData id;
id.path = animation->track_get_path(i);
id.track_idx = i;
id.value = value;
id.type = Animation::TYPE_BEZIER;
id.query = vformat(TTR("property '%s'"), p_property);
id.advance = p_advance;
// Dialog insert.
_query_insert(id);
inserted = true;
}
}
commit_insert_queue(); commit_insert_queue();
} else {
Node *node = Object::cast_to<Node>(obj);
ERR_FAIL_NULL(node);
if (!inserted) { make_insert_queue();
InsertData id; insert_node_value_key(node, p_property, false, p_advance);
id.path = np; commit_insert_queue();
id.track_idx = -1;
id.value = p_value;
id.type = Animation::TYPE_VALUE;
id.query = vformat(TTR("property '%s'"), p_property);
id.advance = p_advance;
// Dialog insert.
_query_insert(id);
} }
} }

View File

@ -712,8 +712,8 @@ public:
void cleanup(); void cleanup();
void set_anim_pos(float p_pos); void set_anim_pos(float p_pos);
void insert_node_value_key(Node *p_node, const String &p_property, const Variant &p_value, bool p_only_if_exists = false); void insert_node_value_key(Node *p_node, const String &p_property, bool p_only_if_exists = false, bool p_advance = false);
void insert_value_key(const String &p_property, const Variant &p_value, bool p_advance); void insert_value_key(const String &p_property, bool p_advance);
void insert_transform_key(Node3D *p_node, const String &p_sub, const Animation::TrackType p_type, const Variant &p_value); void insert_transform_key(Node3D *p_node, const String &p_sub, const Animation::TrackType p_type, const Variant &p_value);
bool has_track(Node3D *p_node, const String &p_sub, const Animation::TrackType p_type); bool has_track(Node3D *p_node, const String &p_sub, const Animation::TrackType p_type);
void make_insert_queue(); void make_insert_queue();

View File

@ -2165,7 +2165,7 @@ void AnimationPlayerEditorPlugin::_property_keyed(const String &p_keyed, const V
return; return;
} }
te->_clear_selection(); te->_clear_selection();
te->insert_value_key(p_keyed, p_value, p_advance); te->insert_value_key(p_keyed, p_advance);
} }
void AnimationPlayerEditorPlugin::_transform_key_request(Object *sp, const String &p_sub, const Transform3D &p_key) { void AnimationPlayerEditorPlugin::_transform_key_request(Object *sp, const String &p_sub, const Transform3D &p_key) {

View File

@ -4319,13 +4319,13 @@ void CanvasItemEditor::_insert_animation_keys(bool p_location, bool p_rotation,
Node2D *n2d = Object::cast_to<Node2D>(ci); Node2D *n2d = Object::cast_to<Node2D>(ci);
if (key_pos && p_location) { if (key_pos && p_location) {
te->insert_node_value_key(n2d, "position", n2d->get_position(), p_on_existing); te->insert_node_value_key(n2d, "position", p_on_existing);
} }
if (key_rot && p_rotation) { if (key_rot && p_rotation) {
te->insert_node_value_key(n2d, "rotation", n2d->get_rotation(), p_on_existing); te->insert_node_value_key(n2d, "rotation", p_on_existing);
} }
if (key_scale && p_scale) { if (key_scale && p_scale) {
te->insert_node_value_key(n2d, "scale", n2d->get_scale(), p_on_existing); te->insert_node_value_key(n2d, "scale", p_on_existing);
} }
if (n2d->has_meta("_edit_bone_") && n2d->get_parent_item()) { if (n2d->has_meta("_edit_bone_") && n2d->get_parent_item()) {
@ -4351,13 +4351,13 @@ void CanvasItemEditor::_insert_animation_keys(bool p_location, bool p_rotation,
if (has_chain && ik_chain.size()) { if (has_chain && ik_chain.size()) {
for (Node2D *&F : ik_chain) { for (Node2D *&F : ik_chain) {
if (key_pos) { if (key_pos) {
te->insert_node_value_key(F, "position", F->get_position(), p_on_existing); te->insert_node_value_key(F, "position", p_on_existing);
} }
if (key_rot) { if (key_rot) {
te->insert_node_value_key(F, "rotation", F->get_rotation(), p_on_existing); te->insert_node_value_key(F, "rotation", p_on_existing);
} }
if (key_scale) { if (key_scale) {
te->insert_node_value_key(F, "scale", F->get_scale(), p_on_existing); te->insert_node_value_key(F, "scale", p_on_existing);
} }
} }
} }
@ -4367,13 +4367,13 @@ void CanvasItemEditor::_insert_animation_keys(bool p_location, bool p_rotation,
Control *ctrl = Object::cast_to<Control>(ci); Control *ctrl = Object::cast_to<Control>(ci);
if (key_pos) { if (key_pos) {
te->insert_node_value_key(ctrl, "position", ctrl->get_position(), p_on_existing); te->insert_node_value_key(ctrl, "position", p_on_existing);
} }
if (key_rot) { if (key_rot) {
te->insert_node_value_key(ctrl, "rotation", ctrl->get_rotation(), p_on_existing); te->insert_node_value_key(ctrl, "rotation", p_on_existing);
} }
if (key_scale) { if (key_scale) {
te->insert_node_value_key(ctrl, "size", ctrl->get_size(), p_on_existing); te->insert_node_value_key(ctrl, "size", p_on_existing);
} }
} }
} }