| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007 | /**************************************************************************//*  animation_track_editor.h                                              *//**************************************************************************//*                         This file is part of:                          *//*                             GODOT ENGINE                               *//*                        https://godotengine.org                         *//**************************************************************************//* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). *//* Copyright (c) 2007-2014 Juan Linietsky, Ariel Manzur.                  *//*                                                                        *//* 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.                 *//**************************************************************************/#ifndef ANIMATION_TRACK_EDITOR_H#define ANIMATION_TRACK_EDITOR_H#include "editor/editor_data.h"#include "editor/editor_properties.h"#include "editor/property_selector.h"#include "scene/3d/node_3d.h"#include "scene/gui/control.h"#include "scene/gui/menu_button.h"#include "scene/gui/scroll_bar.h"#include "scene/gui/tree.h"#include "scene/resources/animation.h"class AnimationMarkerEdit;class AnimationTrackEditor;class AnimationTrackEdit;class CheckBox;class ColorPickerButton;class EditorSpinSlider;class HSlider;class OptionButton;class PanelContainer;class SceneTreeDialog;class SpinBox;class TextureRect;class ViewPanner;class EditorValidationPanel;class AnimationTrackKeyEdit : public Object {	GDCLASS(AnimationTrackKeyEdit, Object);public:	bool setting = false;	bool animation_read_only = false;	Ref<Animation> animation;	int track = -1;	float key_ofs = 0;	Node *root_path = nullptr;	PropertyInfo hint;	NodePath base;	bool use_fps = false;	AnimationTrackEditor *editor = nullptr;	bool _hide_script_from_inspector() { return true; }	bool _hide_metadata_from_inspector() { return true; }	bool _dont_undo_redo() { return true; }	bool _is_read_only() { return animation_read_only; }	void notify_change();	Node *get_root_path();	void set_use_fps(bool p_enable);protected:	static void _bind_methods();	void _fix_node_path(Variant &value);	void _update_obj(const Ref<Animation> &p_anim);	void _key_ofs_changed(const Ref<Animation> &p_anim, float from, float to);	bool _set(const StringName &p_name, const Variant &p_value);	bool _get(const StringName &p_name, Variant &r_ret) const;	void _get_property_list(List<PropertyInfo> *p_list) const;};class AnimationMultiTrackKeyEdit : public Object {	GDCLASS(AnimationMultiTrackKeyEdit, Object);public:	bool setting = false;	bool animation_read_only = false;	Ref<Animation> animation;	RBMap<int, List<float>> key_ofs_map;	RBMap<int, NodePath> base_map;	PropertyInfo hint;	Node *root_path = nullptr;	bool use_fps = false;	AnimationTrackEditor *editor = nullptr;	bool _hide_script_from_inspector() { return true; }	bool _hide_metadata_from_inspector() { return true; }	bool _dont_undo_redo() { return true; }	bool _is_read_only() { return animation_read_only; }	void notify_change();	Node *get_root_path();	void set_use_fps(bool p_enable);protected:	static void _bind_methods();	void _fix_node_path(Variant &value, NodePath &base);	void _update_obj(const Ref<Animation> &p_anim);	void _key_ofs_changed(const Ref<Animation> &p_anim, float from, float to);	bool _set(const StringName &p_name, const Variant &p_value);	bool _get(const StringName &p_name, Variant &r_ret) const;	void _get_property_list(List<PropertyInfo> *p_list) const;};class AnimationMarkerKeyEdit : public Object {	GDCLASS(AnimationMarkerKeyEdit, Object);public:	bool animation_read_only = false;	Ref<Animation> animation;	StringName marker_name;	bool use_fps = false;	AnimationMarkerEdit *marker_edit = nullptr;	bool _hide_script_from_inspector() { return true; }	bool _hide_metadata_from_inspector() { return true; }	bool _dont_undo_redo() { return true; }	bool _is_read_only() { return animation_read_only; }	float get_time() const;protected:	static void _bind_methods();	void _set_marker_name(const StringName &p_name);	bool _set(const StringName &p_name, const Variant &p_value);	bool _get(const StringName &p_name, Variant &r_ret) const;	void _get_property_list(List<PropertyInfo> *p_list) const;};class AnimationMultiMarkerKeyEdit : public Object {	GDCLASS(AnimationMultiMarkerKeyEdit, Object);public:	bool animation_read_only = false;	Ref<Animation> animation;	Vector<StringName> marker_names;	AnimationMarkerEdit *marker_edit = nullptr;	bool _hide_script_from_inspector() { return true; }	bool _hide_metadata_from_inspector() { return true; }	bool _dont_undo_redo() { return true; }	bool _is_read_only() { return animation_read_only; }protected:	static void _bind_methods();	bool _set(const StringName &p_name, const Variant &p_value);	bool _get(const StringName &p_name, Variant &r_ret) const;	void _get_property_list(List<PropertyInfo> *p_list) const;};class AnimationTimelineEdit : public Range {	GDCLASS(AnimationTimelineEdit, Range);	friend class AnimationBezierTrackEdit;	friend class AnimationTrackEditor;	static constexpr float SCROLL_ZOOM_FACTOR_IN = 1.02f; // Zoom factor per mouse scroll in the animation editor when zooming in. The closer to 1.0, the finer the control.	static constexpr float SCROLL_ZOOM_FACTOR_OUT = 0.98f; // Zoom factor when zooming out. Similar to SCROLL_ZOOM_FACTOR_IN but less than 1.0.	Ref<Animation> animation;	bool read_only = false;	AnimationTrackEdit *track_edit = nullptr;	int name_limit = 0;	Range *zoom = nullptr;	Range *h_scroll = nullptr;	float play_position_pos = 0.0f;	HBoxContainer *len_hb = nullptr;	EditorSpinSlider *length = nullptr;	Button *loop = nullptr;	TextureRect *time_icon = nullptr;	MenuButton *add_track = nullptr;	Control *play_position = nullptr; //separate control used to draw so updates for only position changed are much faster	HScrollBar *hscroll = nullptr;	void _zoom_changed(double);	void _anim_length_changed(double p_new_len);	void _anim_loop_pressed();	void _play_position_draw();	Rect2 hsize_rect;	bool editing = false;	bool use_fps = false;	Ref<ViewPanner> panner;	void _pan_callback(Vector2 p_scroll_vec, Ref<InputEvent> p_event);	void _zoom_callback(float p_zoom_factor, Vector2 p_origin, Ref<InputEvent> p_event);	bool dragging_timeline = false;	bool dragging_hsize = false;	float dragging_hsize_from = 0.0f;	float dragging_hsize_at = 0.0f;	double last_zoom_scale = 1.0;	double hscroll_on_zoom_buffer = -1.0;	Vector2 zoom_scroll_origin;	bool zoom_callback_occurred = false;	virtual void gui_input(const Ref<InputEvent> &p_event) override;	void _track_added(int p_track);	float _get_zoom_scale(double p_zoom_value) const;	void _scroll_to_start();protected:	static void _bind_methods();	void _notification(int p_what);public:	int get_name_limit() const;	int get_buttons_width() const;	float get_zoom_scale() const;	virtual Size2 get_minimum_size() const override;	void set_animation(const Ref<Animation> &p_animation, bool p_read_only);	void set_track_edit(AnimationTrackEdit *p_track_edit);	void set_zoom(Range *p_zoom);	Range *get_zoom() const { return zoom; }	void auto_fit();	void set_play_position(float p_pos);	float get_play_position() const;	void update_play_position();	void update_values();	void set_use_fps(bool p_use_fps);	bool is_using_fps() const;	void set_hscroll(HScrollBar *p_hscroll);	virtual CursorShape get_cursor_shape(const Point2 &p_pos) const override;	AnimationTimelineEdit();};class AnimationMarkerEdit : public Control {	GDCLASS(AnimationMarkerEdit, Control);	friend class AnimationTimelineEdit;	enum {		MENU_KEY_INSERT,		MENU_KEY_RENAME,		MENU_KEY_DELETE,		MENU_KEY_TOGGLE_MARKER_NAMES,	};	AnimationTimelineEdit *timeline = nullptr;	Control *play_position = nullptr; // Separate control used to draw so updates for only position changed are much faster.	float play_position_pos = 0.0f;	HashSet<StringName> selection;	Ref<Animation> animation;	bool read_only = false;	Ref<Texture2D> type_icon;	Ref<Texture2D> selected_icon;	PopupMenu *menu = nullptr;	bool hovered = false;	StringName hovering_marker;	void _zoom_changed();	Ref<Texture2D> icon_cache;	void _menu_selected(int p_index);	void _play_position_draw();	bool _try_select_at_ui_pos(const Point2 &p_pos, bool p_aggregate, bool p_deselectable);	bool _is_ui_pos_in_current_section(const Point2 &p_pos);	float insert_at_pos = 0.0f;	bool moving_selection_attempt = false;	bool moving_selection_effective = false;	float moving_selection_offset = 0.0f;	float moving_selection_pivot = 0.0f;	float moving_selection_mouse_begin_x = 0.0f;	float moving_selection_mouse_begin_y = 0.0f;	StringName select_single_attempt;	bool moving_selection = false;	void _move_selection_begin();	void _move_selection(float p_offset);	void _move_selection_commit();	void _move_selection_cancel();	void _clear_selection_for_anim(const Ref<Animation> &p_anim);	void _select_key(const StringName &p_name, bool is_single = false);	void _deselect_key(const StringName &p_name);	void _insert_marker(float p_ofs);	void _rename_marker(const StringName &p_name);	void _delete_selected_markers();	ConfirmationDialog *marker_insert_confirm = nullptr;	LineEdit *marker_insert_new_name = nullptr;	ColorPickerButton *marker_insert_color = nullptr;	AcceptDialog *marker_insert_error_dialog = nullptr;	float marker_insert_ofs = 0;	ConfirmationDialog *marker_rename_confirm = nullptr;	LineEdit *marker_rename_new_name = nullptr;	StringName marker_rename_prev_name;	AcceptDialog *marker_rename_error_dialog = nullptr;	bool should_show_all_marker_names = false;	////////////// edit menu stuff	void _marker_insert_confirmed();	void _marker_insert_new_name_changed(const String &p_text);	void _marker_rename_confirmed();	void _marker_rename_new_name_changed(const String &p_text);	AnimationTrackEditor *editor = nullptr;	HBoxContainer *_create_hbox_labeled_control(const String &p_text, Control *p_control) const;	void _update_key_edit();	void _clear_key_edit();	AnimationMarkerKeyEdit *key_edit = nullptr;	AnimationMultiMarkerKeyEdit *multi_key_edit = nullptr;protected:	static void _bind_methods();	void _notification(int p_what);	virtual void gui_input(const Ref<InputEvent> &p_event) override;public:	virtual String get_tooltip(const Point2 &p_pos) const override;	virtual int get_key_height() const;	virtual Rect2 get_key_rect(float p_pixels_sec) const;	virtual bool is_key_selectable_by_distance() const;	virtual void draw_key(const StringName &p_name, float p_pixels_sec, int p_x, bool p_selected, int p_clip_left, int p_clip_right);	virtual void draw_bg(int p_clip_left, int p_clip_right);	virtual void draw_fg(int p_clip_left, int p_clip_right);	Ref<Animation> get_animation() const;	AnimationTimelineEdit *get_timeline() const { return timeline; }	AnimationTrackEditor *get_editor() const { return editor; }	bool is_selection_active() const { return !selection.is_empty(); }	bool is_moving_selection() const { return moving_selection; }	float get_moving_selection_offset() const { return moving_selection_offset; }	void set_animation(const Ref<Animation> &p_animation, bool p_read_only);	virtual Size2 get_minimum_size() const override;	void set_timeline(AnimationTimelineEdit *p_timeline);	void set_editor(AnimationTrackEditor *p_editor);	void set_play_position(float p_pos);	void update_play_position();	void set_use_fps(bool p_use_fps);	PackedStringArray get_selected_section() const;	bool is_marker_selected(const StringName &p_marker) const;	// For use by AnimationTrackEditor.	void _clear_selection(bool p_update);	AnimationMarkerEdit();	~AnimationMarkerEdit();};class AnimationTrackEdit : public Control {	GDCLASS(AnimationTrackEdit, Control);	friend class AnimationTimelineEdit;	enum {		MENU_CALL_MODE_CONTINUOUS,		MENU_CALL_MODE_DISCRETE,		MENU_CALL_MODE_CAPTURE,		MENU_INTERPOLATION_NEAREST,		MENU_INTERPOLATION_LINEAR,		MENU_INTERPOLATION_CUBIC,		MENU_INTERPOLATION_LINEAR_ANGLE,		MENU_INTERPOLATION_CUBIC_ANGLE,		MENU_LOOP_WRAP,		MENU_LOOP_CLAMP,		MENU_KEY_INSERT,		MENU_KEY_DUPLICATE,		MENU_KEY_CUT,		MENU_KEY_COPY,		MENU_KEY_PASTE,		MENU_KEY_ADD_RESET,		MENU_KEY_DELETE,		MENU_USE_BLEND_ENABLED,		MENU_USE_BLEND_DISABLED,	};	AnimationTimelineEdit *timeline = nullptr;	Popup *path_popup = nullptr;	LineEdit *path = nullptr;	Node *root = nullptr;	Control *play_position = nullptr; //separate control used to draw so updates for only position changed are much faster	float play_position_pos = 0.0f;	NodePath node_path;	Ref<Animation> animation;	bool read_only = false;	int track = 0;	Rect2 check_rect;	Rect2 path_rect;	Rect2 update_mode_rect;	Rect2 interp_mode_rect;	Rect2 loop_wrap_rect;	Rect2 remove_rect;	Ref<Texture2D> type_icon;	Ref<Texture2D> selected_icon;	PopupMenu *menu = nullptr;	bool hovered = false;	bool clicking_on_name = false;	int hovering_key_idx = -1;	void _zoom_changed();	Ref<Texture2D> icon_cache;	String path_cache;	void _menu_selected(int p_index);	void _path_submitted(const String &p_text);	void _play_position_draw();	bool _is_value_key_valid(const Variant &p_key_value, Variant::Type &r_valid_type) const;	bool _try_select_at_ui_pos(const Point2 &p_pos, bool p_aggregate, bool p_deselectable);	Ref<Texture2D> _get_key_type_icon() const;	mutable int dropping_at = 0;	float insert_at_pos = 0.0f;	bool moving_selection_attempt = false;	bool moving_selection_effective = false;	float moving_selection_pivot = 0.0f;	float moving_selection_mouse_begin_x = 0.0f;	int select_single_attempt = -1;	bool moving_selection = false;	bool in_group = false;	AnimationTrackEditor *editor = nullptr;protected:	static void _bind_methods();	void _notification(int p_what);	virtual void gui_input(const Ref<InputEvent> &p_event) override;public:	virtual Variant get_drag_data(const Point2 &p_point) override;	virtual bool can_drop_data(const Point2 &p_point, const Variant &p_data) const override;	virtual void drop_data(const Point2 &p_point, const Variant &p_data) override;	virtual String get_tooltip(const Point2 &p_pos) const override;	virtual int get_key_height() const;	virtual Rect2 get_key_rect(int p_index, float p_pixels_sec);	virtual bool is_key_selectable_by_distance() const;	virtual void draw_key_link(int p_index, float p_pixels_sec, int p_x, int p_next_x, int p_clip_left, int p_clip_right);	virtual void draw_key(int p_index, float p_pixels_sec, int p_x, bool p_selected, int p_clip_left, int p_clip_right);	virtual void draw_bg(int p_clip_left, int p_clip_right);	virtual void draw_fg(int p_clip_left, int p_clip_right);	//helper	void draw_texture_region_clipped(const Ref<Texture2D> &p_texture, const Rect2 &p_rect, const Rect2 &p_region);	void draw_rect_clipped(const Rect2 &p_rect, const Color &p_color, bool p_filled = true);	int get_track() const;	Ref<Animation> get_animation() const;	AnimationTimelineEdit *get_timeline() const { return timeline; }	AnimationTrackEditor *get_editor() const { return editor; }	NodePath get_path() const;	void set_animation_and_track(const Ref<Animation> &p_animation, int p_track, bool p_read_only);	virtual Size2 get_minimum_size() const override;	void set_timeline(AnimationTimelineEdit *p_timeline);	void set_editor(AnimationTrackEditor *p_editor);	void set_root(Node *p_root);	void set_play_position(float p_pos);	void update_play_position();	void cancel_drop();	void set_in_group(bool p_enable);	void append_to_selection(const Rect2 &p_box, bool p_deselection);	AnimationTrackEdit();};class AnimationTrackEditPlugin : public RefCounted {	GDCLASS(AnimationTrackEditPlugin, RefCounted);public:	virtual AnimationTrackEdit *create_value_track_edit(Object *p_object, Variant::Type p_type, const String &p_property, PropertyHint p_hint, const String &p_hint_string, int p_usage);	virtual AnimationTrackEdit *create_audio_track_edit();	virtual AnimationTrackEdit *create_animation_track_edit(Object *p_object);};class AnimationTrackKeyEdit;class AnimationMultiTrackKeyEdit;class AnimationBezierTrackEdit;class AnimationTrackEditGroup : public Control {	GDCLASS(AnimationTrackEditGroup, Control);	Ref<Texture2D> icon;	Vector2 icon_size;	String node_name;	NodePath node;	Node *root = nullptr;	AnimationTimelineEdit *timeline = nullptr;	AnimationTrackEditor *editor = nullptr;	void _zoom_changed();protected:	void _notification(int p_what);	virtual void gui_input(const Ref<InputEvent> &p_event) override;public:	void set_type_and_name(const Ref<Texture2D> &p_type, const String &p_name, const NodePath &p_node);	virtual Size2 get_minimum_size() const override;	void set_timeline(AnimationTimelineEdit *p_timeline);	void set_root(Node *p_root);	void set_editor(AnimationTrackEditor *p_editor);	AnimationTrackEditGroup();};class AnimationTrackEditor : public VBoxContainer {	GDCLASS(AnimationTrackEditor, VBoxContainer);	friend class AnimationTimelineEdit;	friend class AnimationBezierTrackEdit;	friend class AnimationMarkerKeyEditEditor;	Ref<Animation> animation;	bool read_only = false;	Node *root = nullptr;	MenuButton *edit = nullptr;	PanelContainer *main_panel = nullptr;	HScrollBar *hscroll = nullptr;	ScrollContainer *scroll = nullptr;	VBoxContainer *track_vbox = nullptr;	AnimationBezierTrackEdit *bezier_edit = nullptr;	VBoxContainer *timeline_vbox = nullptr;	Label *info_message = nullptr;	AnimationTimelineEdit *timeline = nullptr;	AnimationMarkerEdit *marker_edit = nullptr;	HSlider *zoom = nullptr;	EditorSpinSlider *step = nullptr;	Button *fps_compat = nullptr;	Label *nearest_fps_label = nullptr;	TextureRect *zoom_icon = nullptr;	Button *snap_keys = nullptr;	Button *snap_timeline = nullptr;	Button *bezier_edit_icon = nullptr;	OptionButton *snap_mode = nullptr;	Button *auto_fit = nullptr;	Button *auto_fit_bezier = nullptr;	Button *imported_anim_warning = nullptr;	void _show_imported_anim_warning();	Button *dummy_player_warning = nullptr;	void _show_dummy_player_warning();	Button *inactive_player_warning = nullptr;	void _show_inactive_player_warning();	void _snap_mode_changed(int p_mode);	Vector<AnimationTrackEdit *> track_edits;	Vector<AnimationTrackEditGroup *> groups;	bool animation_changing_awaiting_update = false;	void _animation_update(); // Updated by AnimationTrackEditor(this)	int _get_track_selected();	void _animation_changed();	void _update_tracks();	void _redraw_tracks();	void _redraw_groups();	void _check_bezier_exist();	void _name_limit_changed();	void _timeline_changed(float p_new_pos, bool p_timeline_only);	void _track_remove_request(int p_track);	void _animation_track_remove_request(int p_track, Ref<Animation> p_from_animation);	void _track_grab_focus(int p_track);	void _update_scroll(double);	void _update_nearest_fps_label();	void _update_fps_compat_mode(bool p_enabled);	void _update_step(double p_new_step);	void _update_length(double p_new_len);	void _dropped_track(int p_from_track, int p_to_track);	void _add_track(int p_type);	void _new_track_node_selected(NodePath p_path);	void _new_track_property_selected(const String &p_name);	void _update_step_spinbox();	PropertySelector *prop_selector = nullptr;	PropertySelector *method_selector = nullptr;	SceneTreeDialog *pick_track = nullptr;	int adding_track_type = 0;	NodePath adding_track_path;	bool keying = false;	struct InsertData {		Animation::TrackType type;		NodePath path;		int track_idx = 0;		float time = FLT_MAX; // Defaults to current timeline position.		Variant value;		String query;		bool advance = false;	};	Label *insert_confirm_text = nullptr;	CheckBox *insert_confirm_bezier = nullptr;	CheckBox *insert_confirm_reset = nullptr;	ConfirmationDialog *insert_confirm = nullptr;	bool insert_queue = false;	List<InsertData> insert_data;	void _query_insert(const InsertData &p_id);	Ref<Animation> _create_and_get_reset_animation();	void _confirm_insert_list();	struct TrackIndices {		int normal;		int reset;		TrackIndices(const Animation *p_anim = nullptr, const Animation *p_reset_anim = nullptr) {			normal = p_anim ? p_anim->get_track_count() : 0;			reset = p_reset_anim ? p_reset_anim->get_track_count() : 0;		}	};	TrackIndices _confirm_insert(InsertData p_id, TrackIndices p_next_tracks, bool p_reset_wanted, Ref<Animation> p_reset_anim, bool p_create_beziers);	void _insert_track(bool p_reset_wanted, bool p_create_beziers);	void _root_removed();	PropertyInfo _find_hint_for_track(int p_idx, NodePath &r_base_path, Variant *r_current_val = nullptr);	void _scroll_changed(const Vector2 &p_val);	void _v_scroll_changed(float p_val);	void _h_scroll_changed(float p_val);	Ref<ViewPanner> panner;	void _pan_callback(Vector2 p_scroll_vec, Ref<InputEvent> p_event);	void _zoom_callback(float p_zoom_factor, Vector2 p_origin, Ref<InputEvent> p_event);	void _timeline_value_changed(double);	float insert_key_from_track_call_ofs = 0.0f;	int insert_key_from_track_call_track = 0;	void _insert_key_from_track(float p_ofs, int p_track);	void _add_method_key(const String &p_method);	void _fetch_value_track_options(const NodePath &p_path, Animation::UpdateMode *r_update_mode, Animation::InterpolationType *r_interpolation_type, bool *r_loop_wrap);	void _clear_selection_for_anim(const Ref<Animation> &p_anim);	void _select_at_anim(const Ref<Animation> &p_anim, int p_track, float p_pos);	//selection	struct SelectedKey {		int track = 0;		int key = 0;		bool operator<(const SelectedKey &p_key) const { return track == p_key.track ? key < p_key.key : track < p_key.track; }	};	struct KeyInfo {		float pos = 0;	};	RBMap<SelectedKey, KeyInfo> selection;	bool moving_selection = false;	float moving_selection_offset = 0.0f;	void _move_selection_begin();	void _move_selection(float p_offset);	void _move_selection_commit();	void _move_selection_cancel();	AnimationTrackKeyEdit *key_edit = nullptr;	AnimationMultiTrackKeyEdit *multi_key_edit = nullptr;	void _update_key_edit();	void _clear_key_edit();	Control *box_selection_container = nullptr;	Control *box_selection = nullptr;	void _box_selection_draw();	bool box_selecting = false;	Vector2 box_selecting_from;	Vector2 box_selecting_to;	Rect2 box_select_rect;	Vector2 prev_scroll_position;	void _scroll_input(const Ref<InputEvent> &p_event);	Vector<Ref<AnimationTrackEditPlugin>> track_edit_plugins;	void _toggle_bezier_edit();	void _cancel_bezier_edit();	void _bezier_edit(int p_for_track);	void _bezier_track_set_key_handle_mode(Animation *p_anim, int p_track, int p_index, Animation::HandleMode p_mode, Animation::HandleSetMode p_set_mode = Animation::HANDLE_SET_MODE_NONE);	////////////// edit menu stuff	ConfirmationDialog *bake_dialog = nullptr;	CheckBox *bake_trs = nullptr;	CheckBox *bake_blendshape = nullptr;	CheckBox *bake_value = nullptr;	SpinBox *bake_fps = nullptr;	ConfirmationDialog *optimize_dialog = nullptr;	SpinBox *optimize_velocity_error = nullptr;	SpinBox *optimize_angular_error = nullptr;	SpinBox *optimize_precision_error = nullptr;	ConfirmationDialog *cleanup_dialog = nullptr;	CheckBox *cleanup_keys_with_trimming_head = nullptr;	CheckBox *cleanup_keys_with_trimming_end = nullptr;	CheckBox *cleanup_keys = nullptr;	CheckBox *cleanup_tracks = nullptr;	CheckBox *cleanup_all = nullptr;	ConfirmationDialog *scale_dialog = nullptr;	SpinBox *scale = nullptr;	ConfirmationDialog *ease_dialog = nullptr;	OptionButton *transition_selection = nullptr;	OptionButton *ease_selection = nullptr;	SpinBox *ease_fps = nullptr;	void _select_all_tracks_for_copy();	void _edit_menu_about_to_popup();	void _edit_menu_pressed(int p_option);	int last_menu_track_opt = 0;	void _cleanup_animation(Ref<Animation> p_animation);	void _anim_duplicate_keys(float p_ofs, bool p_ofs_valid, int p_track);	void _anim_copy_keys(bool p_cut);	bool _is_track_compatible(int p_target_track_idx, Variant::Type p_source_value_type, Animation::TrackType p_source_track_type);	void _anim_paste_keys(float p_ofs, bool p_ofs_valid, int p_track);	void _view_group_toggle();	Button *view_group = nullptr;	Button *selected_filter = nullptr;	void _auto_fit();	void _auto_fit_bezier();	void _selection_changed();	ConfirmationDialog *track_copy_dialog = nullptr;	Tree *track_copy_select = nullptr;	struct TrackClipboard {		NodePath full_path;		NodePath base_path;		Animation::TrackType track_type = Animation::TYPE_ANIMATION;		Animation::InterpolationType interp_type = Animation::INTERPOLATION_CUBIC_ANGLE;		Animation::UpdateMode update_mode = Animation::UPDATE_CAPTURE;		Animation::LoopMode loop_mode = Animation::LOOP_PINGPONG;		bool loop_wrap = false;		bool enabled = false;		bool use_blend = false;		struct Key {			float time = 0;			float transition = 0;			Variant value;		};		Vector<Key> keys;	};	struct KeyClipboard {		int top_track;		struct Key {			Animation::TrackType track_type;			int track;			float time = 0;			float transition = 0;			Variant value;		};		Vector<Key> keys;	};	Vector<TrackClipboard> track_clipboard;	KeyClipboard key_clipboard;	void _set_key_clipboard(int p_top_track, float p_top_time, RBMap<SelectedKey, KeyInfo> &p_keymap);	void _insert_animation_key(NodePath p_path, const Variant &p_value);	void _pick_track_filter_text_changed(const String &p_newtext);	void _pick_track_select_recursive(TreeItem *p_item, const String &p_filter, Vector<Node *> &p_select_candidates);	double snap_unit;	bool fps_compatible = true;	int nearest_fps = 0;	void _update_snap_unit();protected:	static void _bind_methods();	void _notification(int p_what);public:	// Public for use with callable_mp.	void _clear_selection(bool p_update = false);	void _key_selected(int p_key, bool p_single, int p_track);	void _key_deselected(int p_key, int p_track);	enum {		EDIT_COPY_TRACKS,		EDIT_COPY_TRACKS_CONFIRM,		EDIT_PASTE_TRACKS,		EDIT_CUT_KEYS,		EDIT_COPY_KEYS,		EDIT_PASTE_KEYS,		EDIT_SCALE_SELECTION,		EDIT_SCALE_FROM_CURSOR,		EDIT_SCALE_CONFIRM,		EDIT_SET_START_OFFSET,		EDIT_SET_END_OFFSET,		EDIT_EASE_SELECTION,		EDIT_EASE_CONFIRM,		EDIT_DUPLICATE_SELECTED_KEYS,		EDIT_DUPLICATE_SELECTION,		EDIT_DUPLICATE_TRANSPOSED,		EDIT_MOVE_FIRST_SELECTED_KEY_TO_CURSOR,		EDIT_MOVE_LAST_SELECTED_KEY_TO_CURSOR,		EDIT_ADD_RESET_KEY,		EDIT_DELETE_SELECTION,		EDIT_GOTO_NEXT_STEP,		EDIT_GOTO_NEXT_STEP_TIMELINE_ONLY, // Next step without updating animation.		EDIT_GOTO_PREV_STEP,		EDIT_APPLY_RESET,		EDIT_BAKE_ANIMATION,		EDIT_BAKE_ANIMATION_CONFIRM,		EDIT_OPTIMIZE_ANIMATION,		EDIT_OPTIMIZE_ANIMATION_CONFIRM,		EDIT_CLEAN_UP_ANIMATION,		EDIT_CLEAN_UP_ANIMATION_CONFIRM	};	void add_track_edit_plugin(const Ref<AnimationTrackEditPlugin> &p_plugin);	void remove_track_edit_plugin(const Ref<AnimationTrackEditPlugin> &p_plugin);	void set_animation(const Ref<Animation> &p_anim, bool p_read_only);	Ref<Animation> get_current_animation() const;	void set_root(Node *p_root);	Node *get_root() const;	void update_keying();	bool has_keying() const;	Dictionary get_state() const;	void set_state(const Dictionary &p_state);	void cleanup();	void set_anim_pos(float p_pos);	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, bool p_advance);	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);	void make_insert_queue();	void commit_insert_queue();	void show_select_node_warning(bool p_show);	void show_dummy_player_warning(bool p_show);	void show_inactive_player_warning(bool p_show);	bool is_key_selected(int p_track, int p_key) const;	bool is_selection_active() const;	bool is_key_clipboard_active() const;	bool is_moving_selection() const;	bool is_snap_timeline_enabled() const;	bool is_snap_keys_enabled() const;	bool is_bezier_editor_active() const;	bool can_add_reset_key() const;	float get_moving_selection_offset() const;	float snap_time(float p_value, bool p_relative = false);	float get_snap_unit();	bool is_grouping_tracks();	PackedStringArray get_selected_section() const;	bool is_marker_selected(const StringName &p_marker) const;	bool is_marker_moving_selection() const;	float get_marker_moving_selection_offset() const;	/** If `p_from_mouse_event` is `true`, handle Shift key presses for precise snapping. */	void goto_prev_step(bool p_from_mouse_event);	/** If `p_from_mouse_event` is `true`, handle Shift key presses for precise snapping. */	void goto_next_step(bool p_from_mouse_event, bool p_timeline_only = false);	MenuButton *get_edit_menu();	AnimationTrackEditor();	~AnimationTrackEditor();};// AnimationTrackKeyEditEditorPluginclass AnimationTrackKeyEditEditor : public EditorProperty {	GDCLASS(AnimationTrackKeyEditEditor, EditorProperty);	Ref<Animation> animation;	int track = -1;	real_t key_ofs = 0.0;	bool use_fps = false;	EditorSpinSlider *spinner = nullptr;	struct KeyDataCache {		real_t time = 0.0;		float transition = 0.0;		Variant value;	} key_data_cache;	void _time_edit_spun();	void _time_edit_entered();	void _time_edit_exited();public:	AnimationTrackKeyEditEditor(Ref<Animation> p_animation, int p_track, real_t p_key_ofs, bool p_use_fps);	~AnimationTrackKeyEditEditor();};// AnimationMarkerKeyEditEditorPluginclass AnimationMarkerKeyEditEditor : public EditorProperty {	GDCLASS(AnimationMarkerKeyEditEditor, EditorProperty);	Ref<Animation> animation;	StringName marker_name;	bool use_fps = false;	EditorSpinSlider *spinner = nullptr;	void _time_edit_exited();public:	AnimationMarkerKeyEditEditor(Ref<Animation> p_animation, const StringName &p_name, bool p_use_fps);	~AnimationMarkerKeyEditEditor();};#endif // ANIMATION_TRACK_EDITOR_H
 |