167 lines
4.9 KiB
Haxe
167 lines
4.9 KiB
Haxe
|
/*
|
||
|
* Copyright (C)2005-2019 Haxe Foundation
|
||
|
*
|
||
|
* 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.
|
||
|
*/
|
||
|
|
||
|
import lua.Lua;
|
||
|
import lua.TableTools;
|
||
|
import lua.Boot;
|
||
|
|
||
|
@:coreApi class Reflect {
|
||
|
public inline static function hasField(o:Dynamic, field:String):Bool {
|
||
|
if (Lua.type(o) == "string" && (untyped String.prototype[field] != null || field == "length")) {
|
||
|
return true;
|
||
|
} else
|
||
|
return untyped o.__fields__ != null ? o.__fields__[field] != null : o[field] != null;
|
||
|
}
|
||
|
|
||
|
public static function field(o:Dynamic, field:String):Dynamic
|
||
|
untyped {
|
||
|
if (Lua.type(o) == "string") {
|
||
|
if (field == "length") {
|
||
|
return cast(o : String).length;
|
||
|
} else
|
||
|
return untyped String.prototype[field];
|
||
|
} else {
|
||
|
return try o[field] catch (e:Dynamic) null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public inline static function setField(o:Dynamic, field:String, value:Dynamic):Void
|
||
|
untyped {
|
||
|
o[field] = value;
|
||
|
}
|
||
|
|
||
|
public static function getProperty(o:Dynamic, field:String):Dynamic {
|
||
|
return if (o == null) {
|
||
|
untyped __define_feature__("Reflect.getProperty", null);
|
||
|
} else if (o.__properties__ != null && Reflect.field(o, "get_" + field) != null) {
|
||
|
callMethod(o, Reflect.field(o, "get_" + field), []);
|
||
|
} else {
|
||
|
Reflect.field(o, field);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public static function setProperty(o:Dynamic, field:String, value:Dynamic):Void
|
||
|
untyped {
|
||
|
if (o.__properties__ != null && o.__properties__["set_" + field]) {
|
||
|
var tmp:String = o.__properties__["set_" + field];
|
||
|
callMethod(o, Reflect.field(o, tmp), [value]);
|
||
|
} else {
|
||
|
o[field] = __define_feature__("Reflect.setProperty", value);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public static function callMethod(o:Dynamic, func:haxe.Constraints.Function, args:Array<Dynamic>):Dynamic {
|
||
|
if (args == null || args.length == 0) {
|
||
|
return func(o);
|
||
|
} else {
|
||
|
var self_arg = false;
|
||
|
if (o != null && o.__name__ == null) {
|
||
|
self_arg = true;
|
||
|
}
|
||
|
return if (self_arg) {
|
||
|
func(o, lua.TableTools.unpack(cast args, 0, args.length - 1));
|
||
|
} else {
|
||
|
func(lua.TableTools.unpack(cast args, 0, args.length - 1));
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public static function fields(o:Dynamic):Array<String> {
|
||
|
if (lua.Lua.type(o) == "string") {
|
||
|
return Reflect.fields(untyped String.prototype);
|
||
|
} else {
|
||
|
return untyped _hx_field_arr(o);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
public static function isFunction(f:Dynamic):Bool {
|
||
|
return Lua.type(f) == "function" && !(Boot.isClass(f) || Boot.isEnum(f));
|
||
|
}
|
||
|
|
||
|
public static function compare<T>(a:T, b:T):Int {
|
||
|
if (a == b)
|
||
|
return 0
|
||
|
else if (a == null)
|
||
|
return -1
|
||
|
else if (b == null)
|
||
|
return 1
|
||
|
else
|
||
|
return (cast a) > (cast b) ? 1 : -1;
|
||
|
}
|
||
|
|
||
|
public static function compareMethods(f1:Dynamic, f2:Dynamic):Bool {
|
||
|
return f1 == f2;
|
||
|
}
|
||
|
|
||
|
public static function isObject(v:Dynamic):Bool
|
||
|
untyped {
|
||
|
if (v == null)
|
||
|
return false;
|
||
|
var t = __lua__("type(v)");
|
||
|
return (t == "string" || (t == "table" && v.__enum__ == null))
|
||
|
|| (t == "function" && (lua.Boot.isClass(v) || lua.Boot.isEnum(v)) != null);
|
||
|
}
|
||
|
|
||
|
public static function isEnumValue(v:Dynamic):Bool {
|
||
|
return v != null && Std.isOfType(v, lua.Table) && v.__enum__ != null;
|
||
|
}
|
||
|
|
||
|
public static function deleteField(o:Dynamic, field:String):Bool
|
||
|
untyped {
|
||
|
if (!hasField(o, field))
|
||
|
return false;
|
||
|
o[field] = null;
|
||
|
o.__fields__[field] = null;
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
public static function copy<T>(o:Null<T>):Null<T> {
|
||
|
if (o == null)
|
||
|
return null;
|
||
|
var o2:Dynamic = {};
|
||
|
for (f in Reflect.fields(o))
|
||
|
Reflect.setField(o2, f, Reflect.field(o, f));
|
||
|
return o2;
|
||
|
}
|
||
|
|
||
|
@:overload(function(f:Array<Dynamic>->Void):Dynamic {})
|
||
|
public static function makeVarArgs(f:Array<Dynamic>->Dynamic):Dynamic {
|
||
|
/*
|
||
|
- Convert var arg to table
|
||
|
- Set indexing from zero
|
||
|
- Extract real table length
|
||
|
- Recreate as dynamic array
|
||
|
- Return function called with this array
|
||
|
*/
|
||
|
return untyped __lua__("function(...)
|
||
|
local a = {...}
|
||
|
local b = {}
|
||
|
local l = 0
|
||
|
for k, v in pairs(a) do
|
||
|
b[k-1] = v
|
||
|
l = math.max(k,l)
|
||
|
end
|
||
|
return f(_hx_tab_array(b, l))
|
||
|
end");
|
||
|
}
|
||
|
}
|