blob: e16cbef904c657979f7124d82b9b251a687e3adb [file] [log] [blame]
// Copyright (c) 2012, the Dart project authors. Please see the AUTHORS file
// for details. All rights reserved. Use of this source code is governed by a
// BSD-style license that can be found in the LICENSE file.
// TODO(srdjan): Use shared array implementation.
class _ObjectArray<E> implements List<E> {
static final int _classId = (new _ObjectArray(0))._cid;
factory _ObjectArray(length) native "ObjectArray_allocate";
E operator [](int index) native "ObjectArray_getIndexed";
void operator []=(int index, E value) native "ObjectArray_setIndexed";
String toString() {
return IterableMixinWorkaround.toStringIterable(this,'[' , ']');
}
int get length native "ObjectArray_getLength";
void _copyFromObjectArray(_ObjectArray src,
int srcStart,
int dstStart,
int count)
native "ObjectArray_copyFromObjectArray";
void insert(int index, E element) {
throw new UnsupportedError(
"Cannot add to a non-extendable array");
}
void insertAll(int index, Iterable<E> iterable) {
throw new UnsupportedError(
"Cannot add to a non-extendable array");
}
void setAll(int index, Iterable<E> iterable) {
IterableMixinWorkaround.setAllList(this, index, iterable);
}
E removeAt(int index) {
throw new UnsupportedError(
"Cannot remove element of a non-extendable array");
}
bool remove(Object element) {
throw new UnsupportedError(
"Cannot remove element of a non-extendable array");
}
void removeWhere(bool test(E element)) {
throw new UnsupportedError(
"Cannot remove element of a non-extendable array");
}
void retainWhere(bool test(E element)) {
throw new UnsupportedError(
"Cannot remove element of a non-extendable array");
}
Iterable<E> getRange(int start, [int end]) {
return IterableMixinWorkaround.getRangeList(this, start, end);
}
// List interface.
void setRange(int start, int end, Iterable<E> iterable, [int skipCount = 0]) {
if (start < 0 || start > this.length) {
throw new RangeError.range(start, 0, this.length);
}
if (end < start || end > this.length) {
throw new RangeError.range(end, start, this.length);
}
int length = end - start;
if (length == 0) return;
if (iterable is _ObjectArray) {
_copyFromObjectArray(iterable, skipCount, start, length);
} else {
List otherList;
int otherStart;
if (iterable is List) {
otherList = iterable;
otherStart = skipCount;
} else {
otherList =
iterable.skip(skipCount).take(length).toList(growable: false);
otherStart = 0;
}
Arrays.copy(otherList, otherStart, this, start, length);
}
}
void removeRange(int start, int end) {
throw new UnsupportedError(
"Cannot remove range of a non-extendable array");
}
void replaceRange(int start, int end, Iterable<E> iterable) {
throw new UnsupportedError(
"Cannot remove range of a non-extendable array");
}
void fillRange(int start, int end, [E fillValue]) {
IterableMixinWorkaround.fillRangeList(this, start, end, fillValue);
}
List<E> sublist(int start, [int end]) {
Arrays.indicesCheck(this, start, end);
if (end == null) end = this.length;
int length = end - start;
if (start == end) return [];
List list = new _GrowableObjectArray<E>.withCapacity(length);
list.length = length;
Arrays.copy(this, start, list, 0, length);
return list;
}
// Iterable interface.
bool contains(Object element) {
return IterableMixinWorkaround.contains(this, element);
}
void forEach(f(E element)) {
IterableMixinWorkaround.forEach(this, f);
}
String join([String separator = ""]) {
return IterableMixinWorkaround.joinList(this, separator);
}
Iterable map(f(E element)) {
return IterableMixinWorkaround.mapList(this, f);
}
E reduce(E combine(E value, E element)) {
return IterableMixinWorkaround.reduce(this, combine);
}
fold(initialValue, combine(previousValue, E element)) {
return IterableMixinWorkaround.fold(this, initialValue, combine);
}
Iterable<E> where(bool f(E element)) {
return IterableMixinWorkaround.where(this, f);
}
Iterable expand(Iterable f(E element)) {
return IterableMixinWorkaround.expand(this, f);
}
Iterable<E> take(int n) {
return IterableMixinWorkaround.takeList(this, n);
}
Iterable<E> takeWhile(bool test(E value)) {
return IterableMixinWorkaround.takeWhile(this, test);
}
Iterable<E> skip(int n) {
return IterableMixinWorkaround.skipList(this, n);
}
Iterable<E> skipWhile(bool test(E value)) {
return IterableMixinWorkaround.skipWhile(this, test);
}
bool every(bool f(E element)) {
return IterableMixinWorkaround.every(this, f);
}
bool any(bool f(E element)) {
return IterableMixinWorkaround.any(this, f);
}
E firstWhere(bool test(E value), {E orElse()}) {
return IterableMixinWorkaround.firstWhere(this, test, orElse);
}
E lastWhere(bool test(E value), {E orElse()}) {
return IterableMixinWorkaround.lastWhereList(this, test, orElse);
}
E singleWhere(bool test(E value)) {
return IterableMixinWorkaround.singleWhere(this, test);
}
E elementAt(int index) {
return this[index];
}
bool get isEmpty {
return this.length == 0;
}
bool get isNotEmpty => !isEmpty;
Iterable<E> get reversed => IterableMixinWorkaround.reversedList(this);
void sort([int compare(E a, E b)]) {
IterableMixinWorkaround.sortList(this, compare);
}
int indexOf(Object element, [int start = 0]) {
return Arrays.indexOf(this, element, start, this.length);
}
int lastIndexOf(Object element, [int start = null]) {
if (start == null) start = length - 1;
return Arrays.lastIndexOf(this, element, start);
}
Iterator<E> get iterator {
return new _FixedSizeArrayIterator<E>(this);
}
void add(E element) {
throw new UnsupportedError(
"Cannot add to a non-extendable array");
}
void addAll(Iterable<E> iterable) {
throw new UnsupportedError(
"Cannot add to a non-extendable array");
}
void clear() {
throw new UnsupportedError(
"Cannot clear a non-extendable array");
}
void set length(int length) {
throw new UnsupportedError(
"Cannot change the length of a non-extendable array");
}
E removeLast() {
throw new UnsupportedError(
"Cannot remove in a non-extendable array");
}
E get first {
if (length > 0) return this[0];
throw new StateError("No elements");
}
E get last {
if (length > 0) return this[length - 1];
throw new StateError("No elements");
}
E get single {
if (length == 1) return this[0];
if (length == 0) throw new StateError("No elements");
throw new StateError("More than one element");
}
List<E> toList({ bool growable: true}) {
return new List<E>.from(this, growable: growable);
}
Set<E> toSet() {
return new Set<E>.from(this);
}
Map<int, E> asMap() {
return IterableMixinWorkaround.asMapList(this);
}
}
// This is essentially the same class as _ObjectArray, but it does not
// permit any modification of array elements from Dart code. We use
// this class for arrays constructed from Dart array literals.
// TODO(hausner): We should consider the trade-offs between two
// classes (and inline cache misses) versus a field in the native
// implementation (checks when modifying). We should keep watching
// the inline cache misses.
class _ImmutableArray<E> implements List<E> {
static final int _classId = (const [])._cid;
factory _ImmutableArray._uninstantiable() {
throw new UnsupportedError(
"ImmutableArray can only be allocated by the VM");
}
E operator [](int index) native "ObjectArray_getIndexed";
void operator []=(int index, E value) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
int get length native "ObjectArray_getLength";
void insert(int index, E element) {
throw new UnsupportedError(
"Cannot add to an immutable array");
}
void insertAll(int index, Iterable<E> iterable) {
throw new UnsupportedError(
"Cannot add to an immutable array");
}
void setAll(int index, Iterable<E> iterable) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
E removeAt(int index) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
bool remove(Object element) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
void removeWhere(bool test(E element)) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
void retainWhere(bool test(E element)) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
void copyFrom(List src, int srcStart, int dstStart, int count) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
void setRange(int start, int end, Iterable<E> iterable, [int skipCount = 0]) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
void removeRange(int start, int end) {
throw new UnsupportedError(
"Cannot remove range of an immutable array");
}
void fillRange(int start, int end, [E fillValue]) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
void replaceRange(int start, int end, Iterable<E> iterable) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
List<E> sublist(int start, [int end]) {
Arrays.indicesCheck(this, start, end);
if (end == null) end = this.length;
int length = end - start;
if (start == end) return [];
List list = new List<E>();
list.length = length;
Arrays.copy(this, start, list, 0, length);
return list;
}
Iterable<E> getRange(int start, int end) {
return IterableMixinWorkaround.getRangeList(this, start, end);
}
// Collection interface.
bool contains(Object element) {
return IterableMixinWorkaround.contains(this, element);
}
void forEach(f(E element)) {
IterableMixinWorkaround.forEach(this, f);
}
Iterable map(f(E element)) {
return IterableMixinWorkaround.mapList(this, f);
}
String join([String separator = ""]) {
return IterableMixinWorkaround.joinList(this, separator);
}
E reduce(E combine(E value, E element)) {
return IterableMixinWorkaround.reduce(this, combine);
}
fold(initialValue, combine(previousValue, E element)) {
return IterableMixinWorkaround.fold(this, initialValue, combine);
}
Iterable<E> where(bool f(E element)) {
return IterableMixinWorkaround.where(this, f);
}
Iterable expand(Iterable f(E element)) {
return IterableMixinWorkaround.expand(this, f);
}
Iterable<E> take(int n) {
return IterableMixinWorkaround.takeList(this, n);
}
Iterable<E> takeWhile(bool test(E value)) {
return IterableMixinWorkaround.takeWhile(this, test);
}
Iterable<E> skip(int n) {
return IterableMixinWorkaround.skipList(this, n);
}
Iterable<E> skipWhile(bool test(E value)) {
return IterableMixinWorkaround.skipWhile(this, test);
}
bool every(bool f(E element)) {
return IterableMixinWorkaround.every(this, f);
}
bool any(bool f(E element)) {
return IterableMixinWorkaround.any(this, f);
}
E firstWhere(bool test(E value), {E orElse()}) {
return IterableMixinWorkaround.firstWhere(this, test, orElse);
}
E lastWhere(bool test(E value), {E orElse()}) {
return IterableMixinWorkaround.lastWhereList(this, test, orElse);
}
E singleWhere(bool test(E value)) {
return IterableMixinWorkaround.singleWhere(this, test);
}
E elementAt(int index) {
return this[index];
}
bool get isEmpty {
return this.length == 0;
}
bool get isNotEmpty => !isEmpty;
Iterable<E> get reversed => IterableMixinWorkaround.reversedList(this);
void sort([int compare(E a, E b)]) {
throw new UnsupportedError(
"Cannot modify an immutable array");
}
String toString() {
return IterableMixinWorkaround.toStringIterable(this, '[', ']');
}
int indexOf(Object element, [int start = 0]) {
return Arrays.indexOf(this, element, start, this.length);
}
int lastIndexOf(Object element, [int start = null]) {
if (start == null) start = length - 1;
return Arrays.lastIndexOf(this, element, start);
}
Iterator<E> get iterator {
return new _FixedSizeArrayIterator<E>(this);
}
void add(E element) {
throw new UnsupportedError(
"Cannot add to an immutable array");
}
void addAll(Iterable<E> elements) {
throw new UnsupportedError(
"Cannot add to an immutable array");
}
void clear() {
throw new UnsupportedError(
"Cannot clear an immutable array");
}
void set length(int length) {
throw new UnsupportedError(
"Cannot change the length of an immutable array");
}
E removeLast() {
throw new UnsupportedError(
"Cannot remove in a non-extendable array");
}
E get first {
if (length > 0) return this[0];
throw new StateError("No elements");
}
E get last {
if (length > 0) return this[length - 1];
throw new StateError("No elements");
}
E get single {
if (length == 1) return this[0];
if (length == 0) throw new StateError("No elements");
throw new StateError("More than one element");
}
List<E> toList({ bool growable: true }) {
return new List<E>.from(this, growable: growable);
}
Set<E> toSet() {
return new Set<E>.from(this);
}
Map<int, E> asMap() {
return IterableMixinWorkaround.asMapList(this);
}
}
// Iterator for arrays with fixed size.
class _FixedSizeArrayIterator<E> implements Iterator<E> {
final List<E> _array;
final int _length; // Cache array length for faster access.
int _position;
E _current;
_FixedSizeArrayIterator(List array)
: _array = array, _length = array.length, _position = -1 {
assert(array is _ObjectArray || array is _ImmutableArray);
}
bool moveNext() {
int nextPosition = _position + 1;
if (nextPosition < _length) {
_current = _array[nextPosition];
_position = nextPosition;
return true;
}
_position = _length;
_current = null;
return false;
}
E get current {
return _current;
}
}