OLD | NEW |
---|---|
(Empty) | |
1 // Copyright (c) 2013, the Dart project authors. Please see the AUTHORS file | |
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. | |
4 | |
5 part of observe; | |
6 | |
7 /** | |
8 * Interface representing an [Observable] object that performs its own change | |
9 * notifications, and does not need to be considered by [Observable.dirtyCheck]. | |
10 */ | |
11 abstract class ChangeNotifier extends Observable { | |
Jennifer Messerly
2013/07/19 01:32:58
splitting Observable and ChangeNotifier into disti
| |
12 /** | |
13 * Notify observers of a change. | |
14 * | |
15 * For most objects [ChangeNotifierMixin.notifyPropertyChange] is more | |
16 * convenient, but collections sometimes deliver other types of changes such | |
17 * as a [ListChangeRecord]. | |
18 */ | |
19 void notifyChange(ChangeRecord record); | |
20 } | |
21 | |
22 /** | |
23 * Base class implementing [ChangeNotifier]. | |
24 * | |
25 * When a field, property, or indexable item is changed, a derived class should | |
26 * call [notifyPropertyChange]. See that method for an example. | |
27 */ | |
28 typedef ChangeNotifierBase = Object with ChangeNotifierMixin; | |
Jennifer Messerly
2013/07/19 01:32:58
I'm not sure about the split here into ChangeNotif
| |
29 | |
30 /** | |
31 * Mixin for implementing [ChangeNotifier] objects. | |
32 * | |
33 * When a field, property, or indexable item is changed, a derived class should | |
34 * call [notifyPropertyChange]. See that method for an example. | |
35 */ | |
36 abstract class ChangeNotifierMixin implements ChangeNotifier { | |
Jennifer Messerly
2013/07/19 01:32:58
this is essentially the old ObservableMixin impl
| |
37 StreamController _changes; | |
38 List<ChangeRecord> _records; | |
39 | |
40 Stream<List<ChangeRecord>> get changes { | |
41 if (_changes == null) { | |
42 _changes = new StreamController.broadcast(sync: true, | |
43 onListen: _observed, onCancel: _unobserved); | |
44 } | |
45 return _changes.stream; | |
46 } | |
47 | |
48 // TODO(jmesserly): should these be public? They're useful lifecycle methods | |
49 // for subclasses. Ideally they'd be protected. | |
50 /** | |
51 * Override this method to be called when the [changes] are first observed. | |
52 */ | |
53 void _observed() {} | |
54 | |
55 /** | |
56 * Override this method to be called when the [changes] are no longer being | |
57 * observed. | |
58 */ | |
59 void _unobserved() {} | |
60 | |
61 bool deliverChanges() { | |
62 var records = _records; | |
63 _records = null; | |
64 if (hasObservers && records != null) { | |
65 // TODO(jmesserly): make "records" immutable | |
66 _changes.add(records); | |
67 return true; | |
68 } | |
69 return false; | |
70 } | |
71 | |
72 /** | |
73 * True if this object has any observers, and should call | |
74 * [notifyPropertyChange] for changes. | |
75 */ | |
76 bool get hasObservers => _changes != null && _changes.hasListener; | |
77 | |
78 /** | |
79 * Notify that the field [name] of this object has been changed. | |
80 * | |
81 * The [oldValue] and [newValue] are also recorded. If the two values are | |
82 * identical, no change will be recorded. | |
83 * | |
84 * For convenience this returns [newValue]. This makes it easy to use in a | |
85 * setter: | |
86 * | |
87 * var _myField; | |
88 * get myField => _myField; | |
89 * set myField(value) { | |
90 * _myField = notifyPropertyChange( | |
91 * const Symbol('myField'), _myField, value); | |
92 * } | |
93 */ | |
94 // TODO(jmesserly): should this be == instead of identical, to prevent | |
95 // spurious loops? | |
96 notifyPropertyChange(Symbol field, Object oldValue, Object newValue) { | |
97 if (hasObservers && !identical(oldValue, newValue)) { | |
98 notifyChange(new PropertyChangeRecord(field)); | |
99 } | |
100 return newValue; | |
101 } | |
102 | |
103 void notifyChange(ChangeRecord record) { | |
104 if (!hasObservers) return; | |
105 | |
106 if (_records == null) { | |
107 _records = []; | |
108 runAsync(deliverChanges); | |
109 } | |
110 _records.add(record); | |
111 } | |
112 } | |
OLD | NEW |