mirror of
https://github.com/Relintai/procedural_animations.git
synced 2024-11-13 08:27:19 +01:00
481 lines
14 KiB
C++
481 lines
14 KiB
C++
/*
|
|
Copyright (c) 2020 Péter Magyar
|
|
|
|
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
of this software and associated documentation files (the "Software"), to deal
|
|
in the Software without restriction, including without limitation the rights
|
|
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
copies of the Software, and to permit persons to whom the Software is
|
|
furnished to do so, subject to the following conditions:
|
|
|
|
The above copyright notice and this permission notice shall be included in all
|
|
copies or substantial portions of the Software.
|
|
|
|
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
SOFTWARE.
|
|
*/
|
|
|
|
#include "procedural_animation_editor_plugin.h"
|
|
|
|
#include "editor/editor_scale.h"
|
|
|
|
// S -------- ProceduralAnimationEditor --------
|
|
|
|
void ProceduralAnimationEditor::edit(const Ref<ProceduralAnimation> &animation) {
|
|
_animation = animation;
|
|
|
|
clear_keyframe_nodes();
|
|
|
|
if (animation.is_valid())
|
|
load_animation();
|
|
}
|
|
|
|
void ProceduralAnimationEditor::load_animation() {
|
|
clear_keyframe_nodes();
|
|
|
|
ERR_FAIL_COND(!_animation.is_valid());
|
|
|
|
_start_node->set_offset(_animation->get_start_node_position());
|
|
|
|
const PoolVector<String> &animation_names = _animation->get_animation_keyframe_names();
|
|
|
|
PoolVector<int> kfind = _animation->get_keyframe_indices();
|
|
|
|
for (int i = 0; i < kfind.size(); ++i) {
|
|
int id = kfind[i];
|
|
|
|
ProceduralAnimationEditorGraphNode *gn = memnew(ProceduralAnimationEditorGraphNode);
|
|
_graph_edit->add_child(gn);
|
|
gn->set_name(String::num(id));
|
|
gn->set_id(id);
|
|
gn->set_animation(_animation);
|
|
}
|
|
|
|
for (int i = 0; i < kfind.size(); ++i) {
|
|
int id = kfind[i];
|
|
|
|
int ni = _animation->get_keyframe_next_keyframe_index(id);
|
|
|
|
if (ni != -1)
|
|
_graph_edit->connect_node(String::num(id), 0, String::num(ni), 0);
|
|
}
|
|
|
|
int st = _animation->get_start_frame_index();
|
|
|
|
if (st != -1)
|
|
_graph_edit->connect_node("Start", 0, String::num(st), 0);
|
|
}
|
|
|
|
void ProceduralAnimationEditor::clear_keyframe_nodes() {
|
|
_graph_edit->clear_connections();
|
|
|
|
for (int i = 0; i < _graph_edit->get_child_count(); ++i) {
|
|
Node *n = _graph_edit->get_child(i);
|
|
|
|
if (n == _start_node) {
|
|
continue;
|
|
}
|
|
|
|
ProceduralAnimationEditorGraphNode *gn = Object::cast_to<ProceduralAnimationEditorGraphNode>(n);
|
|
|
|
if (!ObjectDB::instance_validate(gn)) {
|
|
continue;
|
|
}
|
|
|
|
gn->set_name("d" + gn->get_name());
|
|
gn->queue_delete();
|
|
}
|
|
}
|
|
|
|
void ProceduralAnimationEditor::on_delete_popup_confirmed() {
|
|
switch (_delete_popup_action) {
|
|
case DELETE_POPUP_KEYFRAME:
|
|
break;
|
|
}
|
|
}
|
|
|
|
void ProceduralAnimationEditor::on_connection_request(const String &from, const int from_slot, const String &to, const int to_slot) {
|
|
Node *f = _graph_edit->get_node_or_null(from);
|
|
|
|
ProceduralAnimationEditorGraphNode *gn = Object::cast_to<ProceduralAnimationEditorGraphNode>(f);
|
|
|
|
if (gn != NULL) {
|
|
int ni = _animation->get_keyframe_next_keyframe_index(gn->get_id());
|
|
|
|
if (ni != -1) {
|
|
_graph_edit->disconnect_node(from, from_slot, String::num(ni), 0);
|
|
}
|
|
|
|
_animation->set_keyframe_next_keyframe_index(gn->get_id(), to.to_int());
|
|
} else {
|
|
GraphNode *g = Object::cast_to<GraphNode>(f);
|
|
|
|
if (g != NULL) {
|
|
int st = _animation->get_start_frame_index();
|
|
|
|
if (st != -1) {
|
|
_graph_edit->disconnect_node("Start", 0, String::num(st), 0);
|
|
}
|
|
|
|
_animation->set_start_frame_index(to.to_int());
|
|
}
|
|
}
|
|
|
|
_graph_edit->connect_node(from, from_slot, to, to_slot);
|
|
}
|
|
void ProceduralAnimationEditor::on_disconnection_request(const String &from, const int from_slot, const String &to, const int to_slot) {
|
|
Node *f = _graph_edit->get_node_or_null(from);
|
|
|
|
ProceduralAnimationEditorGraphNode *gn = Object::cast_to<ProceduralAnimationEditorGraphNode>(f);
|
|
|
|
if (gn != NULL) {
|
|
_animation->set_keyframe_next_keyframe_index(gn->get_id(), -1);
|
|
} else {
|
|
GraphNode *g = Object::cast_to<GraphNode>(f);
|
|
|
|
if (g != NULL) {
|
|
_animation->set_start_frame_index(-1);
|
|
}
|
|
}
|
|
|
|
_graph_edit->disconnect_node(from, from_slot, to, to_slot);
|
|
}
|
|
|
|
void ProceduralAnimationEditor::add_frame_button_pressed() {
|
|
int id = _animation->add_keyframe();
|
|
|
|
ProceduralAnimationEditorGraphNode *gn = memnew(ProceduralAnimationEditorGraphNode);
|
|
gn->set_id(id);
|
|
_graph_edit->add_child(gn);
|
|
}
|
|
|
|
void ProceduralAnimationEditor::_notification(int p_what) {
|
|
|
|
switch (p_what) {
|
|
case NOTIFICATION_THEME_CHANGED: {
|
|
_pin->set_icon(get_icon("Pin", "EditorIcons"));
|
|
} break;
|
|
}
|
|
}
|
|
|
|
void ProceduralAnimationEditor::_bind_methods() {
|
|
ClassDB::bind_method(D_METHOD("on_delete_popup_confirmed"), &ProceduralAnimationEditor::on_delete_popup_confirmed);
|
|
|
|
ClassDB::bind_method(D_METHOD("add_frame_button_pressed"), &ProceduralAnimationEditor::add_frame_button_pressed);
|
|
|
|
ClassDB::bind_method(D_METHOD("on_connection_request", "from", "from_slot", "to", "to_slot"), &ProceduralAnimationEditor::on_connection_request);
|
|
ClassDB::bind_method(D_METHOD("on_disconnection_request", "from", "from_slot", "to", "to_slot"), &ProceduralAnimationEditor::on_disconnection_request);
|
|
}
|
|
|
|
ProceduralAnimationEditor::ProceduralAnimationEditor() {
|
|
}
|
|
|
|
ProceduralAnimationEditor::ProceduralAnimationEditor(EditorNode *p_editor) {
|
|
set_h_size_flags(SIZE_EXPAND_FILL);
|
|
|
|
//top bar
|
|
HBoxContainer *hbc = memnew(HBoxContainer);
|
|
add_child(hbc);
|
|
|
|
Control *spacer = memnew(Control);
|
|
spacer->set_h_size_flags(SIZE_EXPAND_FILL);
|
|
hbc->add_child(spacer);
|
|
|
|
Button *aafb = memnew(Button);
|
|
aafb->set_text("add frame");
|
|
aafb->connect("pressed", this, "add_frame_button_pressed");
|
|
hbc->add_child(aafb);
|
|
|
|
_pin = memnew(ToolButton);
|
|
_pin->set_toggle_mode(true);
|
|
_pin->set_tooltip(TTR("Pin"));
|
|
hbc->add_child(_pin);
|
|
|
|
//bottom
|
|
_graph_edit = memnew(GraphEdit);
|
|
_graph_edit->set_right_disconnects(true);
|
|
_graph_edit->set_h_size_flags(SIZE_EXPAND_FILL);
|
|
_graph_edit->set_v_size_flags(SIZE_EXPAND_FILL);
|
|
_graph_edit->set_custom_minimum_size(Size2(0, 200) * EDSCALE);
|
|
_graph_edit->connect("connection_request", this, "on_connection_request");
|
|
_graph_edit->connect("disconnection_request", this, "on_disconnection_request");
|
|
add_child(_graph_edit);
|
|
|
|
_start_node = memnew(GraphNode);
|
|
_start_node->set_name("Start");
|
|
_start_node->set_title("Start");
|
|
_start_node->set_show_close_button(false);
|
|
_graph_edit->add_child(_start_node);
|
|
|
|
Label *slb = memnew(Label);
|
|
slb->set_text("Frame");
|
|
_start_node->add_child(slb);
|
|
|
|
_start_node->set_slot(0, false, 0, Color(0, 1, 0), true, 0, Color(0, 1, 0));
|
|
|
|
Control *popups = memnew(Control);
|
|
add_child(popups);
|
|
|
|
//delete confirm popup
|
|
_delete_popuop = memnew(ConfirmationDialog);
|
|
_delete_popuop->connect("confirmed", this, "on_delete_popup_confirmed");
|
|
popups->add_child(_delete_popuop);
|
|
|
|
Label *dellabel = memnew(Label);
|
|
dellabel->set_text("Delete?");
|
|
_delete_popuop->add_child(dellabel);
|
|
|
|
//name popup
|
|
_name_popuop = memnew(ConfirmationDialog);
|
|
_name_popuop->connect("confirmed", this, "on_name_popup_confirmed");
|
|
popups->add_child(_name_popuop);
|
|
|
|
VBoxContainer *name_popup_container = memnew(VBoxContainer);
|
|
_name_popuop->add_child(name_popup_container);
|
|
|
|
_name_pupup_label = memnew(Label);
|
|
_name_pupup_label->set_text("Enter name");
|
|
name_popup_container->add_child(_name_pupup_label);
|
|
|
|
_name_popup_line_edit = memnew(LineEdit);
|
|
name_popup_container->add_child(_name_popup_line_edit);
|
|
}
|
|
|
|
ProceduralAnimationEditor::~ProceduralAnimationEditor() {
|
|
}
|
|
|
|
// E -------- ProceduralAnimationEditor --------
|
|
|
|
// S -------- ProceduralAnimationEditorGraphNode --------
|
|
|
|
int ProceduralAnimationEditorGraphNode::get_id() const {
|
|
return _id;
|
|
}
|
|
void ProceduralAnimationEditorGraphNode::set_id(const int id) {
|
|
_id = id;
|
|
}
|
|
|
|
String ProceduralAnimationEditorGraphNode::get_keyframe_name() const {
|
|
return _name->get_text();
|
|
}
|
|
void ProceduralAnimationEditorGraphNode::set_keyframe_name(const String &value) {
|
|
_name->set_text(value);
|
|
|
|
if (!_animation.is_valid())
|
|
return;
|
|
|
|
_animation->set_keyframe_name(_id, value);
|
|
|
|
changed();
|
|
}
|
|
void ProceduralAnimationEditorGraphNode::on_keyframe_name_modified(const String &value) {
|
|
set_keyframe_name(value);
|
|
}
|
|
|
|
int ProceduralAnimationEditorGraphNode::get_animation_keyframe_index() const {
|
|
return _animation_keyframe_index;
|
|
}
|
|
void ProceduralAnimationEditorGraphNode::set_animation_keyframe_index(const int value) {
|
|
if (_animation_keyframe_index == value)
|
|
return;
|
|
|
|
_animation_keyframe_spinbox->set_value(value);
|
|
|
|
_animation_keyframe_index = value;
|
|
|
|
if (!_animation.is_valid())
|
|
return;
|
|
|
|
_animation->set_keyframe_animation_keyframe_index(_id, value);
|
|
|
|
changed();
|
|
}
|
|
|
|
int ProceduralAnimationEditorGraphNode::get_next_keyframe() const {
|
|
return _next_keyframe;
|
|
}
|
|
void ProceduralAnimationEditorGraphNode::set_next_keyframe(const int value) {
|
|
_next_keyframe = value;
|
|
|
|
if (!_animation.is_valid())
|
|
return;
|
|
|
|
_animation->set_keyframe_next_keyframe_index(_id, value);
|
|
|
|
changed();
|
|
}
|
|
|
|
float ProceduralAnimationEditorGraphNode::get_transition() const {
|
|
return _transition;
|
|
}
|
|
void ProceduralAnimationEditorGraphNode::set_transition(const float value) {
|
|
_transition = value;
|
|
|
|
if (!_animation.is_valid())
|
|
return;
|
|
|
|
_animation->set_keyframe_transition(_id, value);
|
|
|
|
changed();
|
|
}
|
|
|
|
Ref<ProceduralAnimation> ProceduralAnimationEditorGraphNode::get_animation() {
|
|
return _animation;
|
|
}
|
|
void ProceduralAnimationEditorGraphNode::set_animation(const Ref<ProceduralAnimation> &animation) {
|
|
_animation.unref();
|
|
|
|
if (!animation.is_valid())
|
|
return;
|
|
|
|
set_offset(animation->get_keyframe_node_position(_id));
|
|
set_keyframe_name(animation->get_keyframe_name(_id));
|
|
set_next_keyframe(animation->get_keyframe_next_keyframe_index(_id));
|
|
set_transition(animation->get_keyframe_transition(_id));
|
|
set_animation_keyframe_index(animation->get_keyframe_animation_keyframe_index(_id));
|
|
|
|
_animation = animation;
|
|
}
|
|
|
|
ProceduralAnimationEditorGraphNode::ProceduralAnimationEditorGraphNode() {
|
|
_id = 0;
|
|
|
|
_animation_keyframe_index = 0;
|
|
_next_keyframe = 0;
|
|
|
|
set_title("Animation Frame");
|
|
set_show_close_button(true);
|
|
|
|
Label *l1 = memnew(Label);
|
|
l1->set_text("Name");
|
|
add_child(l1);
|
|
|
|
_name = memnew(LineEdit);
|
|
_name->connect("text_entered", this, "on_keyframe_name_modified");
|
|
add_child(_name);
|
|
|
|
Label *l2 = memnew(Label);
|
|
l2->set_text("Keyframe");
|
|
add_child(l2);
|
|
|
|
_animation_keyframe_spinbox = memnew(SpinBox);
|
|
_animation_keyframe_spinbox->set_max(999999999);
|
|
_animation_keyframe_spinbox->set_h_size_flags(SIZE_EXPAND_FILL);
|
|
_animation_keyframe_spinbox->connect("value_changed", this, "on_animation_keyframe_spinbox_value_changed");
|
|
add_child(_animation_keyframe_spinbox);
|
|
|
|
Label *l3 = memnew(Label);
|
|
l3->set_text("Easing");
|
|
add_child(l3);
|
|
|
|
_curve_editor = memnew(CurveEditor);
|
|
_curve_editor->set_custom_minimum_size(Size2(0, 69) * EDSCALE);
|
|
add_child(_curve_editor);
|
|
|
|
set_slot(0, true, 0, Color(0, 1, 0), true, 0, Color(0, 1, 0));
|
|
}
|
|
|
|
ProceduralAnimationEditorGraphNode::~ProceduralAnimationEditorGraphNode() {
|
|
}
|
|
|
|
void ProceduralAnimationEditorGraphNode::on_animation_keyframe_spinbox_value_changed(float value) {
|
|
set_animation_keyframe_index(value);
|
|
}
|
|
|
|
void ProceduralAnimationEditorGraphNode::on_dragged(Vector2 from, Vector2 to) {
|
|
if (!_animation.is_valid())
|
|
return;
|
|
|
|
_animation->set_keyframe_node_position(_id, get_offset());
|
|
|
|
changed();
|
|
}
|
|
|
|
void ProceduralAnimationEditorGraphNode::changed() {
|
|
emit_signal("graphnode_changed", this);
|
|
}
|
|
|
|
void ProceduralAnimationEditorGraphNode::_notification(int p_what) {
|
|
switch (p_what) {
|
|
case NOTIFICATION_ENTER_TREE:
|
|
connect("offset_changed", this, "changed");
|
|
break;
|
|
}
|
|
}
|
|
|
|
void ProceduralAnimationEditorGraphNode::_bind_methods() {
|
|
ADD_SIGNAL(MethodInfo("graphnode_changed", PropertyInfo(Variant::OBJECT, "node", PROPERTY_HINT_RESOURCE_TYPE, "Node")));
|
|
|
|
ClassDB::bind_method(D_METHOD("on_keyframe_name_modified", "name"), &ProceduralAnimationEditorGraphNode::on_keyframe_name_modified);
|
|
ClassDB::bind_method(D_METHOD("changed"), &ProceduralAnimationEditorGraphNode::changed);
|
|
ClassDB::bind_method(D_METHOD("on_animation_keyframe_spinbox_value_changed", "value"), &ProceduralAnimationEditorGraphNode::on_animation_keyframe_spinbox_value_changed);
|
|
|
|
ClassDB::bind_method(D_METHOD("on_dragged", "from", "to"), &ProceduralAnimationEditorGraphNode::on_dragged);
|
|
}
|
|
|
|
// E -------- ProceduralAnimationEditorGraphNode --------
|
|
|
|
// S -------- ProceduralAnimationEditorPlugin --------
|
|
|
|
void ProceduralAnimationEditorPlugin::edit(Object *p_object) {
|
|
if (Object::cast_to<ProceduralAnimation>(p_object)) {
|
|
animation_editor->edit(Object::cast_to<ProceduralAnimation>(p_object));
|
|
animation_editor_button->show();
|
|
} else {
|
|
animation_editor_button->set_pressed(false);
|
|
animation_editor_button->hide();
|
|
}
|
|
}
|
|
|
|
bool ProceduralAnimationEditorPlugin::handles(Object *p_object) const {
|
|
if (p_object->is_class("ProceduralAnimation")) {
|
|
animation_editor_button->show();
|
|
animation_editor_button->set_pressed(true);
|
|
return true;
|
|
}
|
|
|
|
if (p_object->is_class("AnimationPlayer")) {
|
|
AnimationPlayer *player = Object::cast_to<AnimationPlayer>(p_object);
|
|
|
|
String an = player->get_assigned_animation();
|
|
|
|
if (an != "") {
|
|
Ref<ProceduralAnimation> anim = player->get_animation(an);
|
|
|
|
if (anim.is_valid()) {
|
|
animation_editor_button->show();
|
|
//animation_editor_button->set_pressed(true);
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
|
|
animation_editor_button->set_pressed(false);
|
|
animation_editor_button->hide();
|
|
|
|
return false;
|
|
}
|
|
|
|
void ProceduralAnimationEditorPlugin::make_visible(bool p_visible) {
|
|
}
|
|
|
|
ProceduralAnimationEditorPlugin::ProceduralAnimationEditorPlugin(EditorNode *p_node) {
|
|
|
|
editor = p_node;
|
|
|
|
animation_editor = memnew(ProceduralAnimationEditor(editor));
|
|
|
|
animation_editor_button = add_control_to_bottom_panel(animation_editor, "Procedural Animations");
|
|
|
|
animation_editor->hide();
|
|
}
|
|
|
|
ProceduralAnimationEditorPlugin::~ProceduralAnimationEditorPlugin() {
|
|
}
|
|
|
|
// E -------- ProceduralAnimationEditorPlugin --------
|