|  | // Copyright (c) 2013, 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. | 
|  |  | 
|  | part of html; | 
|  |  | 
|  | // This code is inspired by ChangeSummary: | 
|  | // https://github.com/rafaelw/ChangeSummary/blob/master/change_summary.js | 
|  | // ...which underlies MDV. Since we don't need the functionality of | 
|  | // ChangeSummary, we just implement what we need for data bindings. | 
|  | // This allows our implementation to be much simpler. | 
|  |  | 
|  | // TODO(jmesserly): should we make these types stronger, and require | 
|  | // Observable objects? Currently, it is fine to say something like: | 
|  | //     var path = new PathObserver(123, ''); | 
|  | //     print(path.value); // "123" | 
|  | // | 
|  | // Furthermore this degenerate case is allowed: | 
|  | //     var path = new PathObserver(123, 'foo.bar.baz.qux'); | 
|  | //     print(path.value); // "null" | 
|  | // | 
|  | // Here we see that any invalid (i.e. not Observable) value will break the | 
|  | // path chain without producing an error or exception. | 
|  | // | 
|  | // Now the real question: should we do this? For the former case, the behavior | 
|  | // is correct but we could chose to handle it in the dart:html bindings layer. | 
|  | // For the latter case, it might be better to throw an error so users can find | 
|  | // the problem. | 
|  |  | 
|  |  | 
|  | /** | 
|  | * A data-bound path starting from a view-model or model object, for example | 
|  | * `foo.bar.baz`. | 
|  | * | 
|  | * When the [values] stream is being listened to, this will observe changes to | 
|  | * the object and any intermediate object along the path, and send [values] | 
|  | * accordingly. When all listeners are unregistered it will stop observing | 
|  | * the objects. | 
|  | * | 
|  | * This class is used to implement [Node.bind] and similar functionality. | 
|  | */ | 
|  | // TODO(jmesserly): find a better home for this type. | 
|  | @Experimental | 
|  | class PathObserver { | 
|  | /** The object being observed. */ | 
|  | final object; | 
|  |  | 
|  | /** The path string. */ | 
|  | final String path; | 
|  |  | 
|  | /** True if the path is valid, otherwise false. */ | 
|  | final bool _isValid; | 
|  |  | 
|  | // TODO(jmesserly): same issue here as ObservableMixin: is there an easier | 
|  | // way to get a broadcast stream? | 
|  | StreamController _values; | 
|  | Stream _valueStream; | 
|  |  | 
|  | _PropertyObserver _observer, _lastObserver; | 
|  |  | 
|  | Object _lastValue; | 
|  | bool _scheduled = false; | 
|  |  | 
|  | /** | 
|  | * Observes [path] on [object] for changes. This returns an object that can be | 
|  | * used to get the changes and get/set the value at this path. | 
|  | * See [PathObserver.values] and [PathObserver.value]. | 
|  | */ | 
|  | PathObserver(this.object, String path) | 
|  | : path = path, _isValid = _isPathValid(path) { | 
|  |  | 
|  | // TODO(jmesserly): if the path is empty, or the object is! Observable, we | 
|  | // can optimize the PathObserver to be more lightweight. | 
|  |  | 
|  | _values = new StreamController.broadcast(sync: true, | 
|  | onListen: _observe, | 
|  | onCancel: _unobserve); | 
|  |  | 
|  | if (_isValid) { | 
|  | var segments = []; | 
|  | for (var segment in path.trim().split('.')) { | 
|  | if (segment == '') continue; | 
|  | var index = int.parse(segment, onError: (_) {}); | 
|  | segments.add(index != null ? index : new Symbol(segment)); | 
|  | } | 
|  |  | 
|  | // Create the property observer linked list. | 
|  | // Note that the structure of a path can't change after it is initially | 
|  | // constructed, even though the objects along the path can change. | 
|  | for (int i = segments.length - 1; i >= 0; i--) { | 
|  | _observer = new _PropertyObserver(this, segments[i], _observer); | 
|  | if (_lastObserver == null) _lastObserver = _observer; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // TODO(jmesserly): we could try adding the first value to the stream, but | 
|  | // that delivers the first record async. | 
|  | /** | 
|  | * Listens to the stream, and invokes the [callback] immediately with the | 
|  | * current [value]. This is useful for bindings, which want to be up-to-date | 
|  | * immediately. | 
|  | */ | 
|  | StreamSubscription bindSync(void callback(value)) { | 
|  | var result = values.listen(callback); | 
|  | callback(value); | 
|  | return result; | 
|  | } | 
|  |  | 
|  | // TODO(jmesserly): should this be a change record with the old value? | 
|  | // TODO(jmesserly): should this be a broadcast stream? We only need | 
|  | // single-subscription in the bindings system, so single sub saves overhead. | 
|  | /** | 
|  | * Gets the stream of values that were observed at this path. | 
|  | * This returns a single-subscription stream. | 
|  | */ | 
|  | Stream get values => _values.stream; | 
|  |  | 
|  | /** Force synchronous delivery of [values]. */ | 
|  | void _deliverValues() { | 
|  | _scheduled = false; | 
|  |  | 
|  | var newValue = value; | 
|  | if (!identical(_lastValue, newValue)) { | 
|  | _values.add(newValue); | 
|  | _lastValue = newValue; | 
|  | } | 
|  | } | 
|  |  | 
|  | void _observe() { | 
|  | if (_observer != null) { | 
|  | _lastValue = value; | 
|  | _observer.observe(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void _unobserve() { | 
|  | if (_observer != null) _observer.unobserve(); | 
|  | } | 
|  |  | 
|  | void _notifyChange() { | 
|  | if (_scheduled) return; | 
|  | _scheduled = true; | 
|  |  | 
|  | // TODO(jmesserly): should we have a guarenteed order with respect to other | 
|  | // paths? If so, we could implement this fairly easily by sorting instances | 
|  | // of this class by birth order before delivery. | 
|  | queueChangeRecords(_deliverValues); | 
|  | } | 
|  |  | 
|  | /** Gets the last reported value at this path. */ | 
|  | get value { | 
|  | if (!_isValid) return null; | 
|  | if (_observer == null) return object; | 
|  | _observer.ensureValue(object); | 
|  | return _lastObserver.value; | 
|  | } | 
|  |  | 
|  | /** Sets the value at this path. */ | 
|  | void set value(Object value) { | 
|  | // TODO(jmesserly): throw if property cannot be set? | 
|  | // MDV seems tolerant of these error. | 
|  | if (_observer == null || !_isValid) return; | 
|  | _observer.ensureValue(object); | 
|  | var last = _lastObserver; | 
|  | if (_setObjectProperty(last._object, last._property, value)) { | 
|  | // Technically, this would get updated asynchronously via a change record. | 
|  | // However, it is nice if calling the getter will yield the same value | 
|  | // that was just set. So we use this opportunity to update our cache. | 
|  | last.value = value; | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // TODO(jmesserly): these should go away in favor of mirrors! | 
|  | _getObjectProperty(object, property) { | 
|  | if (object is List && property is int) { | 
|  | if (property >= 0 && property < object.length) { | 
|  | return object[property]; | 
|  | } else { | 
|  | return null; | 
|  | } | 
|  | } | 
|  |  | 
|  | // TODO(jmesserly): what about length? | 
|  | if (object is Map) return object[property]; | 
|  |  | 
|  | if (object is Observable) return object.getValueWorkaround(property); | 
|  |  | 
|  | return null; | 
|  | } | 
|  |  | 
|  | bool _setObjectProperty(object, property, value) { | 
|  | if (object is List && property is int) { | 
|  | object[property] = value; | 
|  | } else if (object is Map) { | 
|  | object[property] = value; | 
|  | } else if (object is Observable) { | 
|  | (object as Observable).setValueWorkaround(property, value); | 
|  | } else { | 
|  | return false; | 
|  | } | 
|  | return true; | 
|  | } | 
|  |  | 
|  |  | 
|  | class _PropertyObserver { | 
|  | final PathObserver _path; | 
|  | final _property; | 
|  | final _PropertyObserver _next; | 
|  |  | 
|  | // TODO(jmesserly): would be nice not to store both of these. | 
|  | Object _object; | 
|  | Object _value; | 
|  | StreamSubscription _sub; | 
|  |  | 
|  | _PropertyObserver(this._path, this._property, this._next); | 
|  |  | 
|  | get value => _value; | 
|  |  | 
|  | void set value(Object newValue) { | 
|  | _value = newValue; | 
|  | if (_next != null) { | 
|  | if (_sub != null) _next.unobserve(); | 
|  | _next.ensureValue(_value); | 
|  | if (_sub != null) _next.observe(); | 
|  | } | 
|  | } | 
|  |  | 
|  | void ensureValue(object) { | 
|  | // If we're observing, values should be up to date already. | 
|  | if (_sub != null) return; | 
|  |  | 
|  | _object = object; | 
|  | value = _getObjectProperty(object, _property); | 
|  | } | 
|  |  | 
|  | void observe() { | 
|  | if (_object is Observable) { | 
|  | assert(_sub == null); | 
|  | _sub = (_object as Observable).changes.listen(_onChange); | 
|  | } | 
|  | if (_next != null) _next.observe(); | 
|  | } | 
|  |  | 
|  | void unobserve() { | 
|  | if (_sub == null) return; | 
|  |  | 
|  | _sub.cancel(); | 
|  | _sub = null; | 
|  | if (_next != null) _next.unobserve(); | 
|  | } | 
|  |  | 
|  | void _onChange(List<ChangeRecord> changes) { | 
|  | for (var change in changes) { | 
|  | // TODO(jmesserly): what to do about "new Symbol" here? | 
|  | // Ideally this would only preserve names if the user has opted in to | 
|  | // them being preserved. | 
|  | // TODO(jmesserly): should we drop observable maps with String keys? | 
|  | // If so then we only need one check here. | 
|  | if (change.changes(_property)) { | 
|  | value = _getObjectProperty(_object, _property); | 
|  | _path._notifyChange(); | 
|  | return; | 
|  | } | 
|  | } | 
|  | } | 
|  | } | 
|  |  | 
|  | // From: https://github.com/rafaelw/ChangeSummary/blob/master/change_summary.js | 
|  |  | 
|  | const _pathIndentPart = r'[$a-z0-9_]+[$a-z0-9_\d]*'; | 
|  | final _pathRegExp = new RegExp('^' | 
|  | '(?:#?' + _pathIndentPart + ')?' | 
|  | '(?:' | 
|  | '(?:\\.' + _pathIndentPart + ')' | 
|  | ')*' | 
|  | r'$', caseSensitive: false); | 
|  |  | 
|  | final _spacesRegExp = new RegExp(r'\s'); | 
|  |  | 
|  | bool _isPathValid(String s) { | 
|  | s = s.replaceAll(_spacesRegExp, ''); | 
|  |  | 
|  | if (s == '') return true; | 
|  | if (s[0] == '.') return false; | 
|  | return _pathRegExp.hasMatch(s); | 
|  | } |