mirror of
https://github.com/FunkinCrew/Funkin.git
synced 2024-11-26 01:28:16 -05:00
140 lines
3.3 KiB
Haxe
140 lines
3.3 KiB
Haxe
package funkin.util;
|
|
|
|
import haxe.Json;
|
|
import haxe.io.Bytes;
|
|
import thx.semver.Version;
|
|
|
|
typedef ScoreInput =
|
|
{
|
|
var d:Int; // Key pressed
|
|
var l:Int; // Duration
|
|
var t:Int; // Start timestamp
|
|
}
|
|
|
|
/**
|
|
* Functions dedicated to serializing and deserializing data.
|
|
* NOTE: Use `json2object` wherever possible, it's way more efficient.
|
|
*/
|
|
class SerializerUtil
|
|
{
|
|
static final INDENT_CHAR = "\t";
|
|
|
|
/**
|
|
* Convert a Haxe object to a JSON string.
|
|
* NOTE: Use `json2object.JsonWriter<T>` WHEREVER POSSIBLE. Do not use this one unless you ABSOLUTELY HAVE TO it's SLOW!
|
|
* And don't even THINK about using `haxe.Json.stringify` without the replacer!
|
|
*/
|
|
public static function toJSON(input:Dynamic, pretty:Bool = true):String
|
|
{
|
|
return Json.stringify(input, replacer, pretty ? INDENT_CHAR : null);
|
|
}
|
|
|
|
/**
|
|
* Convert a JSON string to a Haxe object.
|
|
*/
|
|
public static function fromJSON(input:String):Dynamic
|
|
{
|
|
try
|
|
{
|
|
return Json.parse(input);
|
|
}
|
|
catch (e)
|
|
{
|
|
trace('An error occurred while parsing JSON from string data');
|
|
trace(e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Convert a JSON byte array to a Haxe object.
|
|
*/
|
|
public static function fromJSONBytes(input:Bytes):Dynamic
|
|
{
|
|
try
|
|
{
|
|
return Json.parse(input.toString());
|
|
}
|
|
catch (e:Dynamic)
|
|
{
|
|
trace('An error occurred while parsing JSON from byte data');
|
|
trace(e);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public static function initSerializer():Void
|
|
{
|
|
haxe.Unserializer.DEFAULT_RESOLVER = new FunkinTypeResolver();
|
|
}
|
|
|
|
/**
|
|
* Serialize a Haxe object using the built-in Serializer.
|
|
* @param input The object to serialize
|
|
* @return The serialized object as a string
|
|
*/
|
|
public static function fromHaxeObject(input:Dynamic):String
|
|
{
|
|
return haxe.Serializer.run(input);
|
|
}
|
|
|
|
/**
|
|
* Convert a serialized Haxe object back into a Haxe object.
|
|
* @param input The serialized object as a string
|
|
* @return The deserialized object
|
|
*/
|
|
public static function toHaxeObject(input:String):Dynamic
|
|
{
|
|
return haxe.Unserializer.run(input);
|
|
}
|
|
|
|
/**
|
|
* Customize how certain types are serialized when converting to JSON.
|
|
*/
|
|
static function replacer(key:String, value:Dynamic):Dynamic
|
|
{
|
|
// Hacky because you can't use `isOfType` on a struct.
|
|
if (key == "version")
|
|
{
|
|
if (Std.isOfType(value, String)) return value;
|
|
|
|
// Stringify Version objects.
|
|
return serializeVersion(cast value);
|
|
}
|
|
|
|
// Else, return the value as-is.
|
|
return value;
|
|
}
|
|
|
|
static inline function serializeVersion(value:thx.semver.Version):String
|
|
{
|
|
var result = '${value.major}.${value.minor}.${value.patch}';
|
|
if (value.hasPre) result += '-${value.pre}';
|
|
// TODO: Merge fix for version.hasBuild
|
|
if (value.build.length > 0) result += '+${value.build}';
|
|
return result;
|
|
}
|
|
}
|
|
|
|
class FunkinTypeResolver
|
|
{
|
|
public function new()
|
|
{
|
|
// Blank constructor.
|
|
}
|
|
|
|
public function resolveClass(name:String):Class<Dynamic>
|
|
{
|
|
if (name == 'Dynamic')
|
|
{
|
|
FlxG.log.warn('Found invalid class type in save data, indicates partial save corruption.');
|
|
return null;
|
|
}
|
|
return Type.resolveClass(name);
|
|
};
|
|
|
|
public function resolveEnum(name:String):Enum<Dynamic>
|
|
{
|
|
return Type.resolveEnum(name);
|
|
};
|
|
}
|