Reflect.hx 5.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. /*
  2. * Copyright (C)2005-2019 Haxe Foundation
  3. *
  4. * Permission is hereby granted, free of charge, to any person obtaining a
  5. * copy of this software and associated documentation files (the "Software"),
  6. * to deal in the Software without restriction, including without limitation
  7. * the rights to use, copy, modify, merge, publish, distribute, sublicense,
  8. * and/or sell copies of the Software, and to permit persons to whom the
  9. * Software is furnished to do so, subject to the following conditions:
  10. *
  11. * The above copyright notice and this permission notice shall be included in
  12. * all copies or substantial portions of the Software.
  13. *
  14. * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  15. * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  16. * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  17. * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  18. * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
  19. * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
  20. * DEALINGS IN THE SOFTWARE.
  21. */
  22. import php.Boot;
  23. import php.Syntax;
  24. import php.Closure;
  25. import php.Const;
  26. import php.NativeAssocArray;
  27. import haxe.Constraints;
  28. using php.Global;
  29. @:coreApi class Reflect {
  30. public static function hasField( o : Dynamic, field : String ) : Bool {
  31. if (!o.is_object()) return false;
  32. if (o.property_exists(field)) return true;
  33. if (Boot.isClass(o)) {
  34. var phpClassName = Boot.castClass(o).phpClassName;
  35. return Global.property_exists(phpClassName, field) || Global.method_exists(phpClassName, field) || Global.defined('$phpClassName::$field');
  36. }
  37. return false;
  38. }
  39. public static function field( o : Dynamic, field : String ) : Dynamic {
  40. if (o.is_string()) {
  41. return Syntax.field(Boot.dynamicString(o), field);
  42. }
  43. if (!o.is_object()) return null;
  44. if (field == '' && Const.PHP_VERSION_ID < 70100) {
  45. return Syntax.coalesce(Syntax.array(o)[field], null);
  46. }
  47. if (o.property_exists(field)) {
  48. return Syntax.field(o, field);
  49. }
  50. if (o.method_exists(field)) {
  51. return Boot.getInstanceClosure(o, field);
  52. }
  53. if (Boot.isClass(o)) {
  54. var phpClassName = Boot.castClass(o).phpClassName;
  55. if (Global.defined('$phpClassName::$field')) {
  56. return Global.constant('$phpClassName::$field');
  57. }
  58. if (Global.property_exists(phpClassName, field)) {
  59. return Syntax.field(o, field);
  60. }
  61. if (Global.method_exists(phpClassName, field)) {
  62. return Boot.getStaticClosure(phpClassName, field);
  63. }
  64. }
  65. return null;
  66. }
  67. public static function setField( o : Dynamic, field : String, value : Dynamic ) : Void {
  68. Syntax.setField(o, field, value);
  69. }
  70. public static function getProperty( o : Dynamic, field : String ) : Dynamic {
  71. if (o.is_object()) {
  72. if (Boot.isClass(o)) {
  73. var phpClassName = Boot.castClass(o).phpClassName;
  74. if (Boot.hasGetter(phpClassName, field)) {
  75. return Syntax.staticCall(phpClassName, 'get_$field');
  76. }
  77. } else if (Boot.hasGetter(Global.get_class(o), field)) {
  78. return Syntax.call(o, 'get_$field');
  79. }
  80. }
  81. return Reflect.field(o, field);
  82. }
  83. public static function setProperty( o : Dynamic, field : String, value : Dynamic ) : Void {
  84. if (o.is_object()) {
  85. if (Boot.hasSetter(Global.get_class(o), field)) {
  86. Syntax.call(o, 'set_$field', value);
  87. } else {
  88. Syntax.setField(o, field, value);
  89. }
  90. }
  91. }
  92. public static function callMethod( o : Dynamic, func : Function, args : Array<Dynamic> ) : Dynamic {
  93. return Global.call_user_func_array(func, @:privateAccess args.arr);
  94. }
  95. public static function fields( o : Dynamic ) : Array<String> {
  96. if (Global.is_object(o)) {
  97. return @:privateAccess Array.wrap(Global.get_object_vars(o).array_keys());
  98. }
  99. return [];
  100. }
  101. public static inline function isFunction( f : Dynamic ) : Bool {
  102. return Boot.isFunction(f);
  103. }
  104. public static function compare<T>( a : T, b : T ) : Int {
  105. if (a == b) return 0;
  106. if (Global.is_string(a)){
  107. return Global.strcmp(cast a, cast b);
  108. } else {
  109. return ((cast a) > (cast b) ? 1 : -1);
  110. }
  111. }
  112. public static function compareMethods( f1 : Dynamic, f2 : Dynamic ) : Bool {
  113. if (Boot.isHxClosure(f1) && Boot.isHxClosure(f2)) {
  114. return f1.equals(f2);
  115. } else {
  116. return f1 == f2;
  117. }
  118. }
  119. public static function isObject( v : Dynamic ) : Bool {
  120. if (Boot.isEnumValue(v)) {
  121. return false;
  122. } else {
  123. return v.is_object() || v.is_string();
  124. }
  125. }
  126. public static inline function isEnumValue( v : Dynamic ) : Bool {
  127. return Boot.isEnumValue(v);
  128. }
  129. public static function deleteField( o : Dynamic, field : String ) : Bool {
  130. if (hasField(o, field)) {
  131. Global.unset(Syntax.field(o, field));
  132. return true;
  133. } else {
  134. return false;
  135. }
  136. }
  137. public static function copy<T>( o : Null<T> ) : Null<T> {
  138. if (Boot.isAnon(o)) {
  139. return Syntax.clone(o);
  140. } else {
  141. return null;
  142. }
  143. }
  144. @:overload(function( f : Array<Dynamic> -> Void ) : Dynamic {})
  145. public static function makeVarArgs( f : Array<Dynamic> -> Dynamic ) : Dynamic {
  146. return function () {
  147. return Global.call_user_func(f, @:privateAccess Array.wrap(Global.func_get_args()));
  148. }
  149. }
  150. }