save prefs in bg to avoid data loss
* add `now` to simplify usage of prefs.subscribe * tweak/simplify bits by separating bg/content concerns
This commit is contained in:
		
							parent
							
								
									be47cfc471
								
							
						
					
					
						commit
						b56dacb6b2
					
				|  | @ -59,7 +59,8 @@ window.API_METHODS = Object.assign(window.API_METHODS || {}, { | ||||||
|   parseCss({code}) { |   parseCss({code}) { | ||||||
|     return backgroundWorker.parseMozFormat({code}); |     return backgroundWorker.parseMozFormat({code}); | ||||||
|   }, |   }, | ||||||
|   getPrefs: prefs.getAll, |   getPrefs: () => prefs.values, // will be deepCopy'd by invokeAPI handler
 | ||||||
|  |   setPref: (key, value) => prefs.set(key, value), | ||||||
| 
 | 
 | ||||||
|   openEditor, |   openEditor, | ||||||
| 
 | 
 | ||||||
|  | @ -218,6 +219,14 @@ function createContextMenus(ids) { | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| if (chrome.contextMenus) { | if (chrome.contextMenus) { | ||||||
|  |   // "Delete" item in context menu for browsers that don't have it
 | ||||||
|  |   if (CHROME && | ||||||
|  |       // looking at the end of UA string
 | ||||||
|  |       /(Vivaldi|Safari)\/[\d.]+$/.test(navigator.userAgent) && | ||||||
|  |       // skip forks with Flash as those are likely to have the menu e.g. CentBrowser
 | ||||||
|  |       !Array.from(navigator.plugins).some(p => p.name === 'Shockwave Flash')) { | ||||||
|  |     prefs.defaults['editor.contextDelete'] = true; | ||||||
|  |   } | ||||||
|   // circumvent the bug with disabling check marks in Chrome 62-64
 |   // circumvent the bug with disabling check marks in Chrome 62-64
 | ||||||
|   const toggleCheckmark = CHROME >= 62 && CHROME <= 64 ? |   const toggleCheckmark = CHROME >= 62 && CHROME <= 64 ? | ||||||
|     (id => chrome.contextMenus.remove(id, () => createContextMenus([id]) + ignoreChromeError())) : |     (id => chrome.contextMenus.remove(id, () => createContextMenus([id]) + ignoreChromeError())) : | ||||||
|  | @ -233,7 +242,7 @@ if (chrome.contextMenus) { | ||||||
| 
 | 
 | ||||||
|   const keys = Object.keys(contextMenus); |   const keys = Object.keys(contextMenus); | ||||||
|   prefs.subscribe(keys.filter(id => typeof prefs.defaults[id] === 'boolean'), toggleCheckmark); |   prefs.subscribe(keys.filter(id => typeof prefs.defaults[id] === 'boolean'), toggleCheckmark); | ||||||
|   prefs.subscribe(keys.filter(id => contextMenus[id].presentIf), togglePresence); |   prefs.subscribe(keys.filter(id => contextMenus[id].presentIf && id in prefs.defaults), togglePresence); | ||||||
|   createContextMenus(keys); |   createContextMenus(keys); | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										292
									
								
								js/prefs.js
									
									
									
									
									
								
							
							
						
						
									
										292
									
								
								js/prefs.js
									
									
									
									
									
								
							|  | @ -1,9 +1,11 @@ | ||||||
| /* global promisifyChrome msg API */ | /* global promisifyChrome msg API */ | ||||||
|  | /* global deepCopy deepEqual debounce */ // not used in content scripts
 | ||||||
| 'use strict'; | 'use strict'; | ||||||
| 
 | 
 | ||||||
| // Needs msg.js loaded first
 | // eslint-disable-next-line no-unused-expressions
 | ||||||
| 
 | window.INJECTED !== 1 && (() => { | ||||||
| self.prefs = self.INJECTED === 1 ? self.prefs : (() => { |   const STORAGE_KEY = 'settings'; | ||||||
|  |   const clone = msg.isBg ? deepCopy : (val => JSON.parse(JSON.stringify(val))); | ||||||
|   const defaults = { |   const defaults = { | ||||||
|     'openEditInWindow': false,      // new editor opens in a own browser window
 |     'openEditInWindow': false,      // new editor opens in a own browser window
 | ||||||
|     'openEditInWindow.popup': false, // new editor opens in a simplified browser window without omnibox
 |     'openEditInWindow.popup': false, // new editor opens in a simplified browser window without omnibox
 | ||||||
|  | @ -72,7 +74,8 @@ self.prefs = self.INJECTED === 1 ? self.prefs : (() => { | ||||||
|                                       // '' (empty string) = disabled
 |                                       // '' (empty string) = disabled
 | ||||||
|     'editor.autoCloseBrackets': true,    // auto-add a closing pair when typing an opening one of ()[]{}''""
 |     'editor.autoCloseBrackets': true,    // auto-add a closing pair when typing an opening one of ()[]{}''""
 | ||||||
|     'editor.autocompleteOnTyping': false, // show autocomplete dropdown on typing a word token
 |     'editor.autocompleteOnTyping': false, // show autocomplete dropdown on typing a word token
 | ||||||
|     'editor.contextDelete': contextDeleteMissing(), // "Delete" item in context menu
 |     // "Delete" item in context menu for browsers that don't have it
 | ||||||
|  |     'editor.contextDelete': null, | ||||||
|     'editor.selectByTokens': true, |     'editor.selectByTokens': true, | ||||||
| 
 | 
 | ||||||
|     'editor.appliesToLineWidget': true, // show applies-to line widget on the editor
 |     'editor.appliesToLineWidget': true, // show applies-to line widget on the editor
 | ||||||
|  | @ -105,189 +108,126 @@ self.prefs = self.INJECTED === 1 ? self.prefs : (() => { | ||||||
| 
 | 
 | ||||||
|     'updateInterval': 24,           // user-style automatic update interval, hours (0 = disable)
 |     'updateInterval': 24,           // user-style automatic update interval, hours (0 = disable)
 | ||||||
|   }; |   }; | ||||||
|   const values = deepCopy(defaults); |   const values = clone(defaults); | ||||||
| 
 |  | ||||||
|   const onChange = { |   const onChange = { | ||||||
|     any: new Set(), |     any: new Set(), | ||||||
|     specific: new Map(), |     specific: {}, | ||||||
|   }; |   }; | ||||||
| 
 |   if (msg.isBg) { | ||||||
|   promisifyChrome({ |     promisifyChrome({ | ||||||
|     'storage.sync': ['get', 'set'], |       'storage.sync': ['get', 'set'], | ||||||
|   }); |  | ||||||
| 
 |  | ||||||
|   const initializing = ( |  | ||||||
|     msg.isBg |  | ||||||
|       ? browser.storage.sync.get('settings').then(res => res.settings) |  | ||||||
|       : API.getPrefs() |  | ||||||
|   ).then(res => res && setAll(res, true)); |  | ||||||
| 
 |  | ||||||
|   chrome.storage.onChanged.addListener((changes, area) => { |  | ||||||
|     if (area !== 'sync' || !changes.settings || !changes.settings.newValue) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     initializing.then(() => setAll(changes.settings.newValue, true)); |  | ||||||
|   }); |  | ||||||
| 
 |  | ||||||
|   let timer; |  | ||||||
| 
 |  | ||||||
|   // coalesce multiple pref changes in broadcast
 |  | ||||||
|   // let changes = {};
 |  | ||||||
| 
 |  | ||||||
|   return { |  | ||||||
|     initializing, |  | ||||||
|     defaults, |  | ||||||
|     get(key, defaultValue) { |  | ||||||
|       if (key in values) { |  | ||||||
|         return values[key]; |  | ||||||
|       } |  | ||||||
|       if (defaultValue !== undefined) { |  | ||||||
|         return defaultValue; |  | ||||||
|       } |  | ||||||
|       if (key in defaults) { |  | ||||||
|         return defaults[key]; |  | ||||||
|       } |  | ||||||
|       console.warn("No default preference for '%s'", key); |  | ||||||
|     }, |  | ||||||
|     getAll() { |  | ||||||
|       return deepCopy(values); |  | ||||||
|     }, |  | ||||||
|     set, |  | ||||||
|     reset: key => set(key, deepCopy(defaults[key])), |  | ||||||
|     subscribe(keys, listener) { |  | ||||||
|       // keys:     string[] ids
 |  | ||||||
|       //           or a falsy value to subscribe to everything
 |  | ||||||
|       // listener: function (key, value)
 |  | ||||||
|       if (keys) { |  | ||||||
|         for (const key of keys) { |  | ||||||
|           const existing = onChange.specific.get(key); |  | ||||||
|           if (!existing) { |  | ||||||
|             onChange.specific.set(key, listener); |  | ||||||
|           } else if (existing instanceof Set) { |  | ||||||
|             existing.add(listener); |  | ||||||
|           } else { |  | ||||||
|             onChange.specific.set(key, new Set([existing, listener])); |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } else { |  | ||||||
|         onChange.any.add(listener); |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|     unsubscribe(keys, listener) { |  | ||||||
|       if (keys) { |  | ||||||
|         for (const key of keys) { |  | ||||||
|           const existing = onChange.specific.get(key); |  | ||||||
|           if (existing instanceof Set) { |  | ||||||
|             existing.delete(listener); |  | ||||||
|             if (!existing.size) { |  | ||||||
|               onChange.specific.delete(key); |  | ||||||
|             } |  | ||||||
|           } else if (existing) { |  | ||||||
|             onChange.specific.delete(key); |  | ||||||
|           } |  | ||||||
|         } |  | ||||||
|       } else { |  | ||||||
|         onChange.all.remove(listener); |  | ||||||
|       } |  | ||||||
|     }, |  | ||||||
|   }; |  | ||||||
| 
 |  | ||||||
|   function setAll(settings, synced) { |  | ||||||
|     for (const [key, value] of Object.entries(settings)) { |  | ||||||
|       set(key, value, synced); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   function set(key, value, synced = false) { |  | ||||||
|     const oldValue = values[key]; |  | ||||||
|     switch (typeof defaults[key]) { |  | ||||||
|       case typeof value: |  | ||||||
|         break; |  | ||||||
|       case 'string': |  | ||||||
|         value = String(value); |  | ||||||
|         break; |  | ||||||
|       case 'number': |  | ||||||
|         value |= 0; |  | ||||||
|         break; |  | ||||||
|       case 'boolean': |  | ||||||
|         value = value === true || value === 'true'; |  | ||||||
|         break; |  | ||||||
|     } |  | ||||||
|     if (equal(value, oldValue)) { |  | ||||||
|       return; |  | ||||||
|     } |  | ||||||
|     values[key] = value; |  | ||||||
|     emitChange(key, value); |  | ||||||
|     if (!synced && !timer) { |  | ||||||
|       timer = syncPrefsLater(); |  | ||||||
|     } |  | ||||||
|     return timer; |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   function emitChange(key, value) { |  | ||||||
|     const specific = onChange.specific.get(key); |  | ||||||
|     if (typeof specific === 'function') { |  | ||||||
|       specific(key, value); |  | ||||||
|     } else if (specific instanceof Set) { |  | ||||||
|       for (const listener of specific.values()) { |  | ||||||
|         listener(key, value); |  | ||||||
|       } |  | ||||||
|     } |  | ||||||
|     for (const listener of onChange.any.values()) { |  | ||||||
|       listener(key, value); |  | ||||||
|     } |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   function syncPrefsLater() { |  | ||||||
|     return new Promise((resolve, reject) => { |  | ||||||
|       setTimeout(() => { |  | ||||||
|         timer = null; |  | ||||||
|         browser.storage.sync.set({settings: values}) |  | ||||||
|           .then(resolve, reject); |  | ||||||
|       }); |  | ||||||
|     }); |     }); | ||||||
|   } |   } | ||||||
|  |   const initializing = ( | ||||||
|  |     msg.isBg | ||||||
|  |       ? browser.storage.sync.get(STORAGE_KEY).then(res => res[STORAGE_KEY]) | ||||||
|  |       : API.getPrefs() | ||||||
|  |   ).then(setAll); | ||||||
| 
 | 
 | ||||||
|   function equal(a, b) { |   chrome.storage.onChanged.addListener(async (changes, area) => { | ||||||
|     if (!a || !b || typeof a !== 'object' || typeof b !== 'object') { |     const data = area === 'sync' && changes[STORAGE_KEY]; | ||||||
|       return a === b; |     if (data) { | ||||||
|  |       await initializing; | ||||||
|  |       setAll(data.newValue); | ||||||
|     } |     } | ||||||
|     if (Object.keys(a).length !== Object.keys(b).length) { |   }); | ||||||
|       return false; | 
 | ||||||
|     } |   // This direct assignment allows IDEs to provide correct autocomplete for methods
 | ||||||
|     for (const k in a) { |   const prefs = window.prefs = { | ||||||
|       if (typeof a[k] === 'object') { |     initializing, | ||||||
|         if (!equal(a[k], b[k])) { |     defaults, | ||||||
|           return false; |     values, | ||||||
|  |     get(key) { | ||||||
|  |       return isKnown(key) && values[key]; | ||||||
|  |     }, | ||||||
|  |     set(key, value, isSynced) { | ||||||
|  |       if (!isKnown(key)) return; | ||||||
|  |       const oldValue = values[key]; | ||||||
|  |       const type = typeof defaults[key]; | ||||||
|  |       if (type !== typeof value) { | ||||||
|  |         if (type === 'string') value = String(value); | ||||||
|  |         if (type === 'number') value = Number(value) || 0; | ||||||
|  |         if (type === 'boolean') value = Boolean(value); | ||||||
|  |       } | ||||||
|  |       if (value !== oldValue && !deepEqual(value, oldValue)) { | ||||||
|  |         values[key] = value; | ||||||
|  |         emitChange(key, value, isSynced); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     reset(key) { | ||||||
|  |       prefs.set(key, clone(defaults[key])); | ||||||
|  |     }, | ||||||
|  |     /** | ||||||
|  |      * @param {?string|string[]} keys - pref ids or a falsy value to subscribe to everything | ||||||
|  |      * @param {function(key:string, value:any)} fn | ||||||
|  |      * @param {Object} [opts] | ||||||
|  |      * @param {boolean} [opts.now] - when truthy, the listener is called immediately: | ||||||
|  |      *   1) if `keys` is an array of keys, each `key` will be fired separately with a real `value` | ||||||
|  |      *   2) if `keys` is falsy, no key/value will be provided | ||||||
|  |      */ | ||||||
|  |     subscribe(keys, fn, {now} = {}) { | ||||||
|  |       if (keys) { | ||||||
|  |         for (const key of Array.isArray(keys) ? keys : [keys]) { | ||||||
|  |           if (!isKnown(key)) continue; | ||||||
|  |           const listeners = onChange.specific[key] || | ||||||
|  |             (onChange.specific[key] = new Set()); | ||||||
|  |           listeners.add(fn); | ||||||
|  |           if (now) fn(key, values[key]); | ||||||
|         } |         } | ||||||
|       } else if (a[k] !== b[k]) { |       } else { | ||||||
|         return false; |         onChange.any.add(fn); | ||||||
|  |         if (now) fn(); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |     unsubscribe(keys, fn) { | ||||||
|  |       if (keys) { | ||||||
|  |         for (const key of keys) { | ||||||
|  |           const listeners = onChange.specific[key]; | ||||||
|  |           if (listeners) { | ||||||
|  |             listeners.delete(fn); | ||||||
|  |             if (!listeners.size) { | ||||||
|  |               delete onChange.specific[key]; | ||||||
|  |             } | ||||||
|  |           } | ||||||
|  |         } | ||||||
|  |       } else { | ||||||
|  |         onChange.all.remove(fn); | ||||||
|  |       } | ||||||
|  |     }, | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   function isKnown(key) { | ||||||
|  |     const res = defaults.hasOwnProperty(key); | ||||||
|  |     if (!res) console.warn('Unknown preference "%s"', key); | ||||||
|  |     return res; | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   function setAll(settings) { | ||||||
|  |     for (const [key, value] of Object.entries(settings || {})) { | ||||||
|  |       prefs.set(key, value, true); | ||||||
|  |     } | ||||||
|  |   } | ||||||
|  | 
 | ||||||
|  |   function emitChange(key, value, isSynced) { | ||||||
|  |     for (const fn of onChange.specific[key] || []) { | ||||||
|  |       fn(key, value); | ||||||
|  |     } | ||||||
|  |     for (const fn of onChange.any) { | ||||||
|  |       fn(key, value); | ||||||
|  |     } | ||||||
|  |     if (!isSynced) { | ||||||
|  |       /* browser.storage is slow and can randomly lose values if the tab was closed immediately | ||||||
|  |        so we're sending the value to the background script which will save it to the storage; | ||||||
|  |        the extra bonus is that invokeAPI is immediate in extension tabs */ | ||||||
|  |       if (msg.isBg) { | ||||||
|  |         debounce(updateStorage); | ||||||
|  |       } else { | ||||||
|  |         API.setPref(key, value); | ||||||
|       } |       } | ||||||
|     } |     } | ||||||
|     return true; |  | ||||||
|   } |   } | ||||||
| 
 | 
 | ||||||
|   function contextDeleteMissing() { |   function updateStorage() { | ||||||
|     return /Chrome\/\d+/.test(navigator.userAgent) && ( |     return browser.storage.sync.set({[STORAGE_KEY]: values}); | ||||||
|       // detect browsers without Delete by looking at the end of UA string
 |  | ||||||
|       /Vivaldi\/[\d.]+$/.test(navigator.userAgent) || |  | ||||||
|       // Chrome and co.
 |  | ||||||
|       /Safari\/[\d.]+$/.test(navigator.userAgent) && |  | ||||||
|       // skip forks with Flash as those are likely to have the menu e.g. CentBrowser
 |  | ||||||
|       !Array.from(navigator.plugins).some(p => p.name === 'Shockwave Flash') |  | ||||||
|     ); |  | ||||||
|   } |  | ||||||
| 
 |  | ||||||
|   function deepCopy(obj) { |  | ||||||
|     if (!obj || typeof obj !== 'object') { |  | ||||||
|       return obj; |  | ||||||
|     } |  | ||||||
|     if (Array.isArray(obj)) { |  | ||||||
|       return obj.map(deepCopy); |  | ||||||
|     } |  | ||||||
|     return Object.keys(obj).reduce((output, key) => { |  | ||||||
|       output[key] = deepCopy(obj[key]); |  | ||||||
|       return output; |  | ||||||
|     }, {}); |  | ||||||
|   } |   } | ||||||
| })(); | })(); | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user