Source: Persist.js

  1. /* eslint-disable no-use-before-define */
  2. import {
  3. reset,
  4. setStateByKey,
  5. getStateByKey,
  6. getStorage,
  7. } from "./storageManager";
  8. import PersistQuotaExceededError from "./PersistQuotaExceededError";
  9. import {isNeeded, getUrl, getStorageKey, getNavigationType, isQuotaExceededError} from "./utils";
  10. import {console, window} from "./browser";
  11. import {TYPE_BACK_FORWARD, TYPE_NAVIGATE, CONST_PERSIST_STATE, CONST_DEPTHS, CONST_LAST_URL} from "./consts";
  12. let currentUrl = "";
  13. function execRec(obj, path, func) {
  14. let _obj = obj;
  15. if (!_obj) {
  16. _obj = isNaN(path[0]) ? {} : [];
  17. }
  18. const head = path.shift();
  19. if (path.length === 0) {
  20. if (_obj instanceof Array && isNaN(head)) {
  21. console.warn("Don't use key string on array");
  22. }
  23. func(_obj, head);
  24. return _obj;
  25. }
  26. _obj[head] = execRec(_obj[head], path, func);
  27. return _obj;
  28. }
  29. function setPersistState(key, value) {
  30. try {
  31. setStateByKey(CONST_PERSIST_STATE, key, value);
  32. } catch (e) {
  33. if (catchQuotaExceededError(e, CONST_PERSIST_STATE, value)) {
  34. if (key === CONST_LAST_URL) {
  35. setPersistState(key, value);
  36. } else if (key === CONST_DEPTHS) {
  37. setPersistState(key, value && value.slice(1));
  38. }
  39. }
  40. }
  41. }
  42. function getPersistState(key) {
  43. return getStateByKey(CONST_PERSIST_STATE, key);
  44. }
  45. function replaceDepth() {
  46. const url = getUrl();
  47. if (currentUrl === url) {
  48. return;
  49. }
  50. const prevUrl = currentUrl;
  51. try {
  52. currentUrl = url;
  53. const depths = getPersistState(CONST_DEPTHS) || [];
  54. // remove prev url
  55. const prevIndex = depths.indexOf(prevUrl);
  56. if (prevIndex >= 0) {
  57. depths.splice(prevIndex, 1);
  58. reset(getStorageKey(prevUrl));
  59. }
  60. // remove next url info
  61. const currentIndex = depths.indexOf(url);
  62. if (currentIndex >= 0) {
  63. depths.splice(currentIndex, 1);
  64. reset(getStorageKey(url));
  65. }
  66. depths.push(url);
  67. setPersistState(CONST_DEPTHS, depths);
  68. setPersistState(CONST_LAST_URL, url);
  69. } catch (e) {
  70. // revert currentUrl
  71. currentUrl = prevUrl;
  72. throw e;
  73. }
  74. }
  75. function updateDepth(type = 0) {
  76. const url = getUrl();
  77. if (currentUrl === url) {
  78. return;
  79. }
  80. // url is not the same for the first time, pushState, or replaceState.
  81. const prevUrl = currentUrl;
  82. try {
  83. currentUrl = url;
  84. const depths = getPersistState(CONST_DEPTHS) || [];
  85. if (type === TYPE_BACK_FORWARD) {
  86. // Change current url only
  87. const currentIndex = depths.indexOf(url);
  88. ~currentIndex && setPersistState(CONST_LAST_URL, url);
  89. } else {
  90. const prevLastUrl = getPersistState(CONST_LAST_URL);
  91. reset(getStorageKey(url));
  92. if (type === TYPE_NAVIGATE && url !== prevLastUrl) {
  93. // Remove all url lists with higher index than current index
  94. const prevLastIndex = depths.indexOf(prevLastUrl);
  95. const removedList = depths.splice(prevLastIndex + 1, depths.length);
  96. removedList.forEach(removedUrl => {
  97. reset(getStorageKey(removedUrl));
  98. });
  99. // If the type is NAVIGATE and there is information about current url, delete it.
  100. const currentIndex = depths.indexOf(url);
  101. ~currentIndex && depths.splice(currentIndex, 1);
  102. }
  103. // Add depth for new address.
  104. if (depths.indexOf(url) < 0) {
  105. depths.push(url);
  106. }
  107. setPersistState(CONST_DEPTHS, depths);
  108. setPersistState(CONST_LAST_URL, url);
  109. }
  110. } catch (e) {
  111. // revert currentUrl
  112. currentUrl = prevUrl;
  113. throw e;
  114. }
  115. }
  116. function catchQuotaExceededError(e, key, value) {
  117. if (clearFirst()) {
  118. return true;
  119. } else if (isQuotaExceededError(e)) {
  120. throw new PersistQuotaExceededError(key, value ? JSON.stringify(value) : "");
  121. } else {
  122. throw e;
  123. }
  124. }
  125. function clearFirst() {
  126. const depths = getPersistState(CONST_DEPTHS) || [];
  127. const removed = depths.splice(0, 1);
  128. if (!removed.length) {
  129. // There is an error because there is no depth to add data.
  130. return false;
  131. }
  132. const removedUrl = removed[0];
  133. reset(getStorageKey(removedUrl));
  134. if (currentUrl === removedUrl) {
  135. currentUrl = "";
  136. setPersistState(CONST_LAST_URL, "");
  137. if (!depths.length) {
  138. // I tried to add myself, but it didn't add up, so I got an error.
  139. return false;
  140. }
  141. }
  142. setPersistState(CONST_DEPTHS, depths);
  143. // Clear the previous record and try to add data again.
  144. return true;
  145. }
  146. function clear() {
  147. const depths = getPersistState(CONST_DEPTHS) || [];
  148. depths.forEach(url => {
  149. reset(getStorageKey(url));
  150. });
  151. reset(CONST_PERSIST_STATE);
  152. currentUrl = "";
  153. }
  154. /**
  155. * Get or store the current state of the web page using JSON.
  156. * @ko 웹 페이지의 현재 상태를 JSON 형식으로 저장하거나 읽는다.
  157. * @alias eg.Persist
  158. *
  159. * @support {"ie": "9+", "ch" : "latest", "ff" : "latest", "sf" : "latest" , "edge" : "latest", "ios" : "7+", "an" : "2.3+ (except 3.x)"}
  160. */
  161. class Persist {
  162. static VERSION = "#__VERSION__#";
  163. static StorageManager = {
  164. reset,
  165. setStateByKey,
  166. getStateByKey,
  167. getStorage,
  168. };
  169. /**
  170. * @static
  171. * Clear all information in Persist
  172. */
  173. static clear() {
  174. clear();
  175. }
  176. /**
  177. * @static
  178. * Return whether you need "Persist" module by checking the bfCache support of the current browser
  179. * @return {Boolean}
  180. */
  181. static isNeeded() {
  182. return isNeeded;
  183. }
  184. /**
  185. * Constructor
  186. * @param {String} key The key of the state information to be stored <ko>저장할 상태 정보의 키</ko>
  187. **/
  188. constructor(key) {
  189. this.key = key || "";
  190. }
  191. /**
  192. * Read value
  193. * @param {String?} path target path
  194. * @return {String|Number|Boolean|Object|Array}
  195. */
  196. get(path) {
  197. // update url for pushState, replaceState
  198. updateDepth(TYPE_NAVIGATE);
  199. // find path
  200. const urlKey = getStorageKey(getUrl());
  201. const globalState = getStateByKey(urlKey, this.key);
  202. if (!path || path.length === 0) {
  203. return globalState;
  204. }
  205. const pathToken = path.split(".");
  206. let currentItem = globalState;
  207. let isTargetExist = true;
  208. for (let i = 0; i < pathToken.length; i++) {
  209. if (!currentItem) {
  210. isTargetExist = false;
  211. break;
  212. }
  213. currentItem = currentItem[pathToken[i]];
  214. }
  215. if (!isTargetExist || currentItem == null) {
  216. return null;
  217. }
  218. return currentItem;
  219. }
  220. /**
  221. * Save value
  222. * @param {String} path target path
  223. * @param {String|Number|Boolean|Object|Array} value value to save
  224. * @return {Persist}
  225. */
  226. set(path, value) {
  227. // update url for pushState, replaceState
  228. updateDepth(TYPE_NAVIGATE);
  229. // find path
  230. const key = this.key;
  231. const urlKey = getStorageKey(getUrl());
  232. const globalState = getStateByKey(urlKey, key);
  233. try {
  234. if (path.length === 0) {
  235. setStateByKey(urlKey, key, value);
  236. } else {
  237. const allValue = execRec(globalState, path.split("."), (obj, head) => {
  238. obj[head] = value;
  239. });
  240. setStateByKey(
  241. urlKey,
  242. key,
  243. allValue
  244. );
  245. }
  246. } catch (e) {
  247. if (catchQuotaExceededError(e, urlKey, value)) {
  248. this.set(path, value);
  249. }
  250. }
  251. return this;
  252. }
  253. /**
  254. * Remove value
  255. * @param {String} path target path
  256. * @return {Persist}
  257. */
  258. remove(path) {
  259. // update url for pushState, replaceState
  260. updateDepth(TYPE_NAVIGATE);
  261. // find path
  262. const key = this.key;
  263. const urlKey = getStorageKey(getUrl());
  264. const globalState = getStateByKey(urlKey, key);
  265. try {
  266. if (path.length === 0) {
  267. setStateByKey(urlKey, key, null);
  268. } else {
  269. const value = execRec(globalState, path.split("."), (obj, head) => {
  270. if (typeof obj === "object") {
  271. delete obj[head];
  272. }
  273. });
  274. setStateByKey(
  275. urlKey,
  276. key,
  277. value
  278. );
  279. }
  280. } catch (e) {
  281. if (catchQuotaExceededError(e)) {
  282. this.remove(path);
  283. }
  284. }
  285. return this;
  286. }
  287. }
  288. if ("onpopstate" in window) {
  289. window.addEventListener("popstate", () => {
  290. // popstate event occurs when backward or forward
  291. try {
  292. updateDepth(TYPE_BACK_FORWARD);
  293. } catch (e) {
  294. // Global function calls prevent errors.
  295. if (!isQuotaExceededError(e)) {
  296. throw e;
  297. }
  298. }
  299. });
  300. }
  301. // If navigation's type is not TYPE_BACK_FORWARD, delete information about current url.
  302. try {
  303. updateDepth(getNavigationType());
  304. } catch (e) {
  305. // Global function calls prevent errors.
  306. if (!isQuotaExceededError(e)) {
  307. throw e;
  308. }
  309. }
  310. export {
  311. updateDepth,
  312. replaceDepth,
  313. };
  314. export default Persist;
comments powered by Disqus