2019-07-02 09:44:32 +02:00
|
|
|
/*! Icinga Web 2 | (c) 2019 Icinga GmbH | GPLv2+ */
|
|
|
|
|
|
|
|
(function (Icinga, $) {
|
|
|
|
|
|
|
|
'use strict';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Icinga.Storage
|
|
|
|
*
|
|
|
|
* localStorage access
|
2019-07-04 10:21:50 +02:00
|
|
|
*
|
|
|
|
* @param {string} prefix
|
2019-07-02 09:44:32 +02:00
|
|
|
*/
|
2019-07-04 10:21:50 +02:00
|
|
|
Icinga.Storage = function(prefix) {
|
2019-07-02 09:44:32 +02:00
|
|
|
|
|
|
|
/**
|
2019-07-04 10:21:50 +02:00
|
|
|
* Prefix to use for keys
|
2019-07-02 09:44:32 +02:00
|
|
|
*
|
|
|
|
* @type {string}
|
|
|
|
*/
|
2019-07-04 10:21:50 +02:00
|
|
|
this.prefix = prefix;
|
2019-07-02 09:44:32 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Callbacks for storage events on particular keys
|
|
|
|
*
|
|
|
|
* @type {{function}}
|
|
|
|
*/
|
|
|
|
this.subscribers = {};
|
|
|
|
|
|
|
|
this.setup();
|
|
|
|
};
|
|
|
|
|
2019-07-04 10:21:50 +02:00
|
|
|
/**
|
|
|
|
* Create a new storage with `behavior.<name>` as prefix
|
|
|
|
*
|
|
|
|
* @param {string} name
|
|
|
|
*
|
|
|
|
* @returns {Icinga.Storage}
|
|
|
|
*/
|
|
|
|
Icinga.Storage.BehaviorStorage = function(name) {
|
|
|
|
return new Icinga.Storage('behavior.' + name);
|
|
|
|
};
|
|
|
|
|
2019-07-02 09:44:32 +02:00
|
|
|
Icinga.Storage.prototype = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Prefix the given key
|
|
|
|
*
|
|
|
|
* @param {string} key
|
|
|
|
* @returns {string}
|
|
|
|
*/
|
|
|
|
prefixKey: function(key) {
|
2019-07-04 10:21:50 +02:00
|
|
|
if (typeof this.prefix !== 'undefined') {
|
|
|
|
return this.prefix + '.' + key;
|
|
|
|
}
|
|
|
|
|
2019-07-02 09:44:32 +02:00
|
|
|
return key;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Store the given key-value pair
|
|
|
|
*
|
|
|
|
* @param {string} key
|
|
|
|
* @param {*} value
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
set: function(key, value) {
|
2019-07-08 13:26:32 +02:00
|
|
|
window.localStorage.setItem(this.prefixKey(key), JSON.stringify(value));
|
2019-07-02 09:44:32 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get value for the given key
|
|
|
|
*
|
|
|
|
* @param {string} key
|
|
|
|
*
|
|
|
|
* @returns {*}
|
|
|
|
*/
|
|
|
|
get: function(key) {
|
2019-07-08 13:26:32 +02:00
|
|
|
return JSON.parse(window.localStorage.getItem(this.prefixKey(key)));
|
2019-07-02 09:44:32 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove given key from storage
|
|
|
|
*
|
|
|
|
* @param {string} key
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
remove: function(key) {
|
2019-07-08 13:26:32 +02:00
|
|
|
window.localStorage.removeItem(this.prefixKey(key));
|
2019-07-02 09:44:32 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Subscribe with a callback for events on a particular key
|
|
|
|
*
|
|
|
|
* @param {string} key
|
|
|
|
* @param {function} callback
|
2019-07-08 13:34:38 +02:00
|
|
|
* @param {object} context
|
2019-07-02 09:44:32 +02:00
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
2019-07-08 13:34:38 +02:00
|
|
|
subscribe: function(key, callback, context) {
|
|
|
|
this.subscribers[this.prefixKey(key)] = [callback, context];
|
2019-07-02 09:44:32 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Pass storage events to subscribers
|
|
|
|
*
|
|
|
|
* @param {StorageEvent} event
|
|
|
|
*/
|
|
|
|
onStorage: function(event) {
|
|
|
|
if (typeof this.subscribers[event.key] !== 'undefined') {
|
2019-07-08 13:34:38 +02:00
|
|
|
var subscriber = this.subscribers[event.key];
|
|
|
|
subscriber[0].call(subscriber[1], JSON.parse(event.newValue), JSON.parse(event.oldValue), event);
|
2019-07-02 09:44:32 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Add the event listener
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
setup: function() {
|
|
|
|
window.addEventListener('storage', this.onStorage.bind(this));
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove the event listener
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
destroy: function() {
|
|
|
|
window.removeEventListener('storage', this.onStorage.bind(this));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Icinga.Storage.StorageAwareSet
|
|
|
|
*
|
|
|
|
* Emits events `StorageAwareSetDelete` and `StorageAwareSetAdd` in case an update occurs in the storage.
|
|
|
|
*
|
|
|
|
* @param {Array} values
|
|
|
|
* @constructor
|
|
|
|
*/
|
|
|
|
Icinga.Storage.StorageAwareSet = function(values) {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Storage object
|
|
|
|
*
|
|
|
|
* @type {Icinga.Storage}
|
|
|
|
*/
|
|
|
|
this.storage = undefined;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Storage key
|
|
|
|
*
|
|
|
|
* @type {string}
|
|
|
|
*/
|
|
|
|
this.key = undefined;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The internal (real) set
|
|
|
|
*
|
|
|
|
* @type {Set<*>}
|
|
|
|
*/
|
|
|
|
this.data = new Set();
|
|
|
|
|
|
|
|
// items is not passed directly because IE11 doesn't support constructor arguments
|
|
|
|
if (typeof values !== 'undefined' && !! values && values.length) {
|
|
|
|
values.forEach(function(value) {
|
|
|
|
this.data.add(value);
|
|
|
|
}, this);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Create a new StorageAwareSet for the given storage and key
|
|
|
|
*
|
|
|
|
* @param {Icinga.Storage} storage
|
|
|
|
* @param {string} key
|
|
|
|
*
|
|
|
|
* @returns {Icinga.Storage.StorageAwareSet}
|
|
|
|
*/
|
|
|
|
Icinga.Storage.StorageAwareSet.withStorage = function(storage, key) {
|
|
|
|
return (new Icinga.Storage.StorageAwareSet(storage.get(key)).setStorage(storage, key));
|
|
|
|
};
|
|
|
|
|
|
|
|
Icinga.Storage.StorageAwareSet.prototype = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Bind this set to the given storage and key
|
|
|
|
*
|
|
|
|
* @param {Icinga.Storage} storage
|
|
|
|
* @param {string} key
|
|
|
|
*
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
|
|
|
setStorage: function(storage, key) {
|
|
|
|
this.storage = storage;
|
|
|
|
this.key = key;
|
|
|
|
|
2019-07-08 13:34:38 +02:00
|
|
|
storage.subscribe(key, this.onChange, this);
|
2019-07-02 09:44:32 +02:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a boolean indicating this set got a storage
|
|
|
|
*
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
hasStorage: function() {
|
|
|
|
return typeof this.storage !== 'undefined' && typeof this.key !== 'undefined';
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Update the set
|
|
|
|
*
|
|
|
|
* @param {Array} newValue
|
|
|
|
*/
|
2019-07-08 13:34:38 +02:00
|
|
|
onChange: function(newValue) {
|
2019-07-02 09:44:32 +02:00
|
|
|
// Check for deletions first
|
|
|
|
this.values().forEach(function (value) {
|
|
|
|
if (newValue.indexOf(value) < 0) {
|
|
|
|
this.data.delete(value);
|
|
|
|
$(window).trigger('StorageAwareSetDelete', value);
|
|
|
|
}
|
|
|
|
}, this);
|
|
|
|
|
|
|
|
// Now check for new entries
|
|
|
|
newValue.forEach(function(value) {
|
|
|
|
if (! this.data.has(value)) {
|
|
|
|
this.data.add(value);
|
|
|
|
$(window).trigger('StorageAwareSetAdd', value);
|
|
|
|
}
|
|
|
|
}, this);
|
|
|
|
},
|
|
|
|
|
2019-07-03 15:56:08 +02:00
|
|
|
/**
|
|
|
|
* Register an event handler to handle storage updates
|
|
|
|
*
|
|
|
|
* Available events are: add, delete
|
|
|
|
*
|
|
|
|
* @param {string} event
|
|
|
|
* @param {object} data
|
|
|
|
* @param {function} handler
|
|
|
|
*
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
|
|
|
on: function(event, data, handler) {
|
|
|
|
$(window).on(
|
|
|
|
'StorageAwareSet' + event.charAt(0).toUpperCase() + event.slice(1),
|
|
|
|
data,
|
|
|
|
handler
|
|
|
|
);
|
|
|
|
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
2019-07-02 09:44:32 +02:00
|
|
|
/**
|
|
|
|
* Return the number of (unique) elements in the set
|
|
|
|
*
|
|
|
|
* @returns {number}
|
|
|
|
*/
|
|
|
|
get size() {
|
|
|
|
return this.data.size;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Append the given value to the end of the set
|
|
|
|
*
|
|
|
|
* @param value
|
|
|
|
*
|
|
|
|
* @returns {this}
|
|
|
|
*/
|
|
|
|
add: function(value) {
|
|
|
|
this.data.add(value);
|
|
|
|
|
|
|
|
if (this.hasStorage()) {
|
|
|
|
this.storage.set(this.key, this.values());
|
|
|
|
}
|
|
|
|
|
|
|
|
return this;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove all elements from the set
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
clear: function() {
|
|
|
|
if (this.hasStorage()) {
|
|
|
|
this.storage.remove(this.key);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this.data.clear();
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Remove the given value from the set
|
|
|
|
*
|
|
|
|
* @param value
|
|
|
|
*
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
delete: function(value) {
|
|
|
|
var retVal = this.data.delete(value);
|
|
|
|
|
|
|
|
if (this.hasStorage()) {
|
|
|
|
this.storage.set(this.key, this.values());
|
|
|
|
}
|
|
|
|
|
|
|
|
return retVal;
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an iterable of [v,v] pairs for every value v in the set.
|
|
|
|
*
|
|
|
|
* @returns {IterableIterator<[*, *]>}
|
|
|
|
*/
|
|
|
|
entries: function() {
|
|
|
|
return this.data.entries();
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Execute a provided function once for each value in the Set object, in insertion order.
|
|
|
|
*
|
|
|
|
* @param callback
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
forEach: function(callback) {
|
|
|
|
return this.data.forEach(callback);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Return a boolean indicating whether an element with the specified value exists in a Set object or not.
|
|
|
|
*
|
|
|
|
* @param value
|
|
|
|
*
|
|
|
|
* @returns {boolean}
|
|
|
|
*/
|
|
|
|
has: function(value) {
|
|
|
|
return this.data.has(value);
|
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns an array of values in the set.
|
|
|
|
*
|
|
|
|
* @returns {Array}
|
|
|
|
*/
|
|
|
|
values: function() {
|
|
|
|
var list = [];
|
|
|
|
|
|
|
|
if (this.size > 0) {
|
|
|
|
// .forEach() is used because IE11 doesn't support .values()
|
|
|
|
this.forEach(function(value) {
|
|
|
|
list.push(value);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
}(Icinga, jQuery));
|