ArrayObj.hx 7.9 KB

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