mirror of
https://github.com/gristlabs/grist-core.git
synced 2024-10-27 20:44:07 +00:00
75 lines
2.0 KiB
JavaScript
75 lines
2.0 KiB
JavaScript
|
var _ = require('underscore');
|
||
|
var ko = require('knockout');
|
||
|
var dispose = require('./dispose');
|
||
|
|
||
|
/**
|
||
|
* An ObservableSet keeps track of a set of values whose membership is controlled by a boolean
|
||
|
* observable.
|
||
|
* @property {ko.observable<Number>} count: Count of items that are currently included.
|
||
|
*/
|
||
|
function ObservableSet() {
|
||
|
this._items = {};
|
||
|
this.count = ko.observable(0);
|
||
|
}
|
||
|
dispose.makeDisposable(ObservableSet);
|
||
|
|
||
|
/**
|
||
|
* Adds an item to keep track of. The value is added to the set whenever isIncluded observable is
|
||
|
* true. To stop keeping track of this item, call dispose() on the returned object.
|
||
|
*
|
||
|
* @param {ko.observable<Boolean>} isIncluded: observable for whether to include the value.
|
||
|
* @param {Object} value: Arbitrary value. May be omitted if you only care about the count.
|
||
|
* @return {Object} Object with dispose() method, which can be called to unsubscribe from
|
||
|
* isIncluded, and remove the value from the set.
|
||
|
*/
|
||
|
ObservableSet.prototype.add = function(isIncluded, value) {
|
||
|
var uniqueKey = _.uniqueId();
|
||
|
var sub = this.autoDispose(isIncluded.subscribe(function(include) {
|
||
|
if (include) {
|
||
|
this._add(uniqueKey, value);
|
||
|
} else {
|
||
|
this._remove(uniqueKey);
|
||
|
}
|
||
|
}, this));
|
||
|
|
||
|
if (isIncluded.peek()) {
|
||
|
this._add(uniqueKey, value);
|
||
|
}
|
||
|
|
||
|
return {
|
||
|
dispose: function() {
|
||
|
this._remove(uniqueKey);
|
||
|
this.disposeDiscard(sub);
|
||
|
}.bind(this)
|
||
|
};
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Returns an array of all the values that are currently included in the set.
|
||
|
*/
|
||
|
ObservableSet.prototype.all = function() {
|
||
|
return _.values(this._items);
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Internal helper to add a value to the set.
|
||
|
*/
|
||
|
ObservableSet.prototype._add = function(key, value) {
|
||
|
if (!this._items.hasOwnProperty(key)) {
|
||
|
this._items[key] = value;
|
||
|
this.count(this.count() + 1);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
/**
|
||
|
* Internal helper to remove a value from the set.
|
||
|
*/
|
||
|
ObservableSet.prototype._remove = function(key) {
|
||
|
if (this._items.hasOwnProperty(key)) {
|
||
|
delete this._items[key];
|
||
|
this.count(this.count() - 1);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
module.exports = ObservableSet;
|