# (c) 2023-present Eroax # (c) 2023-present Yagich # GNU General Public License v3.0+ (see COPYING or https://www.gnu.org/licenses/gpl-3.0.txt) extends VBoxContainer class_name VariableViewer const REMOVE_ICON = preload("res://graph_node_renderer/textures/remove-icon.svg") @onready var variable_tree: Tree = %VariableTree @onready var types_popup: PopupMenu = %TypesPopup var root: TreeItem var _old_name: String signal top_field_edited(old_name: String, new_name: String, new_value: Variant) signal top_field_removed(field_name: String) func _ready() -> void: variable_tree.set_column_title(0, "Name") variable_tree.set_column_title(1, "Value") variable_tree.set_column_title(2, "Type") variable_tree.set_column_expand(1, true) variable_tree.set_column_expand_ratio(0, 15) variable_tree.set_column_expand_ratio(1, 70) variable_tree.set_column_expand_ratio(2, 15) root = variable_tree.create_item() rebuild_variable_tree() variable_tree.item_activated.connect( func(): var column := variable_tree.get_selected_column() var item := variable_tree.get_selected() # do nothing if this is an array and user is trying to edit index # (TODO: proper reordering) if column == 0 && item.get_meta(&"indexed"): return if column < 2: _old_name = item.get_text(0) variable_tree.edit_selected(true) else: var pos := get_global_mouse_position() var r := Rect2i(Vector2i(pos), Vector2i(0, 100)) for i in types_popup.get_item_count(): types_popup.set_item_checked(i, false) types_popup.set_item_checked(item.get_metadata(2), true) types_popup.popup(r) ) variable_tree.item_edited.connect( func(): var item := variable_tree.get_selected() var new_name := item.get_text(0) var new_value var type: DeckType.Types = item.get_metadata(2) match type: DeckType.Types.STRING: new_value = item.get_text(1) DeckType.Types.BOOL: new_value = item.is_checked(1) DeckType.Types.NUMERIC: new_value = item.get_range(1) _: new_value = item.get_meta(&"value") item.set_meta(&"value", new_value) if item.get_meta(&"container", -1) is int: top_field_edited.emit(_old_name, new_name, new_value) else: var container = item.get_meta(&"container") # array if item.get_meta(&"indexed", false): var index := item.get_index() (container as Array)[index] = new_value # dictionary else: var key := new_name (container as Dictionary).erase(_old_name) (container as Dictionary)[key] = new_value ) for i in DeckType.Types.size(): types_popup.add_radio_check_item(DeckType.type_str(i)) func rebuild_variable_tree(data: Dictionary = {}) -> void: #variable_tree.clear() # godot will raw dog a nullptr later if we clear the whole tree for i in root.get_children(): i.free() for i in data: add_item(i, data[i]) func add_item(item_name: String, item_value: Variant, parent: TreeItem = root, container: Variant = -1, indexed: bool = false) -> TreeItem: # the container parameter is -1 instead of null because Object#get_metadata() logs an error # if the default parameter is null var item := variable_tree.create_item(parent) item.set_text(0, item_name) var type: DeckType.Types = DeckType.INVERSE_GODOT_TYPES_MAP[typeof(item_value)] match type: DeckType.Types.NUMERIC: item.set_cell_mode(1, TreeItem.CELL_MODE_RANGE) item.set_range(1, item_value) item.set_range_config(1, -9999, 9999, 0.0001) DeckType.Types.BOOL: item.set_cell_mode(1, TreeItem.CELL_MODE_CHECK) item.set_checked(1, item_value) _: item.set_text(1, str(item_value)) item.set_meta(&"container", container) item.set_meta(&"indexed", indexed) item.set_meta(&"value", item_value) item.set_text(2, DeckType.type_str(type)) item.set_metadata(2, type) if item_value is Dictionary: for i in item_value: add_item(i, item_value[i], item, item_value) if item_value is Array: for i in (item_value as Array).size(): add_item(str(i), item_value[i], item, item_value, true) item.collapsed = true item.add_button(2, REMOVE_ICON) return item func _on_variable_tree_button_clicked(item: TreeItem, column: int, id: int, mouse_button_index: int) -> void: if mouse_button_index != MOUSE_BUTTON_LEFT: return if item.get_meta(&"container", -1) is int: var key := item.get_text(0) top_field_removed.emit(key) item.free() else: var container = item.get_meta(&"container") # array if item.get_meta(&"indexed", false): var index := item.get_index() var parent := item.get_parent() (container as Array).remove_at(index) item.free() # go through the array and reset the index strings for i in (container as Array).size(): parent.get_child(i).set_text(0, str(i)) # dictionary else: (container as Dictionary).erase(item.get_text(0)) item.free()