ArrayObj.hx 8.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388
  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. package hl.types;
  23. import haxe.iterators.ArrayIterator;
  24. import haxe.iterators.ArrayKeyValueIterator;
  25. class ArrayObjIterator<T> extends ArrayIterator<T> {
  26. var arr:ArrayObj<T>;
  27. public inline function new(arr:ArrayObj<T>) {
  28. super((null:Dynamic));
  29. this.arr = arr;
  30. }
  31. override public function hasNext():Bool {
  32. return current < arr.length;
  33. }
  34. override public function next():T {
  35. return @:privateAccess arr.array[current++];
  36. }
  37. }
  38. class ArrayObjKeyValueIterator<T> extends ArrayKeyValueIterator<T> {
  39. var arr:ArrayObj<T>;
  40. public inline function new(arr:ArrayObj<T>) {
  41. super((null:Dynamic));
  42. this.arr = arr;
  43. }
  44. override public function hasNext():Bool {
  45. return current < arr.length;
  46. }
  47. override public function next():{key:Int, value:T} {
  48. var v = @:privateAccess arr.array[current];
  49. return {key:current++, value:v};
  50. }
  51. }
  52. @:keep
  53. class ArrayObj<T> extends ArrayBase {
  54. var array:hl.NativeArray<Dynamic>;
  55. public function new() {
  56. length = 0;
  57. array = new NativeArray<Dynamic>(0);
  58. }
  59. public function concat(a:ArrayObj<T>):ArrayObj<T> {
  60. var arr = new hl.NativeArray(length + a.length);
  61. arr.blit(0, array, 0, length);
  62. arr.blit(length, a.array, 0, a.length);
  63. return alloc(cast arr);
  64. }
  65. override function join(sep:String):String {
  66. var b = new StringBuf();
  67. for (i in 0...length) {
  68. if (i > 0)
  69. b.add(sep);
  70. b.add(array[i]);
  71. }
  72. return b.toString();
  73. }
  74. override function isArrayObj() {
  75. return true;
  76. }
  77. public function pop():Null<T> {
  78. if (length == 0)
  79. return null;
  80. length--;
  81. var v = array[length];
  82. array[length] = null;
  83. return v;
  84. }
  85. public function push(x:T):Int {
  86. var len = length;
  87. if (array.length == len)
  88. __expand(len);
  89. else
  90. length++;
  91. array[len] = x;
  92. return length;
  93. }
  94. override function reverse():Void {
  95. for (i in 0...length >> 1) {
  96. var k = length - 1 - i;
  97. var tmp = array[i];
  98. array[i] = array[k];
  99. array[k] = tmp;
  100. }
  101. }
  102. public function shift():Null<T> {
  103. if (length == 0)
  104. return null;
  105. length--;
  106. var v = array[0];
  107. array.blit(0, array, 1, length);
  108. array[length] = null;
  109. return v;
  110. }
  111. override function slice(pos:Int, ?end:Int):ArrayObj<T> {
  112. if (pos < 0) {
  113. pos = this.length + pos;
  114. if (pos < 0)
  115. pos = 0;
  116. }
  117. var pend:Int;
  118. if (end == null)
  119. pend = this.length;
  120. else {
  121. pend = end;
  122. if (pend < 0)
  123. pend += this.length;
  124. if (pend > this.length)
  125. pend = this.length;
  126. }
  127. var len = pend - pos;
  128. if (len < 0)
  129. return new ArrayObj();
  130. return alloc(array.sub(pos, len));
  131. }
  132. public function sort(f:T->T->Int):Void {
  133. // TODO : use native call ?
  134. haxe.ds.ArraySort.sort(cast this, f);
  135. }
  136. override function splice(pos:Int, len:Int):ArrayObj<T> {
  137. if (len < 0)
  138. return new ArrayObj();
  139. if (pos < 0) {
  140. pos = this.length + pos;
  141. if (pos < 0)
  142. pos = 0;
  143. }
  144. if (pos > this.length) {
  145. pos = 0;
  146. len = 0;
  147. } else if (pos + len > this.length) {
  148. len = this.length - pos;
  149. if (len < 0)
  150. len = 0;
  151. }
  152. var a = this.array;
  153. var ret:ArrayObj<T> = alloc(cast a.sub(pos, len));
  154. var end = pos + len;
  155. a.blit(pos, a, end, this.length - end);
  156. this.length -= len;
  157. while (--len >= 0)
  158. a[this.length + len] = null;
  159. return ret;
  160. }
  161. @:access(Std.toStringDepth)
  162. override function toString():String {
  163. if (Std.toStringDepth >= 5)
  164. return "...";
  165. Std.toStringDepth++;
  166. var b = new StringBuf();
  167. b.addChar("[".code);
  168. try {
  169. for (i in 0...length) {
  170. if (i > 0)
  171. b.addChar(",".code);
  172. b.add(array[i]);
  173. }
  174. } catch (e:Dynamic) {
  175. Std.toStringDepth--;
  176. hl.Api.rethrow(e);
  177. }
  178. b.addChar("]".code);
  179. Std.toStringDepth--;
  180. return b.toString();
  181. }
  182. public function unshift(x:T):Void {
  183. if (length == array.length)
  184. __expand(length)
  185. else
  186. length++;
  187. array.blit(1, array, 0, length - 1);
  188. array[0] = x;
  189. }
  190. public function insert(pos:Int, x:T):Void {
  191. if (pos < 0) {
  192. pos = length + pos;
  193. if (pos < 0)
  194. pos = 0;
  195. } else if (pos > length)
  196. pos = length;
  197. if (length == array.length)
  198. __expand(length)
  199. else
  200. length++;
  201. array.blit(pos + 1, array, pos, length - pos - 1);
  202. array[pos] = x;
  203. }
  204. public function contains(x:T):Bool {
  205. return indexOf(x) != -1;
  206. }
  207. public function remove(x:T):Bool {
  208. var i = indexOf(x);
  209. if (i < 0)
  210. return false;
  211. length--;
  212. array.blit(i, array, i + 1, length - i);
  213. array[length] = null;
  214. return true;
  215. }
  216. public function indexOf(x:T, ?fromIndex:Int):Int {
  217. var i:Int = fromIndex;
  218. if (i < 0) {
  219. i += length;
  220. if (i < 0)
  221. i = 0;
  222. }
  223. var length = length;
  224. var array = array;
  225. while (i < length) {
  226. if (array[i] == x)
  227. return i;
  228. i++;
  229. }
  230. return -1;
  231. }
  232. override function blit(pos:Int, src:ArrayBase.ArrayAccess, srcpos:Int, len:Int):Void {
  233. var src = (cast src : ArrayObj<T>);
  234. if (pos < 0 || srcpos < 0 || len < 0 || pos + len > length || srcpos + len > src.length)
  235. throw haxe.io.Error.OutsideBounds;
  236. array.blit(pos, src.array, srcpos, len);
  237. }
  238. public function lastIndexOf(x:T, ?fromIndex:Int):Int {
  239. var len = length;
  240. var i:Int = fromIndex != null ? fromIndex : len - 1;
  241. if (i >= len)
  242. i = len - 1;
  243. else if (i < 0)
  244. i += len;
  245. while (i >= 0) {
  246. if (array[i] == x)
  247. return i;
  248. i--;
  249. }
  250. return -1;
  251. }
  252. public function copy():ArrayObj<T> {
  253. var n = new NativeArray<Dynamic>(length);
  254. n.blit(0, array, 0, length);
  255. return alloc(n);
  256. }
  257. public function iterator():ArrayIterator<T> {
  258. return new ArrayObjIterator(this);
  259. }
  260. public function keyValueIterator():ArrayKeyValueIterator<T> {
  261. return new ArrayObjKeyValueIterator<T>(this);
  262. }
  263. public function map<S>(f:T->S):ArrayDyn {
  264. var a = new ArrayObj();
  265. if (length > 0)
  266. a.__expand(length - 1);
  267. for (i in 0...length)
  268. a.array[i] = f(array[i]);
  269. return ArrayDyn.alloc(a, true);
  270. }
  271. public function filter(f:T->Bool):ArrayObj<T> {
  272. var a = new ArrayObj();
  273. for (i in 0...length) {
  274. var v = array[i];
  275. if (f(v))
  276. a.push(v);
  277. }
  278. return a;
  279. }
  280. override public function resize(len:Int):Void {
  281. if (length < len) {
  282. __expand(len - 1);
  283. } else if (length > len) {
  284. for (i in length...len) {
  285. array[i] = null;
  286. }
  287. this.length = len;
  288. }
  289. }
  290. // called by compiler when accessing the array outside of its bounds, might trigger resize
  291. function __expand(index:Int) {
  292. if (index < 0)
  293. throw "Invalid array index " + index;
  294. var newlen = index + 1;
  295. var size:Int = array.length;
  296. if (newlen > size) {
  297. var next = (size * 3) >> 1;
  298. if (next < newlen)
  299. next = newlen;
  300. var arr2 = new hl.NativeArray<Dynamic>(next);
  301. arr2.blit(0, array, 0, length);
  302. array = arr2;
  303. }
  304. length = newlen;
  305. }
  306. override function getDyn(pos:Int):Dynamic {
  307. var pos:UInt = pos;
  308. if (pos >= length)
  309. return null;
  310. return array[pos];
  311. }
  312. override function setDyn(pos:Int, v:Dynamic) {
  313. var pos:UInt = pos;
  314. if (pos >= length)
  315. __expand(pos);
  316. array[pos] = Api.safeCast(v, array.getType());
  317. }
  318. override function pushDyn(v:Dynamic)
  319. return push(v);
  320. override function popDyn():Null<Dynamic>
  321. return pop();
  322. override function shiftDyn():Null<Dynamic>
  323. return shift();
  324. override function unshiftDyn(v:Dynamic)
  325. unshift(v);
  326. override function insertDyn(pos:Int, v:Dynamic)
  327. insert(pos, v);
  328. override function containsDyn(v:Dynamic)
  329. return contains(v);
  330. override function removeDyn(v:Dynamic)
  331. return remove(v);
  332. override function sortDyn(f:Dynamic->Dynamic->Int)
  333. sort(f);
  334. public static function alloc<T>(a:hl.NativeArray<T>):ArrayObj<T> {
  335. var arr:ArrayObj<T> = untyped $new(ArrayObj);
  336. arr.array = a;
  337. arr.length = a.length;
  338. return arr;
  339. }
  340. }