stick-the-quick/util/property_save_restore_stack.gd

47 lines
1.7 KiB
GDScript

class_name PropertySaveRestoreStack extends Node
## Allows saving and restoring sets of properties on a target Node.
## Stores a batch of saved properties to be restored all at once on pop.
class Frame:
## The saved properties to be restored when this frame is popped.
var saved_properties: Dictionary[NodePath, Variant]
## Saves properties to this frame and then overwrites them on the target.
func save_and_modify(target: Node, properties: Dictionary) -> void:
for key in properties:
var property_path := key as NodePath
saved_properties[property_path] = target.get_indexed(property_path)
target.set_indexed(property_path, properties[property_path])
## Restores saved properties. Called when the frame is popped.
func restore(target: Node) -> void:
for property_path in saved_properties:
target.set_indexed(property_path, saved_properties[property_path])
## All property paths are understood as relative to this Node.
## [br][br]
## Altering this property at runtime will immediately empty the stack,
## restoring all saved properties to the previous target in the process.
@export var target: Node:
set(value):
if value != target:
while !is_empty():
pop()
target = value
var _stack: Array[Frame]
## Saves properties to a new stack frame and then overwrites them on the target.
func push(properties: Dictionary) -> void:
var frame := Frame.new()
_stack.push_back(frame)
frame.save_and_modify(target, properties)
## Pops the top stack frame and restores saved properties from it.
func pop() -> void:
var frame: Frame = _stack.pop_back()
if frame:
frame.restore(target)
## Whether the stack is empty.
func is_empty() -> bool:
return _stack.is_empty()