-
Notifications
You must be signed in to change notification settings - Fork 58
Expand file tree
/
Copy pathangularLocalStorage.js
More file actions
233 lines (211 loc) · 7.76 KB
/
angularLocalStorage.js
File metadata and controls
233 lines (211 loc) · 7.76 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
/*
* Angular.js localStorage module
* https://github.com/agrublev/angularLocalStorage
*/
(function (window, angular, undefined) {
'use strict';
angular.module('angularLocalStorage', ['ngCookies']).factory('storage', ['$parse', '$cookieStore', '$window', '$log', function ($parse, $cookieStore, $window, $log) {
/**
* Global Vars
*/
var storage = (typeof $window.localStorage === 'undefined') ? undefined : $window.localStorage;
var supported = (typeof storage !== 'undefined');
var watchers = {};
if (supported) {
// When Safari (OS X or iOS) is in private browsing mode it appears as though localStorage
// is available, but trying to call .setItem throws an exception below:
// "QUOTA_EXCEEDED_ERR: DOM Exception 22: An attempt was made to add something to storage that exceeded the quota."
var testKey = '__' + Math.round(Math.random() * 1e7);
try {
localStorage.setItem(testKey, testKey);
localStorage.removeItem(testKey);
}
catch (err) {
supported = false;
}
}
var privateMethods = {
/**
* Pass any type of a string from the localStorage to be parsed so it returns a usable version (like an Object)
* @param res - a string that will be parsed for type
* @returns {*} - whatever the real type of stored value was
*/
parseValue: function (res) {
var val;
try {
val = angular.fromJson(res);
if (typeof val === 'undefined') {
val = res;
}
if (val === 'true') {
val = true;
}
if (val === 'false') {
val = false;
}
if ($window.parseFloat(val) === val && !angular.isObject(val)) {
val = $window.parseFloat(val);
}
} catch (e) {
val = res;
}
return val;
},
getWatcherId: function (scope, key) {
return scope.$id + key;
}
};
var publicMethods = {
/**
* Set - let's you set a new localStorage key pair set
* @param key - a string that will be used as the accessor for the pair
* @param value - the value of the localStorage item
* @returns {*} - will return whatever it is you've stored in the local storage
*/
set: function (key, value) {
if (!supported) {
try {
$cookieStore.put(key, value);
return value;
} catch (e) {
$log.log('Local Storage not supported, make sure you have angular-cookies enabled.');
}
}
var saver = angular.toJson(value);
storage.setItem(key, saver);
return privateMethods.parseValue(saver);
},
/**
* Get - let's you get the value of any pair you've stored
* @param key - the string that you set as accessor for the pair
* @returns {*} - Object,String,Float,Boolean depending on what you stored
*/
get: function (key) {
if (!supported) {
try {
return $cookieStore.get(key);
} catch (e) {
return null;
}
}
var item = storage.getItem(key);
return privateMethods.parseValue(item);
},
/**
* Remove - let's you nuke a value from localStorage
* @param key - the accessor value
* @returns {boolean} - if everything went as planned
*/
remove: function (key) {
if (!supported) {
try {
$cookieStore.remove(key);
return true;
} catch (e) {
return false;
}
}
storage.removeItem(key);
return true;
},
/**
* Bind - let's you directly bind a localStorage value to a $scope variable
* @param {Angular $scope} $scope - the current scope you want the variable available in
* @param {String} key - the name of the variable you are binding
* @param {Object|String} opts - (optional) custom options like default value or unique store name
* Here are the available options you can set:
* * defaultValue: the default value
* * storeName: add a custom store key value instead of using the scope variable name
* @returns {*} - returns whatever the stored value is
*/
bind: function ($scope, key, opts) {
var watcherId = privateMethods.getWatcherId($scope, key);
var defaultOpts = {
defaultValue: '',
storeName: ''
};
// Backwards compatibility with old defaultValue string
if (angular.isString(opts)) {
opts = angular.extend({}, defaultOpts, {defaultValue: opts});
} else {
// If no defined options we use defaults otherwise extend defaults
opts = (angular.isUndefined(opts)) ? defaultOpts : angular.extend(defaultOpts, opts);
}
// Set the storeName key for the localStorage entry
// use user defined in specified
var storeName = opts.storeName || key;
var scopeVal = $scope.$eval(key);
// If a value doesn't already exist store it as is
if (publicMethods.get(storeName) === null && typeof scopeVal === 'undefined') {
publicMethods.set(storeName, opts.defaultValue);
}
// assign it to the $scope value
if (typeof scopeVal === 'undefined') {
$parse(key).assign($scope, publicMethods.get(storeName));
}
// Register a listener for changes on the $scope value
// to update the localStorage value
watchers[watcherId] = $scope.$watch(key, function (val) {
if (angular.isDefined(val)) {
publicMethods.set(storeName, val);
}
}, true);
return publicMethods.get(storeName);
},
/**
* Unbind - let's you unbind a variable from localStorage while removing the value from both
* the localStorage and the local variable and sets it to null
* @param $scope - the scope the variable was initially set in
* @param key - the name of the variable you are unbinding
* @param storeName - (optional) if you used a custom storeName you will have to specify it here as well
*/
unbind: function ($scope, key, storeName) {
var watcherId = privateMethods.getWatcherId($scope, key);
storeName = storeName || key;
$parse(key).assign($scope, null);
publicMethods.remove(storeName);
// Trying to unbind a watcher if it really was here
// Maybe it should do nothing at all in this case? E.g. you can't unbind what you haven't bound yet.
if (watchers[watcherId]) {
watchers[watcherId]();
delete watchers[watcherId];
}
},
/**
* Clear All - let's you clear out ALL localStorage variables, use this carefully!
*/
clearAll: function () {
storage.clear();
},
/**
* Check if cookie fallback is active right now
*/
isCookieFallbackActive: function () {
return !supported;
},
/**
* Allows the caller to obtain all the keys that are saved in Cookies or LocalStorage. Gets all the keys
* that are saved in LocalStorage or Cookie.
*
* Uses: String.trim() - ECMAScript 1.5+
*
* @returns array
*/
getKeys: function () {
var keys = [];
if (!supported) {
var cookieArr = document.cookie.split(';');
for (var cnt = 0, cntLen = cookieArr.length; cnt < cntLen; ++cnt) {
keys.push(cookieArr[cnt].split('=')[0].trim());
}
} else {
for (var i = 0, len = localStorage.length; i < len; ++i) {
keys.push(localStorage.key(i));
}
}
return keys;
}
};
return publicMethods;
}]);
})(window, window.angular);