scide: LookupDialog - redo lookup on classes after partial lookup
[supercollider.git] / SCClassLibrary / Common / Collections / ArrayedCollection.sc
blobd9e0307d1207fb47e60c73f550b0f4bd4ed13982
1 ArrayedCollection : SequenceableCollection {
2         *newClear { arg indexedSize = 0;
3                 _BasicNewClear
4                 ^this.primitiveFailed
5                 // creates a new instance with indexedSize indexable slots.
6                 // the slots are filled with nil, zero or something else
7                 // appropriate to the type of indexable slots in the
8                 // object.
9         }
11         // ArrayedCollections are vectors which have a
12         // fixed maximum capacity.
14         indexedSize { _BasicSize }
15         size { _BasicSize }
16         maxSize { _BasicMaxSize }
18         swap { arg i, j; var temp;
19                 _BasicSwap;
20                 ^this.primitiveFailed;
21         }
23         at { arg index;
24                 _BasicAt;
25                 ^this.primitiveFailed;
26         }
27         clipAt { arg index;
28                 _BasicClipAt;
29                 ^this.primitiveFailed;
30         }
31         wrapAt { arg index;
32                 _BasicWrapAt;
33                 ^this.primitiveFailed;
34         }
35         foldAt { arg index;
36                 _BasicFoldAt;
37                 ^this.primitiveFailed;
38         }
39         put { arg index, item;
40                 _BasicPut;
41                 ^this.primitiveFailed;
42         }
43         clipPut { arg index, item;
44                 _BasicClipPut;
45                 ^this.primitiveFailed;
46         }
47         wrapPut { arg index, item;
48                 _BasicWrapPut;
49                 ^this.primitiveFailed;
50         }
51         foldPut { arg index, item;
52                 _BasicFoldPut;
53                 ^this.primitiveFailed;
54         }
55         removeAt { arg index;
56                 _BasicRemoveAt;
57                 ^this.primitiveFailed;
58         }
59         takeAt { arg index;
60                 _BasicTakeAt;
61                 ^this.primitiveFailed;
62         }
64         indexOf { arg item;
65                 _ArrayIndexOf
66                 ^this.primitiveFailed;
67         }
68         indexOfGreaterThan { arg val;
69                 _ArrayIndexOfGreaterThan
70                 ^super.indexOfGreaterThan(val)
71         }
73         takeThese { arg func;
74                 var i = 0;
75                 while { i < this.size }
76                 {
77                         if (func.value(this[i], i)) {
78                                 this.takeAt(i)
79                         }{
80                                 i = i + 1
81                         }
82                 }
83         }
85         replace { arg find, replace;
86                 var index, out = [], array = this;
87                 find = find.asArray;
88                 replace = replace.asArray;
89                 while {
90                         (index = array.find(find)).notNil
91                 }{
92                         out = out ++ array.keep(index) ++ replace;
93                         array = array.drop(index + find.size);
94                 };
95                 ^out ++ array
96         }
98         // see counterparts to these in Object
99         slotSize {
100                 ^this.size;
101         }
102         slotAt { arg index;
103                 ^this.at(index);
104         }
105         slotPut { arg index, value;
106                 ^this.put(index, value);
107         }
108         slotKey { arg index;
109                 ^index
110         }
111         slotIndex { arg key;
112                 ^nil
113         }
114         getSlots {
115                 ^this.copy
116         }
117         setSlots { arg array;
118                 this.overWrite(array)
119         }
121         atModify { arg index, function; this.put(index, function.value(this.at(index), index)) }
122         atInc { arg index, inc=1; this.put(index, this.at(index)+inc); }
123         atDec { arg index, dec=1; this.put(index, this.at(index)-dec); }
125         isArray { ^true }
126         asArray { ^this }
128         copyRange { arg start, end;
129                 // copies the fixed part of an object and the indexed slots
130                 // from start to end.
131                 _ObjectCopyRange;
132                 ^this.primitiveFailed
133         }
134         copySeries { arg first, second, last;
135                 // copies elements from first to last, stepping by (second-first)
136                 // copySeries(3,5,11) copies elements 3,5,7,9,11
137                 // if second is nil then the step is 1.
138                 // copySeries(3,nil,11) copies elements 3,4,5,6,7,8,9,10,11
139                 _ObjectCopySeries;
140                 ^this.primitiveFailed
141         }
142         putSeries { arg first, second, last, value;
143                 // puts value into array at indices defined by the series. see copySeries.
144                 _ArrayPutSeries;
145                 ^this.primitiveFailed
146         }
148         add { arg item;
149                 // add item to end of array.
150                 // if the capacity is exceeded, this returns a new
151                 // ArrayedCollection.
152                 _ArrayAdd
153                 ^this.primitiveFailed;
154         }
155         addAll { arg aCollection;
156                 var array;
157                 _ArrayAddAll
158                 array = this;
159                 aCollection.asCollection.do({ arg item; array = array.add(item) }) ;
160                 ^array
161         }
162         putEach { arg keys, values;
163                 _ArrayPutEach
164                 ^super.putEach(keys, values)
165         }
166         extend { arg size, item;
167                 _ArrayExtend
168                 ^this.primitiveFailed
169         }
170         insert { arg index, item;
171                 // add item at specified index.
172                 // if the capacity is exceeded, this returns a new
173                 // ArrayedCollection.
174                 _ArrayInsert
175                 ^this.primitiveFailed;
176         }
177         addFirst { arg item; ^this.insert(0, item) }
178         addIfNotNil { arg item; if(item.notNil,{ ^this.add(item) }) }
179         pop {
180                 // remove and return last item in array
181                 _ArrayPop
182                 ^nil;
183         }
184         ++ { arg anArray;
185                 // concatenate two arrays of the same type
186                 // this primitive will handle all array element types
187                 _ArrayCat;
188                 // primitive fails if arrays are different types
189                 ^super ++ anArray
190         }
191         overWrite { arg aCollection, pos=0;
192                 var array, grow;
193                 _ArrayOverwrite
194                 ^this.primitiveFailed
195                 //array = this.growClear(pos + aCollection.size - this.size);
196                 //grow = pos + aCollection.size - this.size;
197                 //if (grow > 0, { array = this ++ this.class.newClear(grow); },{ array = this });
198                 //aCollection.do({ arg item, i; array.put(pos + i, item); });
199                 //^array
200         }
201         grow { arg sizeIncrease;
202                 // returns an array of sufficient capacity.
203                 // may return same object if it still has enough space or if sizeIncrease <= 0.
204                 _ArrayGrow
205                 ^this.primitiveFailed
206         }
207         growClear { arg sizeIncrease;
208                 // returns an array of sufficient capacity.
209                 // may return same object if it still has enough space or if sizeIncrease <= 0.
210                 // clears new space
211                 _ArrayGrowClear
212                 ^this.primitiveFailed
213         }
214         seriesFill { arg start, step;
215                 this.size.do({ arg i;
216                         this.put(i, start);
217                         start = start + step;
218                 });
219         }
221         fill { arg value;
222                 _ArrayFill
223                 ^this.primitiveFailed
224                 /* replaced by primitive
225                 var i = 0, size;
226                 size = this.size;
227                 while ({ i < size }, {
228                         this.put(i, val);
229                         i = i + 1;
230                 });
231                 */
232         }
233         do { arg function;
234                 // special byte codes inserted by compiler for this method
235                 var i=0;
236                 while ({ i < this.size }, {
237                         function.value(this.at(i), i);
238                         i = i + 1;
239                 })
240         }
241         reverseDo { arg function;
242                 // special byte codes inserted by compiler for this method
243                 var i=0, j=0;
244                 i = this.size - 1;
245                 while ({ i >= 0 },{
246                         function.value(this.at(i), j);
247                         i = i - 1;
248                         j = j + 1;
249                 })
250         }
251         reverse {
252                 var i = 0;
253                 var res = this.copy;
254                 var size1 = res.size - 1;
255                 var halfsize = res.size div: 2;
256                 halfsize.do({ arg i;
257                         res.swap(i, size1 - i);
258                 });
259                 ^res
260         }
262         windex {
263                 _ArrayWIndex
264                 ^this.primitiveFailed
265                 //              var r, sum = 0.0, index;
266                 //              r = 1.0.rand;
267                 //              this.detect({ arg weight, i;
268                 //                      sum = sum + weight;
269                 //                      if (sum >= r, {
270                 //                              index = i;
271                 //                              true;
272                 //                      },{ false });
273                 //              });
274                 //              ^index;
275         }
276         normalizeSum {
277                 _ArrayNormalizeSum
278                 ^(this * this.sum.reciprocal)
279         }
280         normalize { arg min=0.0, max=1.0;
281                 var minItem = this.minItem;
282                 var maxItem = this.maxItem;
283                 ^this.collect { |el| el.linlin(minItem, maxItem, min, max) };
284         }
286         asciiPlot {
287                 // draw the waveform down the page as asterisks
288                 var lo = this.minItem;
289                 var hi = this.maxItem;
290                 var scale = 80 / (hi - lo);
291                 this.size.do { |i|
292                         var pt = ((this[i] - lo) * scale).asInteger;
293                         pt.do({ " ".post; });
294                         "*\n".post;
295                 };
296         }
298         perfectShuffle {
299                 if(this.size < 2) { ^this.copy };
300                 ^this[(0 .. this.size div: 2 - 1).stutter + [0, this.size + 1 div: 2]]
301         }
303         performInPlace { arg selector, from, to, argList;
304                 ^this.overWrite(this.copyRange(from, to).performList(selector, argList), from)
305         }
307         clipExtend { arg length;
308                 ^this.extend(length, this.last)
309         }
311         // concepts borrowed from J programming language
312         rank {
313                 // rank is the number of dimensions in a multidimensional array.
314                 // see also Object-rank
315                 // this assumes every element has the same rank
316                 ^1 + this.first.rank
317         }
318         shape {
319                 // this assumes every element has the same shape
320                 ^[this.size] ++ this[0].shape
321         }
322         reshape { arg ... shape;
323                 var size = shape.product;
324                 var result = this.flat.wrapExtend(size);
325                 shape[1..].reverseDo {|n| result = result.clump(n) };
326                 ^result
327         }
328         reshapeLike { arg another, indexing=\wrapAt;
329                 var index = 0;
330                 var flat = this.flat;
331                 ^another.deepCollect(0x7FFFFFFF) {
332                         var item = flat.perform(indexing, index);
333                         index = index + 1;
334                         item;
335                 };
336         }
337         deepCollect { arg depth = 1, function, index = 0, rank = 0;
338                 if(depth.isNil) {
339                         rank = rank + 1;
340                         ^this.collect { |item, i| item.deepCollect(depth, function, i, rank) }
341                 };
342                 if (depth <= 0) {
343                         ^function.value(this, index, rank)
344                 };
345                 depth = depth - 1;
346                 rank = rank + 1;
347                 ^this.collect { |item, i| item.deepCollect(depth, function, i, rank) }
348         }
349         deepDo { arg depth = 1, function, index = 0, rank = 0;
350                 if(depth.isNil) {
351                         rank = rank + 1;
352                         ^this.do { |item, i| item.deepDo(depth, function, i, rank) }
353                 };
354                 if (depth <= 0) {
355                         function.value(this, index, rank);
356                         ^this
357                 };
358                 depth = depth - 1;
359                 rank = rank + 1;
360                 ^this.do { |item, i| item.deepDo(depth, function, i, rank) }
361         }
362         unbubble { arg depth=0, levels=1;
363                 if (depth <= 0) {
364                         // converts a size 1 array to the item.
365                         if (this.size > 1) { ^this };
366                         if (levels <= 1) { ^this[0] }
367                         ^this[0].unbubble(depth, levels-1)
368                 };
369                 ^this.collect {|item| item.unbubble(depth-1) }
370         }
371         bubble { arg depth=0, levels=1;
372                 if (depth <= 0) {
373                         if (levels <= 1) { ^[this] }
374                         ^[this.bubble(depth,levels-1)]
375                 };
376                 ^this.collect {|item| item.bubble(depth-1, levels) }
377         }
380         slice { arg ... cuts;
381                 var firstCut, index, list;
382                 if (cuts.size == 0) { ^this.copy };
384                 firstCut = cuts[0];
385                 if (firstCut.isNil) { list = this.copy } { list = this[firstCut.asArray] };
386                 if (cuts.size == 1) {
387                         ^list.unbubble
388                 }{
389                         cuts = cuts[1..];
390                         ^list.collect {|item| item.slice(*cuts) }.unbubble
391                 }
392         }
393         *iota { arg ... sizes;
394                 ^(0..sizes.product-1).reshape(*sizes)
395         }
398         // random distribution table
400         asRandomTable { arg size;
401                 var a=this, b;
402                 if(size.isNil) { size = this.size } { a = a.resamp1(size) };
403                 a = a.integrate; // incrementally integrate
404                 a = a.normalize(0, size-1); // normalize and scale by max index
405                 b = Array.fill(size, { arg i; a.indexInBetween(i) });  // flip array
406                 b = b / size // rescale to 0..1
407                 ^b
408         }
410         tableRand {
411                 ^this.blendAt((this.size - 1).asFloat.rand)
412         }
414         // osc bundle support
416         msgSize {
417                 _NetAddr_MsgSize;
418                 ^this.primitiveFailed
419         }
420         bundleSize {
421                 // array of messages
422                 ^([nil] ++ this).prBundleSize;
423         }
424         clumpBundles {
425                 var size=0, res, clumps, count=0, bundleSizes;
426                 bundleSizes = this.collect {|item| [item].bundleSize };
427                 bundleSizes.do { |a, i|
428                         size = size + a;
429                         if(size >= 8192) { clumps = clumps.add(count); count = 0; size = a };
430                         count = count + 1;
431                 };
432                 ^this.clumps(clumps);
433         }
435         prBundleSize {
436                 _NetAddr_BundleSize;
437                 ^this.primitiveFailed
438         }
439         includes { |item| ^this.indexOf(item).notNil }
442 RawArray : ArrayedCollection {
443         // species { ^this.class }
444         archiveAsCompileString { ^true }
445         archiveAsObject { ^true }
447         rate { ^\scalar }
449         readFromStream { |stream, method|
450                 if(method.notNil) {
451                         this.size.do({ |i|
452                                 this.put(i, stream.perform(method));
453                         })
454                 } {
455                         this.shouldNotImplement(thisMethod);
456                 }
457         }
459         powerset {
460                 ^this.as(Array).powerset
461         }
464 Int8Array[int8] : RawArray {
465         unarchive {
466                 _Unarchive
467                 ^this.primitiveFailed
468         }
470         readFromStream { |stream|
471                 super.readFromStream(stream, \getInt8);
472         }
475 Int16Array[int16] : RawArray {
476         readFromStream { |stream|
477                 super.readFromStream(stream, \getInt16);
478         }
481 Int32Array[int32] : RawArray {
482         readFromStream { |stream|
483                 super.readFromStream(stream, \getInt32);
484         }
487 FloatArray[float] : RawArray {
488         readFromStream { |stream|
489                 super.readFromStream(stream, \getFloat);
490         }
493 DoubleArray[double] : RawArray {
494         readFromStream { |stream|
495                 super.readFromStream(stream, \getDouble);
496         }
499         // readFromStream not implemented yet
500 SymbolArray[symbol] : RawArray {