mj177/addons/cyclops_level_builder/util/cyclops_settings.gd
2025-02-03 19:17:20 +01:00

228 lines
5.8 KiB
GDScript

# MIT License
#
# Copyright (c) 2023 Mark McKay
# https://github.com/blackears/cyclopsLevelBuilder
#
# 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.
@tool
extends Resource
class_name CyclopsSettings
@export var definition_map:Dictionary
@export var lookup:Dictionary
var float_regex_strn:String = "[+-]?([0-9]*[.])?[0-9]+"
var regex_int = RegEx.create_from_string("[0-9]+")
var regex_float = RegEx.create_from_string(float_regex_strn)
var regex_color = RegEx.create_from_string("color\\(" + float_regex_strn + "\\)")
class SettingDef:
var name:String
var default_value
var type:Variant.Type
var hint:PropertyHint
var hint_string:String
func value_to_text(value, type:int)->String:
match type:
TYPE_BOOL:
return "true" if value else "false"
TYPE_COLOR:
return JSON.stringify([value.r, value.g, value.b, value.a])
TYPE_FLOAT:
return str(value)
TYPE_INT:
return str(value)
TYPE_NODE_PATH:
return str(value)
TYPE_STRING:
return "\"" + value + "\""
TYPE_TRANSFORM2D:
var a:Transform2D = value
return JSON.stringify({"x": [a.x.x, a.x.y],
"y": [a.y.x, a.y.y],
"o": [a.origin.x, a.origin.y],
})
TYPE_TRANSFORM3D:
var a:Transform3D = value
return JSON.stringify({"x": [a.basis.x.x, a.basis.x.y, a.basis.x.z],
"y": [a.basis.y.x, a.basis.y.y, a.basis.y.z],
"z": [a.basis.z.x, a.basis.z.y, a.basis.z.z],
"o": [a.origin.x, a.origin.y, a.origin.z],
})
TYPE_VECTOR2:
var a:Vector2 = value
return JSON.stringify([a.x, a.y])
TYPE_VECTOR3:
var a:Vector3 = value
return JSON.stringify([a.x, a.y, a.z])
TYPE_VECTOR4:
var a:Vector4 = value
return JSON.stringify([a.x, a.y, a.z, a.w])
_:
return ""
func text_to_value(text:String, type:int):
text = text.lstrip(" ").rstrip(" ")
match type:
TYPE_BOOL:
return text.to_lower() == "true"
TYPE_COLOR:
var a:Array = JSON.parse_string(text)
return Color(a[0], a[1], a[2], a[3])
TYPE_FLOAT:
return float(text)
TYPE_INT:
return int(text)
TYPE_NODE_PATH:
return NodePath(text)
TYPE_STRING:
#Trim starting and ending quotes
return text.substr(1, text.length() - 2)
TYPE_TRANSFORM2D:
var a:Dictionary = JSON.parse_string(text)
return Transform2D(Vector2(a["x"][0], a["x"][1]),
Vector2(a["y"][0], a["y"][1]),
Vector2(a["o"][0], a["o"][1]))
TYPE_TRANSFORM3D:
var a:Dictionary = JSON.parse_string(text)
return Transform3D(Vector3(a["x"][0], a["x"][1], a["x"][2]),
Vector3(a["y"][0], a["y"][1], a["y"][2]),
Vector3(a["z"][0], a["z"][1], a["z"][2]),
Vector3(a["o"][0], a["o"][1], a["o"][2]))
TYPE_VECTOR2:
var a:Array = JSON.parse_string(text)
return Vector2(a[0], a[1])
TYPE_VECTOR3:
var a:Array = JSON.parse_string(text)
return Vector3(a[0], a[1], a[2])
TYPE_VECTOR4:
var a:Array = JSON.parse_string(text)
return Vector4(a[0], a[1], a[2], a[3])
_:
return null
func save_to_file(path:String):
var keys:Array = lookup.keys()
keys.sort()
var f:FileAccess = FileAccess.open(path, FileAccess.WRITE)
if !f:
return
for key in keys:
var def:SettingDef = definition_map[key]
f.store_line("%s=%s" % [key, value_to_text(lookup[key], def.type)])
f.close()
func load_from_file(path:String):
lookup.clear()
var f:FileAccess = FileAccess.open(path, FileAccess.READ)
while !f.eof_reached():
var line:String = f.get_line()
line = line.lstrip(" ")
if line.is_empty() || line[0] == "#":
continue
var idx = line.find("=")
if idx == -1:
continue
var name:String = line.substr(0, idx)
var value_text:String = line.substr(idx + 1)
if !definition_map.has(name):
continue
var def:SettingDef = definition_map[name]
set_property(name, text_to_value(value_text, def.type))
func add_setting(name:String, default_value, type:Variant.Type, hint:PropertyHint = PROPERTY_HINT_NONE, hint_string:String = ""):
var def:SettingDef = SettingDef.new()
def.name = name
def.default_value = default_value
def.type = type
def.hint = hint
def.hint_string = hint_string
definition_map[name] = def
func set_property(name:String, value):
if !definition_map.has(name):
push_error("Unknown setting name " + name)
return
var def:SettingDef = definition_map[name]
var var_type:int = typeof(value)
if var_type != def.type:
push_error("Settings error: Bad setting type. Needed %s but got %s" % [def.type, var_type])
return
lookup[name] = value
func has_property(name:String)->bool:
return definition_map.has(name)
func get_property(name:String):
#print("lookup ", name)
if !definition_map.has(name):
push_error("Unknown setting name " + name)
return null
#print("is defined ", name)
if lookup.has(name):
return lookup[name]
#print("returning default ", name)
var def:SettingDef = definition_map[name]
return def.default_value