You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

279 lines
9.2 KiB

6 months ago
extends DialogicSubsystem
## Subsystem that manages variables and allows to access them.
## Emitted if a dialogic variable changes, gives a dictionary with the following keys:[br]
## [br]
## Key | Value Type | Value [br]
## ----------- | ------------- | ----- [br]
## `variable` | [type String] | The name of the variable that is getting changed. [br]
## `new_value` | [type Variant]| The value that [variable] has after the change (the result). [br]
signal variable_changed(info:Dictionary)
## Emitted on a set variable event, gives a dictionary with the following keys:[br]
## [br]
## Key | Value Type | Value [br]
## ----------- | ------------- | ----- [br]
## `variable` | [type String] | The name of the variable that is getting changed. [br]
## `orig_value`| [type Variant]| The value that [variable] had before. [br]
## `new_value` | [type Variant]| The value that [variable] has after the change (the result). [br]
## `value` | [type Variant]| The value that the variable is changed by/to. [br]
## `value_str` | [type String] | Whatever has been given as the value (not interpreted, so a variable is just a string).[br]
signal variable_was_set(info:Dictionary)
#region STATE
####################################################################################################
func clear_game_state(clear_flag:=DialogicGameHandler.ClearFlags.FULL_CLEAR):
# loading default variables
if ! clear_flag & DialogicGameHandler.ClearFlags.KEEP_VARIABLES:
reset()
func load_game_state(load_flag:=LoadFlags.FULL_LOAD):
if load_flag == LoadFlags.ONLY_DNODES:
return
dialogic.current_state_info['variables'] = merge_folder(dialogic.current_state_info['variables'], ProjectSettings.get_setting('dialogic/variables', {}).duplicate(true))
#endregion
#region MAIN METHODS
####################################################################################################
## This function will try to get the value of variables provided inside curly brackets
## and replace them with their values.
## It will:
## - look for the strings to replace
## - search all autoloads
## - try to get the value from context
##
## So if you provide a string like `Hello, how are you doing {Game.player_name}
## it will try to search for an autoload with the name `Game` and get the value
## of `player_name` to replace it.
func parse_variables(text:String) -> String:
# First some dirty checks to avoid parsing
if not '{' in text:
return text
# Trying to extract the curly brackets from the text
var regex := RegEx.new()
regex.compile("(?<!\\\\)\\{(?<variable>([^{}]|\\{.*\\})*)\\}")
var parsed := text.replace('\\{', '{')
for result in regex.search_all(text):
var value: Variant = get_variable(result.get_string('variable'), "<NOT FOUND>")
parsed = parsed.replace("{"+result.get_string('variable')+"}", str(value))
return parsed
func set_variable(variable_name: String, value: Variant) -> bool:
variable_name = variable_name.trim_prefix('{').trim_suffix('}')
# First assume this is a simple dialogic variable
if has(variable_name):
DialogicUtil._set_value_in_dictionary(variable_name, dialogic.current_state_info['variables'], value)
variable_changed.emit({'variable':variable_name, 'new_value':value})
return true
# Second assume this is an autoload variable
elif '.' in variable_name:
var from := variable_name.get_slice('.', 0)
var variable := variable_name.trim_prefix(from+'.')
var autoloads := get_autoloads()
var object: Object = null
if from in autoloads:
object = autoloads[from]
while variable.count("."):
from = variable.get_slice('.', 0)
if from in object and object.get(from) is Object:
object = object.get(from)
variable = variable.trim_prefix(from+'.')
if object:
var sub_idx := ""
if '[' in variable:
sub_idx = variable.substr(variable.find('['))
variable = variable.trim_suffix(sub_idx)
sub_idx = sub_idx.trim_prefix('[').trim_suffix(']')
if variable in object:
match typeof(object.get(variable)):
TYPE_ARRAY:
if not sub_idx:
if typeof(value) == TYPE_ARRAY:
object.set(variable, value)
return true
elif sub_idx.is_valid_float():
object.get(variable).remove_at(int(sub_idx))
object.get(variable).insert(int(sub_idx), value)
return true
TYPE_DICTIONARY:
if not sub_idx:
if typeof(value) == TYPE_DICTIONARY:
object.set(variable, value)
return true
else:
object.get(variable).merge({str_to_var(sub_idx):value}, true)
return true
_:
object.set(variable, value)
return true
printerr("[Dialogic] Tried setting non-existant variable '"+variable_name+"'.")
return false
func get_variable(variable_path:String, default: Variant = null, no_warning := false) -> Variant:
if variable_path.begins_with('{') and variable_path.ends_with('}') and variable_path.count('{') == 1:
variable_path = variable_path.trim_prefix('{').trim_suffix('}')
# First assume this is just a single variable
var value: Variant = DialogicUtil._get_value_in_dictionary(variable_path, dialogic.current_state_info['variables'])
if value != null:
return value
# Second assume this is an expression.
else:
value = dialogic.Expressions.execute_string(variable_path, null, no_warning)
if value != null:
return value
# If everything fails, tell the user and return the default
if not no_warning:
printerr("[Dialogic] Failed parsing variable/expression '"+variable_path+"'.")
return default
## Resets all variables or a specific variable to the value(s) defined in the variable editor
func reset(variable:="") -> void:
if variable.is_empty():
dialogic.current_state_info['variables'] = ProjectSettings.get_setting("dialogic/variables", {}).duplicate(true)
else:
DialogicUtil._set_value_in_dictionary(variable, dialogic.current_state_info['variables'], DialogicUtil._get_value_in_dictionary(variable, ProjectSettings.get_setting('dialogic/variables', {})))
## Returns true if a variable with the given path exists
func has(variable:="") -> bool:
return DialogicUtil._get_value_in_dictionary(variable, dialogic.current_state_info['variables']) != null
## Allows to set dialogic built-in variables
func _set(property, value) -> bool:
property = str(property)
var variables: Dictionary = dialogic.current_state_info['variables']
if property in variables.keys():
if typeof(variables[property]) != TYPE_DICTIONARY:
variables[property] = value
return true
if value is VariableFolder:
return true
return false
## Allows to get dialogic built-in variables
func _get(property):
property = str(property)
if property in dialogic.current_state_info['variables'].keys():
if typeof(dialogic.current_state_info['variables'][property]) == TYPE_DICTIONARY:
return VariableFolder.new(dialogic.current_state_info['variables'][property], property, self)
else:
return DialogicUtil.logical_convert(dialogic.current_state_info['variables'][property])
func folders() -> Array:
var result := []
for i in dialogic.current_state_info['variables'].keys():
if dialogic.current_state_info['variables'][i] is Dictionary:
result.append(VariableFolder.new(dialogic.current_state_info['variables'][i], i, self))
return result
func variables(absolute:=false) -> Array:
var result := []
for i in dialogic.current_state_info['variables'].keys():
if not dialogic.current_state_info['variables'][i] is Dictionary:
result.append(i)
return result
#endregion
#region HELPERS
################################################################################
func get_autoloads() -> Dictionary:
var autoloads := {}
for node: Node in get_tree().root.get_children():
autoloads[node.name] = node
return autoloads
func merge_folder(new:Dictionary, defs:Dictionary) -> Dictionary:
# also go through all groups in this folder
for x in new.keys():
if x in defs and typeof(new[x]) == TYPE_DICTIONARY:
new[x] = merge_folder(new[x], defs[x])
# add all new variables
for x in defs.keys():
if not x in new:
new[x] = defs[x]
return new
#endregion
#region VARIABLE FOLDER
################################################################################
class VariableFolder:
var data := {}
var path := ""
var outside : DialogicSubsystem
func _init(_data:Dictionary, _path:String, _outside:DialogicSubsystem):
data = _data
path = _path
outside = _outside
func _get(property:StringName):
property = str(property)
if property in data:
if typeof(data[property]) == TYPE_DICTIONARY:
return VariableFolder.new(data[property], path+"."+property, outside)
else:
return DialogicUtil.logical_convert(data[property])
func _set(property:StringName, value:Variant) -> bool:
property = str(property)
if not value is VariableFolder:
DialogicUtil._set_value_in_dictionary(path+"."+property, outside.dialogic.current_state_info['variables'], value)
return true
func has(key:String) -> bool:
return key in data
func folders() -> Array:
var result := []
for i in data.keys():
if data[i] is Dictionary:
result.append(VariableFolder.new(data[i], path+"."+i, outside))
return result
func variables(absolute:=false) -> Array:
var result := []
for i in data.keys():
if not data[i] is Dictionary:
if absolute:
result.append(path+'.'+i)
else:
result.append(i)
return result
#endregion