OLD | NEW |
1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file | 1 // Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file |
2 // for details. All rights reserved. Use of this source code is governed by a | 2 // for details. All rights reserved. Use of this source code is governed by a |
3 // BSD-style license that can be found in the LICENSE file. | 3 // BSD-style license that can be found in the LICENSE file. |
4 | 4 |
5 part of dart.collection; | 5 part of dart.collection; |
6 | 6 |
7 /** | 7 /** |
8 * This [Iterable] mixin implements all [Iterable] members except `iterator`. | 8 * This [Iterable] mixin implements all [Iterable] members except `iterator`. |
9 * | 9 * |
10 * All other methods are implemented in terms of `iterator`. | 10 * All other methods are implemented in terms of `iterator`. |
(...skipping 188 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
199 | 199 |
200 String toString() => IterableBase.iterableToShortString(this, '(', ')'); | 200 String toString() => IterableBase.iterableToShortString(this, '(', ')'); |
201 } | 201 } |
202 | 202 |
203 /** | 203 /** |
204 * Base class for implementing [Iterable]. | 204 * Base class for implementing [Iterable]. |
205 * | 205 * |
206 * This class implements all methods of [Iterable] except [Iterable.iterator] | 206 * This class implements all methods of [Iterable] except [Iterable.iterator] |
207 * in terms of `iterator`. | 207 * in terms of `iterator`. |
208 */ | 208 */ |
209 abstract class IterableBase<E> implements Iterable<E> { | 209 abstract class IterableBase<E> extends Iterable<E> { |
210 // TODO(lrn): Base this on IterableMixin if there ever becomes a way | |
211 // to combine const constructors and mixins. | |
212 const IterableBase(); | 210 const IterableBase(); |
213 | 211 |
214 Iterable map(f(E element)) => new MappedIterable<E, dynamic>(this, f); | |
215 | |
216 Iterable<E> where(bool f(E element)) => new WhereIterable<E>(this, f); | |
217 | |
218 Iterable expand(Iterable f(E element)) => | |
219 new ExpandIterable<E, dynamic>(this, f); | |
220 | |
221 bool contains(Object element) { | |
222 for (E e in this) { | |
223 if (e == element) return true; | |
224 } | |
225 return false; | |
226 } | |
227 | |
228 void forEach(void f(E element)) { | |
229 for (E element in this) f(element); | |
230 } | |
231 | |
232 E reduce(E combine(E value, E element)) { | |
233 Iterator<E> iterator = this.iterator; | |
234 if (!iterator.moveNext()) { | |
235 throw IterableElementError.noElement(); | |
236 } | |
237 E value = iterator.current; | |
238 while (iterator.moveNext()) { | |
239 value = combine(value, iterator.current); | |
240 } | |
241 return value; | |
242 } | |
243 | |
244 dynamic fold(var initialValue, | |
245 dynamic combine(var previousValue, E element)) { | |
246 var value = initialValue; | |
247 for (E element in this) value = combine(value, element); | |
248 return value; | |
249 } | |
250 | |
251 bool every(bool f(E element)) { | |
252 for (E element in this) { | |
253 if (!f(element)) return false; | |
254 } | |
255 return true; | |
256 } | |
257 | |
258 String join([String separator = ""]) { | |
259 Iterator<E> iterator = this.iterator; | |
260 if (!iterator.moveNext()) return ""; | |
261 StringBuffer buffer = new StringBuffer(); | |
262 if (separator == null || separator == "") { | |
263 do { | |
264 buffer.write("${iterator.current}"); | |
265 } while (iterator.moveNext()); | |
266 } else { | |
267 buffer.write("${iterator.current}"); | |
268 while (iterator.moveNext()) { | |
269 buffer.write(separator); | |
270 buffer.write("${iterator.current}"); | |
271 } | |
272 } | |
273 return buffer.toString(); | |
274 } | |
275 | |
276 bool any(bool f(E element)) { | |
277 for (E element in this) { | |
278 if (f(element)) return true; | |
279 } | |
280 return false; | |
281 } | |
282 | |
283 List<E> toList({ bool growable: true }) => | |
284 new List<E>.from(this, growable: growable); | |
285 | |
286 Set<E> toSet() => new Set<E>.from(this); | |
287 | |
288 int get length { | |
289 assert(this is! EfficientLength); | |
290 int count = 0; | |
291 Iterator it = iterator; | |
292 while (it.moveNext()) { | |
293 count++; | |
294 } | |
295 return count; | |
296 } | |
297 | |
298 bool get isEmpty => !iterator.moveNext(); | |
299 | |
300 bool get isNotEmpty => !isEmpty; | |
301 | |
302 Iterable<E> take(int n) { | |
303 return new TakeIterable<E>(this, n); | |
304 } | |
305 | |
306 Iterable<E> takeWhile(bool test(E value)) { | |
307 return new TakeWhileIterable<E>(this, test); | |
308 } | |
309 | |
310 Iterable<E> skip(int n) { | |
311 return new SkipIterable<E>(this, n); | |
312 } | |
313 | |
314 Iterable<E> skipWhile(bool test(E value)) { | |
315 return new SkipWhileIterable<E>(this, test); | |
316 } | |
317 | |
318 E get first { | |
319 Iterator it = iterator; | |
320 if (!it.moveNext()) { | |
321 throw IterableElementError.noElement(); | |
322 } | |
323 return it.current; | |
324 } | |
325 | |
326 E get last { | |
327 Iterator it = iterator; | |
328 if (!it.moveNext()) { | |
329 throw IterableElementError.noElement(); | |
330 } | |
331 E result; | |
332 do { | |
333 result = it.current; | |
334 } while(it.moveNext()); | |
335 return result; | |
336 } | |
337 | |
338 E get single { | |
339 Iterator it = iterator; | |
340 if (!it.moveNext()) throw IterableElementError.noElement(); | |
341 E result = it.current; | |
342 if (it.moveNext()) throw IterableElementError.tooMany(); | |
343 return result; | |
344 } | |
345 | |
346 E firstWhere(bool test(E value), { E orElse() }) { | |
347 for (E element in this) { | |
348 if (test(element)) return element; | |
349 } | |
350 if (orElse != null) return orElse(); | |
351 throw IterableElementError.noElement(); | |
352 } | |
353 | |
354 E lastWhere(bool test(E value), { E orElse() }) { | |
355 E result = null; | |
356 bool foundMatching = false; | |
357 for (E element in this) { | |
358 if (test(element)) { | |
359 result = element; | |
360 foundMatching = true; | |
361 } | |
362 } | |
363 if (foundMatching) return result; | |
364 if (orElse != null) return orElse(); | |
365 throw IterableElementError.noElement(); | |
366 } | |
367 | |
368 E singleWhere(bool test(E value)) { | |
369 E result = null; | |
370 bool foundMatching = false; | |
371 for (E element in this) { | |
372 if (test(element)) { | |
373 if (foundMatching) { | |
374 throw IterableElementError.tooMany(); | |
375 } | |
376 result = element; | |
377 foundMatching = true; | |
378 } | |
379 } | |
380 if (foundMatching) return result; | |
381 throw IterableElementError.noElement(); | |
382 } | |
383 | |
384 E elementAt(int index) { | |
385 if (index is! int) throw new ArgumentError.notNull("index"); | |
386 RangeError.checkNotNegative(index, "index"); | |
387 int elementIndex = 0; | |
388 for (E element in this) { | |
389 if (index == elementIndex) return element; | |
390 elementIndex++; | |
391 } | |
392 throw new RangeError.index(index, this, "index", null, elementIndex); | |
393 } | |
394 | |
395 /** | |
396 * Returns a string representation of (some of) the elements of `this`. | |
397 * | |
398 * Elements are represented by their own `toString` results. | |
399 * | |
400 * The representation always contains the first three elements. | |
401 * If there are less than a hundred elements in the iterable, it also | |
402 * contains the last two elements. | |
403 * | |
404 * If the resulting string isn't above 80 characters, more elements are | |
405 * included from the start of the iterable. | |
406 * | |
407 * The conversion may omit calling `toString` on some elements if they | |
408 * are known to not occur in the output, and it may stop iterating after | |
409 * a hundred elements. | |
410 */ | |
411 String toString() => iterableToShortString(this, '(', ')'); | |
412 | |
413 /** | 212 /** |
414 * Convert an `Iterable` to a string like [IterableBase.toString]. | 213 * Convert an `Iterable` to a string like [IterableBase.toString]. |
415 * | 214 * |
416 * Allows using other delimiters than '(' and ')'. | 215 * Allows using other delimiters than '(' and ')'. |
417 * | 216 * |
418 * Handles circular references where converting one of the elements | 217 * Handles circular references where converting one of the elements |
419 * to a string ends up converting [iterable] to a string again. | 218 * to a string ends up converting [iterable] to a string again. |
420 */ | 219 */ |
421 static String iterableToShortString(Iterable iterable, | 220 static String iterableToShortString(Iterable iterable, |
422 [String leftDelimiter = '(', | 221 [String leftDelimiter = '(', |
(...skipping 40 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
463 _toStringVisiting.add(iterable); | 262 _toStringVisiting.add(iterable); |
464 try { | 263 try { |
465 buffer.writeAll(iterable, ", "); | 264 buffer.writeAll(iterable, ", "); |
466 } finally { | 265 } finally { |
467 assert(identical(_toStringVisiting.last, iterable)); | 266 assert(identical(_toStringVisiting.last, iterable)); |
468 _toStringVisiting.removeLast(); | 267 _toStringVisiting.removeLast(); |
469 } | 268 } |
470 buffer.write(rightDelimiter); | 269 buffer.write(rightDelimiter); |
471 return buffer.toString(); | 270 return buffer.toString(); |
472 } | 271 } |
| 272 } |
473 | 273 |
474 /** A set used to identify cyclic lists during toString() calls. */ | 274 /** A set used to identify cyclic lists during toString() calls. */ |
475 static final List _toStringVisiting = []; | 275 final List _toStringVisiting = []; |
476 | 276 |
477 /** Check if we are currently visiting `o` in a toString call. */ | 277 /** Check if we are currently visiting `o` in a toString call. */ |
478 static bool _isToStringVisiting(Object o) { | 278 bool _isToStringVisiting(Object o) { |
479 for (int i = 0; i < _toStringVisiting.length; i++) { | 279 for (int i = 0; i < _toStringVisiting.length; i++) { |
480 if (identical(o, _toStringVisiting[i])) return true; | 280 if (identical(o, _toStringVisiting[i])) return true; |
481 } | 281 } |
482 return false; | 282 return false; |
| 283 } |
| 284 |
| 285 /** |
| 286 * Convert elments of [iterable] to strings and store them in [parts]. |
| 287 */ |
| 288 void _iterablePartsToStrings(Iterable iterable, List parts) { |
| 289 /* |
| 290 * This is the complicated part of [iterableToShortString]. |
| 291 * It is extracted as a separate function to avoid having too much code |
| 292 * inside the try/finally. |
| 293 */ |
| 294 /// Try to stay below this many characters. |
| 295 const int LENGTH_LIMIT = 80; |
| 296 /// Always at least this many elements at the start. |
| 297 const int HEAD_COUNT = 3; |
| 298 /// Always at least this many elements at the end. |
| 299 const int TAIL_COUNT = 2; |
| 300 /// Stop iterating after this many elements. Iterables can be infinite. |
| 301 const int MAX_COUNT = 100; |
| 302 // Per entry length overhead. It's for ", " for all after the first entry, |
| 303 // and for "(" and ")" for the initial entry. By pure luck, that's the same |
| 304 // number. |
| 305 const int OVERHEAD = 2; |
| 306 const int ELLIPSIS_SIZE = 3; // "...".length. |
| 307 |
| 308 int length = 0; |
| 309 int count = 0; |
| 310 Iterator it = iterable.iterator; |
| 311 // Initial run of elements, at least HEAD_COUNT, and then continue until |
| 312 // passing at most LENGTH_LIMIT characters. |
| 313 while (length < LENGTH_LIMIT || count < HEAD_COUNT) { |
| 314 if (!it.moveNext()) return; |
| 315 String next = "${it.current}"; |
| 316 parts.add(next); |
| 317 length += next.length + OVERHEAD; |
| 318 count++; |
483 } | 319 } |
484 | 320 |
485 /** | 321 String penultimateString; |
486 * Convert elments of [iterable] to strings and store them in [parts]. | 322 String ultimateString; |
487 */ | |
488 static void _iterablePartsToStrings(Iterable iterable, List parts) { | |
489 /* | |
490 * This is the complicated part of [iterableToShortString]. | |
491 * It is extracted as a separate function to avoid having too much code | |
492 * inside the try/finally. | |
493 */ | |
494 /// Try to stay below this many characters. | |
495 const int LENGTH_LIMIT = 80; | |
496 /// Always at least this many elements at the start. | |
497 const int HEAD_COUNT = 3; | |
498 /// Always at least this many elements at the end. | |
499 const int TAIL_COUNT = 2; | |
500 /// Stop iterating after this many elements. Iterables can be infinite. | |
501 const int MAX_COUNT = 100; | |
502 // Per entry length overhead. It's for ", " for all after the first entry, | |
503 // and for "(" and ")" for the initial entry. By pure luck, that's the same | |
504 // number. | |
505 const int OVERHEAD = 2; | |
506 const int ELLIPSIS_SIZE = 3; // "...".length. | |
507 | 323 |
508 int length = 0; | 324 // Find last two elements. One or more of them may already be in the |
509 int count = 0; | 325 // parts array. Include their length in `length`. |
510 Iterator it = iterable.iterator; | 326 var penultimate = null; |
511 // Initial run of elements, at least HEAD_COUNT, and then continue until | 327 var ultimate = null; |
512 // passing at most LENGTH_LIMIT characters. | 328 if (!it.moveNext()) { |
513 while (length < LENGTH_LIMIT || count < HEAD_COUNT) { | 329 if (count <= HEAD_COUNT + TAIL_COUNT) return; |
514 if (!it.moveNext()) return; | 330 ultimateString = parts.removeLast(); |
515 String next = "${it.current}"; | 331 penultimateString = parts.removeLast(); |
516 parts.add(next); | 332 } else { |
517 length += next.length + OVERHEAD; | 333 penultimate = it.current; |
| 334 count++; |
| 335 if (!it.moveNext()) { |
| 336 if (count <= HEAD_COUNT + 1) { |
| 337 parts.add("$penultimate"); |
| 338 return; |
| 339 } |
| 340 ultimateString = "$penultimate"; |
| 341 penultimateString = parts.removeLast(); |
| 342 length += ultimateString.length + OVERHEAD; |
| 343 } else { |
| 344 ultimate = it.current; |
518 count++; | 345 count++; |
519 } | 346 // Then keep looping, keeping the last two elements in variables. |
| 347 assert(count < MAX_COUNT); |
| 348 while (it.moveNext()) { |
| 349 penultimate = ultimate; |
| 350 ultimate = it.current; |
| 351 count++; |
| 352 if (count > MAX_COUNT) { |
| 353 // If we haven't found the end before MAX_COUNT, give up. |
| 354 // This cannot happen in the code above because each entry |
| 355 // increases length by at least two, so there is no way to |
| 356 // visit more than ~40 elements before this loop. |
520 | 357 |
521 String penultimateString; | 358 // Remove any surplus elements until length, including ", ...)", |
522 String ultimateString; | 359 // is at most LENGTH_LIMIT. |
523 | 360 while (length > LENGTH_LIMIT - ELLIPSIS_SIZE - OVERHEAD && |
524 // Find last two elements. One or more of them may already be in the | 361 count > HEAD_COUNT) { |
525 // parts array. Include their length in `length`. | 362 length -= parts.removeLast().length + OVERHEAD; |
526 var penultimate = null; | 363 count--; |
527 var ultimate = null; | 364 } |
528 if (!it.moveNext()) { | 365 parts.add("..."); |
529 if (count <= HEAD_COUNT + TAIL_COUNT) return; | |
530 ultimateString = parts.removeLast(); | |
531 penultimateString = parts.removeLast(); | |
532 } else { | |
533 penultimate = it.current; | |
534 count++; | |
535 if (!it.moveNext()) { | |
536 if (count <= HEAD_COUNT + 1) { | |
537 parts.add("$penultimate"); | |
538 return; | 366 return; |
539 } | 367 } |
540 ultimateString = "$penultimate"; | 368 } |
541 penultimateString = parts.removeLast(); | 369 penultimateString = "$penultimate"; |
542 length += ultimateString.length + OVERHEAD; | 370 ultimateString = "$ultimate"; |
543 } else { | 371 length += |
544 ultimate = it.current; | 372 ultimateString.length + penultimateString.length + 2 * OVERHEAD; |
545 count++; | 373 } |
546 // Then keep looping, keeping the last two elements in variables. | 374 } |
547 assert(count < MAX_COUNT); | |
548 while (it.moveNext()) { | |
549 penultimate = ultimate; | |
550 ultimate = it.current; | |
551 count++; | |
552 if (count > MAX_COUNT) { | |
553 // If we haven't found the end before MAX_COUNT, give up. | |
554 // This cannot happen in the code above because each entry | |
555 // increases length by at least two, so there is no way to | |
556 // visit more than ~40 elements before this loop. | |
557 | 375 |
558 // Remove any surplus elements until length, including ", ...)", | 376 // If there is a gap between the initial run and the last two, |
559 // is at most LENGTH_LIMIT. | 377 // prepare to add an ellipsis. |
560 while (length > LENGTH_LIMIT - ELLIPSIS_SIZE - OVERHEAD && | 378 String elision = null; |
561 count > HEAD_COUNT) { | 379 if (count > parts.length + TAIL_COUNT) { |
562 length -= parts.removeLast().length + OVERHEAD; | 380 elision = "..."; |
563 count--; | 381 length += ELLIPSIS_SIZE + OVERHEAD; |
564 } | 382 } |
565 parts.add("..."); | |
566 return; | |
567 } | |
568 } | |
569 penultimateString = "$penultimate"; | |
570 ultimateString = "$ultimate"; | |
571 length += | |
572 ultimateString.length + penultimateString.length + 2 * OVERHEAD; | |
573 } | |
574 } | |
575 | 383 |
576 // If there is a gap between the initial run and the last two, | 384 // If the last two elements were very long, and we have more than |
577 // prepare to add an ellipsis. | 385 // HEAD_COUNT elements in the initial run, drop some to make room for |
578 String elision = null; | 386 // the last two. |
579 if (count > parts.length + TAIL_COUNT) { | 387 while (length > LENGTH_LIMIT && parts.length > HEAD_COUNT) { |
| 388 length -= parts.removeLast().length + OVERHEAD; |
| 389 if (elision == null) { |
580 elision = "..."; | 390 elision = "..."; |
581 length += ELLIPSIS_SIZE + OVERHEAD; | 391 length += ELLIPSIS_SIZE + OVERHEAD; |
582 } | 392 } |
583 | |
584 // If the last two elements were very long, and we have more than | |
585 // HEAD_COUNT elements in the initial run, drop some to make room for | |
586 // the last two. | |
587 while (length > LENGTH_LIMIT && parts.length > HEAD_COUNT) { | |
588 length -= parts.removeLast().length + OVERHEAD; | |
589 if (elision == null) { | |
590 elision = "..."; | |
591 length += ELLIPSIS_SIZE + OVERHEAD; | |
592 } | |
593 } | |
594 if (elision != null) { | |
595 parts.add(elision); | |
596 } | |
597 parts.add(penultimateString); | |
598 parts.add(ultimateString); | |
599 } | 393 } |
| 394 if (elision != null) { |
| 395 parts.add(elision); |
| 396 } |
| 397 parts.add(penultimateString); |
| 398 parts.add(ultimateString); |
600 } | 399 } |
OLD | NEW |