You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
366 lines
9.4 KiB
366 lines
9.4 KiB
/*
|
|
YUI 3.17.2 (build 9c3c78e)
|
|
Copyright 2014 Yahoo! Inc. All rights reserved.
|
|
Licensed under the BSD License.
|
|
http://yuilibrary.com/license/
|
|
*/
|
|
|
|
YUI.add('model-sync-local', function (Y, NAME) {
|
|
|
|
/*
|
|
An extension which provides a sync implementation through locally stored
|
|
key value pairs, either through the HTML localStorage API or falling back
|
|
onto an in-memory cache, that can be mixed into a Model or ModelList subclass.
|
|
|
|
@module app
|
|
@submodule model-sync-local
|
|
@since 3.13.0
|
|
**/
|
|
|
|
/**
|
|
An extension which provides a sync implementation through locally stored
|
|
key value pairs, either through the HTML localStorage API or falling back
|
|
onto an in-memory cache, that can be mixed into a Model or ModelList subclass.
|
|
|
|
A group of Models/ModelLists is serialized in localStorage by either its
|
|
class name, or a specified 'root' that is provided.
|
|
|
|
var User = Y.Base.create('user', Y.Model, [Y.ModelSync.Local], {
|
|
root: 'user'
|
|
});
|
|
|
|
var Users = Y.Base.create('users', Y.ModelList, [Y.ModelSync.Local], {
|
|
model: User,
|
|
});
|
|
|
|
@class ModelSync.Local
|
|
@extensionfor Model
|
|
@extensionfor ModelList
|
|
@since 3.13.0
|
|
**/
|
|
function LocalSync() {}
|
|
|
|
/**
|
|
Properties that shouldn't be turned into ad-hoc attributes when passed to a
|
|
Model or ModelList constructor.
|
|
|
|
@property _NON_ATTRS_CFG
|
|
@type Array
|
|
@default ['root']
|
|
@static
|
|
@protected
|
|
@since 3.13.0
|
|
**/
|
|
LocalSync._NON_ATTRS_CFG = ['root'];
|
|
|
|
/**
|
|
Feature testing for `localStorage` availability.
|
|
Will return falsey for browsers with `localStorage`, but that don't
|
|
actually work, such as iOS Safari in private browsing mode.
|
|
|
|
@property _hasLocalStorage
|
|
@type Boolean
|
|
@private
|
|
**/
|
|
LocalSync._hasLocalStorage = (function () {
|
|
var LS = Y.config.win.localStorage,
|
|
test = Y.guid();
|
|
|
|
try {
|
|
LS.setItem(test, test);
|
|
LS.removeItem(test);
|
|
return true;
|
|
} catch (e) {
|
|
return false;
|
|
}
|
|
})(),
|
|
|
|
/**
|
|
Object of key/value pairs to fall back on when localStorage is not available.
|
|
|
|
@property _data
|
|
@type Object
|
|
@private
|
|
**/
|
|
|
|
LocalSync._data = LocalSync._data || {};
|
|
|
|
/**
|
|
Cache to quickly access a specific object with a given ID.
|
|
|
|
@property _store
|
|
@type Array
|
|
@private
|
|
**/
|
|
|
|
LocalSync._store = LocalSync._store || {};
|
|
|
|
LocalSync.prototype = {
|
|
|
|
// -- Public Methods -------------------------------------------------------
|
|
|
|
/**
|
|
Root used as the key inside of localStorage and/or the in-memory store.
|
|
|
|
@property root
|
|
@type String
|
|
@default ""
|
|
@since 3.13.0
|
|
**/
|
|
root: '',
|
|
|
|
/**
|
|
Shortcut for access to localStorage.
|
|
|
|
@property storage
|
|
@type Storage
|
|
@default null
|
|
@since 3.13.0
|
|
**/
|
|
storage: null,
|
|
|
|
// -- Lifecycle Methods -----------------------------------------------------
|
|
initializer: function (config) {
|
|
var store, data;
|
|
|
|
config || (config = {});
|
|
|
|
if ('root' in config) {
|
|
this.root = config.root || '';
|
|
}
|
|
|
|
// This is checking to see if the sync layer is being applied to
|
|
// a ModelList, and if so, is looking for a `root` property on its
|
|
// Model's prototype instead.
|
|
if (!this.root && this.model && this.model.prototype.root) {
|
|
this.root = this.model.prototype.root;
|
|
}
|
|
|
|
if (LocalSync._hasLocalStorage) {
|
|
this.storage = Y.config.win.localStorage;
|
|
store = this.storage.getItem(this.root);
|
|
} else {
|
|
}
|
|
|
|
// Pull in existing data from localStorage, if possible.
|
|
// Otherwise, see if there's existing data on the local cache.
|
|
if (store) {
|
|
LocalSync._store[this.root] = store.split('|') || [];
|
|
|
|
Y.Array.each(LocalSync._store[this.root], function (id) {
|
|
LocalSync._data[id] = Y.JSON.parse(this.storage.getItem(id));
|
|
}, this);
|
|
} else {
|
|
LocalSync._store[this.root] || (LocalSync._store[this.root] = []);
|
|
}
|
|
},
|
|
|
|
// -- Public Methods -----------------------------------------------------------
|
|
|
|
/**
|
|
Creates a synchronization layer with the localStorage API, if available.
|
|
Otherwise, falls back to a in-memory data store.
|
|
|
|
This method is called internally by load(), save(), and destroy().
|
|
|
|
@method sync
|
|
@param {String} action Sync action to perform. May be one of the following:
|
|
|
|
* **create**: Store a newly-created model for the first time.
|
|
* **read** : Load an existing model.
|
|
* **update**: Update an existing model.
|
|
* **delete**: Delete an existing model.
|
|
|
|
@param {Object} [options] Sync options
|
|
@param {Function} [callback] Called when the sync operation finishes.
|
|
@param {Error|null} callback.err If an error occurred, this parameter will
|
|
contain the error. If the sync operation succeeded, _err_ will be
|
|
falsey.
|
|
@param {Any} [callback.response] The response from our sync. This value will
|
|
be passed to the parse() method, which is expected to parse it and
|
|
return an attribute hash.
|
|
**/
|
|
sync: function (action, options, callback) {
|
|
options || (options = {});
|
|
var response, errorInfo;
|
|
|
|
try {
|
|
switch (action) {
|
|
case 'read':
|
|
if (this._isYUIModelList) {
|
|
response = this._index(options);
|
|
} else {
|
|
response = this._show(options);
|
|
}
|
|
break;
|
|
case 'create':
|
|
response = this._create(options);
|
|
break;
|
|
case 'update':
|
|
response = this._update(options);
|
|
break;
|
|
case 'delete':
|
|
response = this._destroy(options);
|
|
break;
|
|
}
|
|
} catch (error) {
|
|
errorInfo = error.message;
|
|
}
|
|
|
|
if (response) {
|
|
callback(null, response);
|
|
} else if (errorInfo) {
|
|
callback(errorInfo);
|
|
} else {
|
|
callback("Data not found in LocalStorage");
|
|
}
|
|
},
|
|
|
|
/**
|
|
Generate a random GUID for our Models. This can be overriden if you have
|
|
another method of generating different IDs.
|
|
|
|
@method generateID
|
|
@protected
|
|
@param {String} pre Optional GUID prefix
|
|
**/
|
|
generateID: function (pre) {
|
|
return Y.guid(pre + '_');
|
|
},
|
|
|
|
// -- Protected Methods ----------------------------------------------------
|
|
|
|
/**
|
|
Sync method correlating to the "read" operation, for a Model List
|
|
|
|
@method _index
|
|
@return {Object[]} Array of objects found for that root key
|
|
@protected
|
|
@since 3.13.0
|
|
**/
|
|
_index: function () {
|
|
var store = LocalSync._store[this.root],
|
|
data = Y.Array.map(store, function (id) {
|
|
return LocalSync._data[id];
|
|
});
|
|
|
|
return data;
|
|
},
|
|
|
|
/**
|
|
Sync method correlating to the "read" operation, for a Model
|
|
|
|
@method _show
|
|
@return {Object} Object found for that root key and model ID
|
|
@protected
|
|
@since 3.13.0
|
|
**/
|
|
_show: function () {
|
|
return LocalSync._data[this.get('id')] || null;
|
|
},
|
|
|
|
/**
|
|
Sync method correlating to the "create" operation
|
|
|
|
@method _show
|
|
@return {Object} The new object created.
|
|
@protected
|
|
@since 3.13.0
|
|
**/
|
|
_create: function () {
|
|
var hash = this.toJSON();
|
|
|
|
hash.id = this.generateID(this.root);
|
|
|
|
LocalSync._data[hash.id] = hash;
|
|
if (this.storage) {
|
|
this.storage.setItem(hash.id, Y.JSON.stringify(hash));
|
|
}
|
|
|
|
LocalSync._store[this.root].push(hash.id);
|
|
|
|
this._save();
|
|
return hash;
|
|
},
|
|
|
|
/**
|
|
Sync method correlating to the "update" operation
|
|
|
|
@method _update
|
|
@return {Object} The updated object.
|
|
@protected
|
|
@since 3.13.0
|
|
**/
|
|
_update: function () {
|
|
var hash = this.toJSON(),
|
|
id = this.get('id');
|
|
|
|
LocalSync._data[id] = hash;
|
|
|
|
if (this.storage) {
|
|
this.storage.setItem(id, hash);
|
|
}
|
|
|
|
if (Y.Array.indexOf(LocalSync._store[this.root], id) === -1) {
|
|
LocalSync._store[this.root].push(id);
|
|
}
|
|
|
|
this._save();
|
|
|
|
return hash;
|
|
},
|
|
|
|
/**
|
|
Sync method correlating to the "delete" operation. Deletes the data
|
|
from the in-memory object, and saves into localStorage if available.
|
|
|
|
@method _destroy
|
|
@protected
|
|
@since 3.13.0
|
|
**/
|
|
_destroy: function () {
|
|
var id = this.get('id'),
|
|
storage = this.storage;
|
|
|
|
if (!LocalSync._data[id]) {
|
|
return;
|
|
}
|
|
|
|
delete LocalSync._data[id];
|
|
|
|
if (storage) {
|
|
storage.removeItem(id);
|
|
}
|
|
|
|
LocalSync._store[this.root] = Y.Array.filter(LocalSync._store[this.root], function (item) {
|
|
return item.id != id;
|
|
});
|
|
|
|
this._save();
|
|
return this.toJSON();
|
|
},
|
|
|
|
/**
|
|
Saves the current in-memory store into a localStorage key/value pair
|
|
if localStorage is available; otherwise, does nothing.
|
|
|
|
@method _save
|
|
@protected
|
|
@since 3.13.0
|
|
**/
|
|
_save: function () {
|
|
if (LocalSync._hasLocalStorage && this.storage) {
|
|
this.storage.setItem(
|
|
this.root,
|
|
LocalSync._store[this.root].join('|')
|
|
);
|
|
}
|
|
}
|
|
};
|
|
|
|
// -- Namespace ---------------------------------------------------------------
|
|
|
|
Y.namespace('ModelSync').Local = LocalSync;
|
|
|
|
|
|
}, '3.17.2', {"requires": ["model", "json-stringify"]});
|
|
|