blob: caebbda69f85e122e7515d02f433fe97e0db1fd8 [file] [log] [blame]
// Copyright (c) 2016, 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.
library observable.src.observable_map;
import 'dart:collection';
import 'observable.dart';
import 'records.dart';
import 'to_observable.dart';
// TODO(jmesserly): this needs to be faster. We currently require multiple
// lookups per key to get the old value.
// TODO(jmesserly): this doesn't implement the precise interfaces like
// LinkedHashMap, SplayTreeMap or HashMap. However it can use them for the
// backing store.
/// Represents an observable map of model values. If any items are added,
/// removed, or replaced, then observers that are listening to [changes]
/// will be notified.
class ObservableMap<K, V> extends Observable implements Map<K, V> {
final Map<K, V> _map;
/// Creates an observable map.
ObservableMap() : _map = new HashMap<K, V>();
/// Creates a new observable map using a [LinkedHashMap].
ObservableMap.linked() : _map = new LinkedHashMap<K, V>();
/// Creates a new observable map using a [SplayTreeMap].
ObservableMap.sorted() : _map = new SplayTreeMap<K, V>();
/// Creates an observable map that contains all key value pairs of [other].
/// It will attempt to use the same backing map type if the other map is a
/// [LinkedHashMap], [SplayTreeMap], or [HashMap]. Otherwise it defaults to
/// [HashMap].
///
/// Note this will perform a shallow conversion. If you want a deep conversion
/// you should use [toObservable].
factory ObservableMap.from(Map<K, V> other) {
return new ObservableMap<K, V>.createFromType(other)..addAll(other);
}
/// Like [ObservableMap.from], but creates an empty map.
factory ObservableMap.createFromType(Map<K, V> other) {
ObservableMap<K, V> result;
if (other is SplayTreeMap) {
result = new ObservableMap<K, V>.sorted();
} else if (other is LinkedHashMap) {
result = new ObservableMap<K, V>.linked();
} else {
result = new ObservableMap<K, V>();
}
return result;
}
/// Creates a new observable map wrapping [other].
ObservableMap.spy(Map<K, V> other) : _map = other;
@override
Iterable<K> get keys => _map.keys;
@override
Iterable<V> get values => _map.values;
@override
int get length => _map.length;
@override
bool get isEmpty => length == 0;
@override
bool get isNotEmpty => !isEmpty;
@override
bool containsValue(Object value) => _map.containsValue(value);
@override
bool containsKey(Object key) => _map.containsKey(key);
@override
V operator [](Object key) => _map[key];
@override
void operator []=(K key, V value) {
if (!hasObservers) {
_map[key] = value;
return;
}
int len = _map.length;
V oldValue = _map[key];
_map[key] = value;
if (len != _map.length) {
notifyPropertyChange(#length, len, _map.length);
notifyChange(new MapChangeRecord.insert(key, value));
_notifyKeysValuesChanged();
} else if (oldValue != value) {
notifyChange(new MapChangeRecord(key, oldValue, value));
_notifyValuesChanged();
}
}
@override
void addAll(Map<K, V> other) {
other.forEach((K key, V value) {
this[key] = value;
});
}
@override
V putIfAbsent(K key, V ifAbsent()) {
int len = _map.length;
V result = _map.putIfAbsent(key, ifAbsent);
if (hasObservers && len != _map.length) {
notifyPropertyChange(#length, len, _map.length);
notifyChange(new MapChangeRecord.insert(key, result));
_notifyKeysValuesChanged();
}
return result;
}
@override
V remove(Object key) {
int len = _map.length;
V result = _map.remove(key);
if (hasObservers && len != _map.length) {
notifyChange(new MapChangeRecord.remove(key, result));
notifyPropertyChange(#length, len, _map.length);
_notifyKeysValuesChanged();
}
return result;
}
@override
void clear() {
int len = _map.length;
if (hasObservers && len > 0) {
_map.forEach((key, value) {
notifyChange(new MapChangeRecord.remove(key, value));
});
notifyPropertyChange(#length, len, 0);
_notifyKeysValuesChanged();
}
_map.clear();
}
@override
void forEach(void f(K key, V value)) => _map.forEach(f);
@override
String toString() => Maps.mapToString(this);
// Note: we don't really have a reasonable old/new value to use here.
// But this should fix "keys" and "values" in templates with minimal overhead.
void _notifyKeysValuesChanged() {
notifyChange(new PropertyChangeRecord(this, #keys, null, null));
_notifyValuesChanged();
}
void _notifyValuesChanged() {
notifyChange(new PropertyChangeRecord(this, #values, null, null));
}
}