Compare commits
	
		
			3 Commits
		
	
	
		
			master
			...
			insertcss2
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 7d094847f6 | ||
|  | 62053316a2 | ||
|  | aff4707bf0 | 
|  | @ -12,10 +12,11 @@ var browserCommands, contextMenus; | |||
| chrome.runtime.onMessage.addListener(onRuntimeMessage); | ||||
| 
 | ||||
| { | ||||
|   const listener = | ||||
|     URLS.chromeProtectsNTP | ||||
|       ? webNavigationListenerChrome | ||||
|       : webNavigationListener; | ||||
|   const [listener] = [ | ||||
|     [webNavigationListenerChrome, CHROME], | ||||
|     [webNavigationListenerFF, FIREFOX], | ||||
|     [webNavigationListener, true], | ||||
|   ].find(([, selected]) => selected); | ||||
| 
 | ||||
|   chrome.webNavigation.onBeforeNavigate.addListener(data => | ||||
|     listener(null, data)); | ||||
|  | @ -44,7 +45,6 @@ if (chrome.contextMenus) { | |||
|   chrome.contextMenus.onClicked.addListener((info, tab) => | ||||
|     contextMenus[info.menuItemId].click(info, tab)); | ||||
| } | ||||
| 
 | ||||
| if (chrome.commands) { | ||||
|   // Not available in Firefox - https://bugzilla.mozilla.org/show_bug.cgi?id=1240350
 | ||||
|   chrome.commands.onCommand.addListener(command => browserCommands[command]()); | ||||
|  | @ -81,6 +81,24 @@ prefs.subscribe(['iconset'], () => updateIcon({id: undefined}, {})); | |||
|         browserUIlanguage: chrome.i18n.getUILanguage(), | ||||
|       }); | ||||
|     } | ||||
|     if (!FIREFOX && chrome.declarativeContent) { | ||||
|       chrome.declarativeContent.onPageChanged.removeRules(null, () => { | ||||
|         chrome.declarativeContent.onPageChanged.addRules([{ | ||||
|           conditions: [ | ||||
|             new chrome.declarativeContent.PageStateMatcher({ | ||||
|               pageUrl: {urlContains: ':'}, | ||||
|             }) | ||||
|           ], | ||||
|           actions: [ | ||||
|             new chrome.declarativeContent.RequestContentScript({ | ||||
|               js: ['/content/apply.js'], | ||||
|               allFrames: true, | ||||
|               matchAboutBlank: true, | ||||
|             }), | ||||
|           ], | ||||
|         }]); | ||||
|       }); | ||||
|     } | ||||
|   }; | ||||
|   // bind for 60 seconds max and auto-unbind if it's a normal run
 | ||||
|   chrome.runtime.onInstalled.addListener(onInstall); | ||||
|  | @ -175,9 +193,37 @@ window.addEventListener('storageReady', function _() { | |||
| 
 | ||||
|   updateIcon({id: undefined}, {}); | ||||
| 
 | ||||
|   if (FIREFOX) { | ||||
|     queryTabs().then(tabs => | ||||
|       tabs.forEach(tab => { | ||||
|         if (!tab.width) { | ||||
|           // skip lazy-loaded tabs (width = 0) that seem to start loading on message
 | ||||
|           return; | ||||
|         } | ||||
|         const tabId = tab.id; | ||||
|         const frameUrls = {0: tab.url}; | ||||
|         styleViaAPI.allFrameUrls.set(tabId, frameUrls); | ||||
|         chrome.webNavigation.getAllFrames({tabId}, frames => frames && | ||||
|           frames.forEach(({frameId, parentFrameId, url}) => { | ||||
|             if (frameId) { | ||||
|               frameUrls[frameId] = url === 'about:blank' ? frameUrls[parentFrameId] : url; | ||||
|             } | ||||
|           })); | ||||
|       })); | ||||
|     return; | ||||
|   } | ||||
| 
 | ||||
|   const NTP = 'chrome://newtab/'; | ||||
|   const ALL_URLS = '<all_urls>'; | ||||
|   const contentScripts = chrome.runtime.getManifest().content_scripts; | ||||
|   contentScripts.push({ | ||||
|     js: ['content/apply.js'], | ||||
|     matches: ['<all_urls>'], | ||||
|     run_at: 'document_start', | ||||
|     match_about_blank: true, | ||||
|     all_frames: true | ||||
|   }); | ||||
| 
 | ||||
|   // expand * as .*?
 | ||||
|   const wildcardAsRegExp = (s, flags) => new RegExp( | ||||
|       s.replace(/[{}()[\]/\\.+?^$:=!|]/g, '\\$&') | ||||
|  | @ -209,13 +255,9 @@ window.addEventListener('storageReady', function _() { | |||
|   }; | ||||
| 
 | ||||
|   queryTabs().then(tabs => | ||||
|     tabs.forEach(tab => { | ||||
|       // skip lazy-loaded aka unloaded tabs that seem to start loading on message in FF
 | ||||
|       if (!FIREFOX || tab.width) { | ||||
|         contentScripts.forEach(cs => | ||||
|           setTimeout(pingCS, 0, cs, tab)); | ||||
|       } | ||||
|     })); | ||||
|     tabs.forEach(tab => tab.width && | ||||
|       contentScripts.forEach(cs => | ||||
|         setTimeout(pingCS, 0, cs, tab)))); | ||||
| }); | ||||
| 
 | ||||
| // *************************************************************************
 | ||||
|  | @ -243,32 +285,60 @@ function webNavigationListener(method, {url, tabId, frameId}) { | |||
| 
 | ||||
| 
 | ||||
| function webNavigationListenerChrome(method, data) { | ||||
|   // Chrome 61.0.3161+ doesn't run content scripts on NTP
 | ||||
|   if ( | ||||
|     !data.url.startsWith('https://www.google.') || | ||||
|     !data.url.includes('/_/chrome/newtab?') | ||||
|   ) { | ||||
|   const {tabId, frameId, url} = data; | ||||
|   if (url.startsWith('https://www.google.') && url.includes('/_/chrome/newtab?')) { | ||||
|     // Chrome 61.0.3161+ doesn't run content scripts on NTP
 | ||||
|     getTab(tabId).then(tab => { | ||||
|       data.url = tab.url === 'chrome://newtab/' ? tab.url : url; | ||||
|       webNavigationListener(method, data); | ||||
|     }); | ||||
|   } else { | ||||
|     webNavigationListener(method, data); | ||||
|     // chrome.declarativeContent doesn't inject scripts in about:blank iframes
 | ||||
|     if (method && frameId && url === 'about:blank') { | ||||
|       chrome.tabs.executeScript(tabId, { | ||||
|         file: '/content/apply.js', | ||||
|         runAt: 'document_start', | ||||
|         matchAboutBlank: true, | ||||
|         frameId, | ||||
|       }, ignoreChromeError); | ||||
|     } | ||||
|   } | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| function webNavigationListenerFF(method, data) { | ||||
|   const {tabId, frameId, url} = data; | ||||
|   //console.log(method, data);
 | ||||
|   if (frameId === 0 || url !== 'about:blank') { | ||||
|     if ((!method || method === 'styleApply') && | ||||
|         styleViaAPI.getFrameUrl(tabId, frameId) !== url) { | ||||
|       styleViaAPI.cache.delete(tabId); | ||||
|     } | ||||
|     styleViaAPI.setFrameUrl(tabId, frameId, url); | ||||
|     webNavigationListener(method, data); | ||||
|     return; | ||||
|   } | ||||
|   getTab(data.tabId).then(tab => { | ||||
|     if (tab.url === 'chrome://newtab/') { | ||||
|       data.url = tab.url; | ||||
|     } | ||||
|     webNavigationListener(method, data); | ||||
|   }); | ||||
|   //const frames = styleViaAPI.allFrameUrls.get(tabId);
 | ||||
|   //if (Object.keys(frames).length === 1) {
 | ||||
|   //  frames[frameId] = frames['0'];
 | ||||
|   //  webNavigationListener(method, data);
 | ||||
|   //  return;
 | ||||
|   //}
 | ||||
|   //chrome.webNavigation.getFrame({tabId, frameId}, info => {
 | ||||
|   //  const hasParent = !chrome.runtime.lastError && info.parentFrameId >= 0;
 | ||||
|   //  frames[frameId] = hasParent ? frames[info.parentFrameId] : url;
 | ||||
|   //  webNavigationListener(method, data);
 | ||||
|   //});
 | ||||
| } | ||||
| 
 | ||||
| 
 | ||||
| function webNavUsercssInstallerFF(data) { | ||||
|   const {tabId} = data; | ||||
|   Promise.all([ | ||||
|     sendMessage({tabId, method: 'ping'}), | ||||
|     // we need tab index to open the installer next to the original one
 | ||||
|     // and also to skip the double-invocation in FF which assigns tab url later
 | ||||
|     getTab(tabId), | ||||
|   ]).then(([pong, tab]) => { | ||||
|     if (pong !== true && tab.url !== 'about:blank') { | ||||
|   // we need tab index to open the installer next to the original one
 | ||||
|   // and also to skip the double-invocation in FF which assigns tab url later
 | ||||
|   getTab(tabId).then(tab => { | ||||
|     if (tab.url !== 'about:blank') { | ||||
|       usercssHelper.openInstallPage(tab, {direct: true}); | ||||
|     } | ||||
|   }); | ||||
|  | @ -367,10 +437,6 @@ function onRuntimeMessage(request, sender, sendResponseInternal) { | |||
|         .catch(() => sendResponse(false)); | ||||
|       return KEEP_CHANNEL_OPEN; | ||||
| 
 | ||||
|     case 'styleViaAPI': | ||||
|       styleViaAPI(request, sender); | ||||
|       return; | ||||
| 
 | ||||
|     case 'download': | ||||
|       download(request.url) | ||||
|         .then(sendResponse) | ||||
|  |  | |||
|  | @ -1,230 +1,220 @@ | |||
| /* global getStyles */ | ||||
| 'use strict'; | ||||
| 
 | ||||
| const styleViaAPI = !CHROME && (() => { | ||||
| // eslint-disable-next-line no-var
 | ||||
| var styleViaAPI = !CHROME && | ||||
| (() => { | ||||
|   const ACTIONS = { | ||||
|     styleApply, | ||||
|     styleDeleted, | ||||
|     styleUpdated, | ||||
|     styleAdded, | ||||
|     styleReplaceAll, | ||||
|     styleReplaceAll: styleApply, | ||||
|     prefChanged, | ||||
|     ping, | ||||
|   }; | ||||
|   const NOP = Promise.resolve(new Error('NOP')); | ||||
|   const onError = () => {}; | ||||
|   const PONG = Promise.resolve(true); | ||||
|   const onError = () => NOP; | ||||
| 
 | ||||
|   /* <tabId>: Object | ||||
|        <frameId>: Object | ||||
|          url: String, non-enumerable | ||||
|          <styleId>: Array of strings | ||||
|            section code */ | ||||
|   const cache = new Map(); | ||||
|   const allFrameUrls = new Map(); | ||||
| 
 | ||||
|   let observingTabs = false; | ||||
|   chrome.tabs.onRemoved.addListener(onTabRemoved); | ||||
|   chrome.tabs.onReplaced.addListener(onTabReplaced); | ||||
| 
 | ||||
|   return (request, sender) => { | ||||
|     const action = ACTIONS[request.action]; | ||||
|     return !action ? NOP : | ||||
|       action(request, sender) | ||||
|         .catch(onError) | ||||
|         .then(maybeToggleObserver); | ||||
|   return { | ||||
|     process, | ||||
|     getFrameUrl, | ||||
|     setFrameUrl, | ||||
|     allFrameUrls, | ||||
|     cache, | ||||
|   }; | ||||
| 
 | ||||
|   function styleApply({id = null, ignoreUrlCheck}, {tab, frameId, url}) { | ||||
|     if (prefs.get('disableAll')) { | ||||
|   //region public methods
 | ||||
| 
 | ||||
|   function process(request, sender) { | ||||
|     console.log(request.action || request.method, request.prefs || request.styles || request.style, sender.tab, sender.frameId); | ||||
|     const action = ACTIONS[request.action || request.method]; | ||||
|     if (!action) { | ||||
|       return NOP; | ||||
|     } | ||||
|     const {tabFrames, frameStyles} = getCachedData(tab.id, frameId); | ||||
|     if (id === null && !ignoreUrlCheck && frameStyles.url === url) { | ||||
|       return NOP; | ||||
|     const {tab} = sender; | ||||
|     if (!isNaN(sender.frameId)) { | ||||
|       const result = action(request, sender); | ||||
|       return result ? result.catch(onError) : NOP; | ||||
|     } | ||||
|     return browser.webNavigation.getAllFrames({tabId: tab.id}).then(frames => | ||||
|       Promise.all((frames || []).map(({frameId}) => | ||||
|         (action(request, {tab, frameId}) || NOP).catch(onError))) | ||||
|     ).catch(onError); | ||||
|   } | ||||
| 
 | ||||
|   function getFrameUrl(tabId, frameId = 0) { | ||||
|     const frameUrls = allFrameUrls.get(tabId); | ||||
|     return frameUrls && frameUrls[frameId] || ''; | ||||
|   } | ||||
| 
 | ||||
|   function setFrameUrl(tabId, frameId, url) { | ||||
|     const frameUrls = allFrameUrls.get(tabId); | ||||
|     if (frameUrls) { | ||||
|       frameUrls[frameId] = url; | ||||
|     } else { | ||||
|       allFrameUrls.set(tabId, {[frameId]: url}); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   //endregion
 | ||||
|   //region actions
 | ||||
| 
 | ||||
|   function styleApply({styles, disableAll}, sender) { | ||||
|     if (disableAll) { | ||||
|       return; | ||||
|     } | ||||
|     const {tab: {id: tabId}, frameId, url} = sender; | ||||
|     if (!styles || styles === 'DIY') { | ||||
|       return requestStyles({matchUrl: url || getFrameUrl(tabId, frameId)}, sender); | ||||
|     } | ||||
|     const {tabFrames, frameStyles} = getCachedData(tabId, frameId); | ||||
|     const newSorted = getSortedById(styles); | ||||
|     if (!sameArrays(frameStyles, newSorted, sameArrays)) { | ||||
|       tabFrames[frameId] = newSorted; | ||||
|       cache.set(tabId, tabFrames); | ||||
|       return replaceCSS(tabId, frameId, frameStyles, newSorted); | ||||
|     } | ||||
|     return getStyles({id, matchUrl: url, enabled: true, asHash: true}).then(styles => { | ||||
|       const tasks = []; | ||||
|       for (const styleId in styles) { | ||||
|         if (isNaN(parseInt(styleId))) { | ||||
|           continue; | ||||
|         } | ||||
|         // shallow-extract code from the sections array in order to reuse references
 | ||||
|         // in other places whereas the combined string gets garbage-collected
 | ||||
|         const styleSections = styles[styleId].map(section => section.code); | ||||
|         const code = styleSections.join('\n'); | ||||
|         if (!code) { | ||||
|           delete frameStyles[styleId]; | ||||
|           continue; | ||||
|         } | ||||
|         if (code === (frameStyles[styleId] || []).join('\n')) { | ||||
|           continue; | ||||
|         } | ||||
|         frameStyles[styleId] = styleSections; | ||||
|         tasks.push( | ||||
|           browser.tabs.insertCSS(tab.id, { | ||||
|             code, | ||||
|             frameId, | ||||
|             runAt: 'document_start', | ||||
|             matchAboutBlank: true, | ||||
|           }).catch(onError)); | ||||
|       } | ||||
|       if (!removeFrameIfEmpty(tab.id, frameId, tabFrames, frameStyles)) { | ||||
|         Object.defineProperty(frameStyles, 'url', {value: url, configurable: true}); | ||||
|         tabFrames[frameId] = frameStyles; | ||||
|         cache.set(tab.id, tabFrames); | ||||
|       } | ||||
|       return Promise.all(tasks); | ||||
|     }); | ||||
|   } | ||||
| 
 | ||||
|   function styleDeleted({id}, {tab, frameId}) { | ||||
|     const {tabFrames, frameStyles, styleSections} = getCachedData(tab.id, frameId, id); | ||||
|     const code = styleSections.join('\n'); | ||||
|     if (code && !duplicateCodeExists({frameStyles, id, code})) { | ||||
|       delete frameStyles[id]; | ||||
|       removeFrameIfEmpty(tab.id, frameId, tabFrames, frameStyles); | ||||
|       return removeCSS(tab.id, frameId, code); | ||||
|     } else { | ||||
|       return NOP; | ||||
|     const {frameStyles} = getCachedData(tab.id, frameId); | ||||
|     const index = frameStyles.findIndex(item => item.id === id); | ||||
|     if (index >= 0) { | ||||
|       const oldStyles = frameStyles.slice(); | ||||
|       frameStyles.splice(index, 1); | ||||
|       return replaceCSS(tab.id, frameId, oldStyles, frameStyles); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   function styleUpdated({style}, sender) { | ||||
|     if (!style.enabled) { | ||||
|       return styleDeleted(style, sender); | ||||
|     } | ||||
|     const {tab, frameId} = sender; | ||||
|     const {frameStyles, styleSections} = getCachedData(tab.id, frameId, style.id); | ||||
|     const code = styleSections.join('\n'); | ||||
|     return styleApply(style, sender).then(code && (() => { | ||||
|       if (!duplicateCodeExists({frameStyles, code, id: null})) { | ||||
|         return removeCSS(tab.id, frameId, code); | ||||
|       } | ||||
|     })); | ||||
|     return (style.enabled ? styleApply : styleDeleted)(style, sender); | ||||
|   } | ||||
| 
 | ||||
|   function styleAdded({style}, sender) { | ||||
|     return style.enabled ? styleApply(style, sender) : NOP; | ||||
|   } | ||||
| 
 | ||||
|   function styleReplaceAll(request, sender) { | ||||
|     const {tab, frameId} = sender; | ||||
|     const oldStylesCode = getFrameStylesJoined(sender); | ||||
|     return styleApply({ignoreUrlCheck: true}, sender).then(() => { | ||||
|       const newStylesCode = getFrameStylesJoined(sender); | ||||
|       const tasks = oldStylesCode | ||||
|         .filter(code => !newStylesCode.includes(code)) | ||||
|         .map(code => removeCSS(tab.id, frameId, code)); | ||||
|       return Promise.all(tasks); | ||||
|     }); | ||||
|   function styleAdded({style: {enabled}}, sender) { | ||||
|     return enabled && styleApply({}, sender); | ||||
|   } | ||||
| 
 | ||||
|   function prefChanged({prefs}, sender) { | ||||
|     if ('disableAll' in prefs) { | ||||
|       if (!prefs.disableAll) { | ||||
|         return styleApply({}, sender); | ||||
|       } | ||||
|       disableAll(prefs.disableAll, sender); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   function ping() { | ||||
|     return PONG; | ||||
|   } | ||||
| 
 | ||||
|   //endregion
 | ||||
|   //region action helpers
 | ||||
| 
 | ||||
|   function disableAll(state, sender) { | ||||
|     if (state) { | ||||
|       const {tab, frameId} = sender; | ||||
|       const {tabFrames, frameStyles} = getCachedData(tab.id, frameId); | ||||
|       if (isEmpty(frameStyles)) { | ||||
|         return NOP; | ||||
|       } | ||||
|       removeFrameIfEmpty(tab.id, frameId, tabFrames, {}); | ||||
|       const tasks = Object.keys(frameStyles) | ||||
|         .map(id => removeCSS(tab.id, frameId, frameStyles[id].join('\n'))); | ||||
|       return Promise.all(tasks); | ||||
|     } else { | ||||
|       return NOP; | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   /* utilities */ | ||||
| 
 | ||||
|   function maybeToggleObserver() { | ||||
|     let method; | ||||
|     if (!observingTabs && cache.size) { | ||||
|       method = 'addListener'; | ||||
|     } else if (observingTabs && !cache.size) { | ||||
|       method = 'removeListener'; | ||||
|     } else { | ||||
|       return; | ||||
|     } | ||||
|     observingTabs = !observingTabs; | ||||
|     chrome.webNavigation.onCommitted[method](onNavigationCommitted); | ||||
|     chrome.tabs.onRemoved[method](onTabRemoved); | ||||
|     chrome.tabs.onReplaced[method](onTabReplaced); | ||||
|   } | ||||
| 
 | ||||
|   function onNavigationCommitted({tabId, frameId}) { | ||||
|     if (frameId === 0) { | ||||
|       onTabRemoved(tabId); | ||||
|       return; | ||||
|     } | ||||
|     const tabFrames = cache.get(tabId); | ||||
|     if (frameId in tabFrames) { | ||||
|       delete tabFrames[frameId]; | ||||
|       if (isEmpty(tabFrames)) { | ||||
|         onTabRemoved(tabId); | ||||
|       } | ||||
|       return removeCSS(tab.id, frameId, frameStyles); | ||||
|     } else { | ||||
|       return styleApply({}, sender); | ||||
|     } | ||||
|   } | ||||
| 
 | ||||
|   //endregion
 | ||||
|   //region observer
 | ||||
| 
 | ||||
|   function onTabRemoved(tabId) { | ||||
|     cache.delete(tabId); | ||||
|     maybeToggleObserver(); | ||||
|   } | ||||
| 
 | ||||
|   function onTabReplaced(addedTabId, removedTabId) { | ||||
|     onTabRemoved(removedTabId); | ||||
|     cache.delete(removedTabId); | ||||
|   } | ||||
| 
 | ||||
|   function removeFrameIfEmpty(tabId, frameId, tabFrames, frameStyles) { | ||||
|     if (isEmpty(frameStyles)) { | ||||
|       delete tabFrames[frameId]; | ||||
|       if (isEmpty(tabFrames)) { | ||||
|         cache.delete(tabId); | ||||
|   //endregion
 | ||||
|   //region browser API
 | ||||
| 
 | ||||
|   function replaceCSS(tabId, frameId, oldStyles, newStyles) { | ||||
|     console.log.apply(null, arguments); | ||||
|     return insertCSS(tabId, frameId, newStyles).then(() => | ||||
|       removeCSS(tabId, frameId, oldStyles)); | ||||
|   } | ||||
| 
 | ||||
|   function insertCSS(tabId, frameId, frameStyles) { | ||||
|     const code = getFrameCode(frameStyles); | ||||
|     return !code ? NOP : | ||||
|       browser.tabs.insertCSS(tabId, { | ||||
|         code, | ||||
|         frameId, | ||||
|         runAt: 'document_start', | ||||
|         matchAboutBlank: true, | ||||
|       }).catch(onError); | ||||
|   } | ||||
| 
 | ||||
|   function removeCSS(tabId, frameId, frameStyles) { | ||||
|     const code = getFrameCode(frameStyles); | ||||
|     return !code ? NOP : | ||||
|       browser.tabs.removeCSS(tabId, { | ||||
|         code, | ||||
|         frameId, | ||||
|         matchAboutBlank: true | ||||
|       }).catch(onError); | ||||
|   } | ||||
| 
 | ||||
|   //endregion
 | ||||
|   //region utilities
 | ||||
| 
 | ||||
|   function requestStyles(options, sender) { | ||||
|     options.matchUrl = options.matchUrl || sender.url; | ||||
|     options.enabled = true; | ||||
|     options.asHash = true; | ||||
|     return getStyles(options).then(styles => | ||||
|       styleApply({styles}, sender)); | ||||
|   } | ||||
| 
 | ||||
|   function getSortedById(styleHash) { | ||||
|     const styles = []; | ||||
|     let needsSorting = false; | ||||
|     let prevKey = -1; | ||||
|     for (let k in styleHash) { | ||||
|       k = parseInt(k); | ||||
|       if (!isNaN(k)) { | ||||
|         const sections = styleHash[k].map(({code}) => code); | ||||
|         styles.push(sections); | ||||
|         defineProperty(sections, 'id', k); | ||||
|         needsSorting |= k < prevKey; | ||||
|         prevKey = k; | ||||
|       } | ||||
|       return true; | ||||
|     } | ||||
|     return needsSorting ? styles.sort((a, b) => a.id - b.id) : styles; | ||||
|   } | ||||
| 
 | ||||
|   function getCachedData(tabId, frameId, styleId) { | ||||
|     const tabFrames = cache.get(tabId) || {}; | ||||
|     const frameStyles = tabFrames[frameId] || {}; | ||||
|     const styleSections = styleId && frameStyles[styleId] || []; | ||||
|     const frameStyles = tabFrames[frameId] || []; | ||||
|     const styleSections = styleId && frameStyles.find(s => s.id === styleId) || []; | ||||
|     return {tabFrames, frameStyles, styleSections}; | ||||
|   } | ||||
| 
 | ||||
|   function getFrameStylesJoined({ | ||||
|     tab, | ||||
|     frameId, | ||||
|     frameStyles = getCachedData(tab.id, frameId).frameStyles, | ||||
|   }) { | ||||
|     return Object.keys(frameStyles).map(id => frameStyles[id].join('\n')); | ||||
|   function getFrameCode(frameStyles) { | ||||
|     // we cache a shallow copy of code from the sections array in order to reuse references
 | ||||
|     // in other places whereas the combined string gets garbage-collected
 | ||||
|     return typeof frameStyles === 'string' ? frameStyles : [].concat(...frameStyles).join('\n'); | ||||
|   } | ||||
| 
 | ||||
|   function duplicateCodeExists({ | ||||
|     tab, | ||||
|     frameId, | ||||
|     frameStyles = getCachedData(tab.id, frameId).frameStyles, | ||||
|     frameStylesCode = {}, | ||||
|     id, | ||||
|     code = frameStylesCode[id] || frameStyles[id].join('\n'), | ||||
|   }) { | ||||
|     id = String(id); | ||||
|     for (const styleId in frameStyles) { | ||||
|       if (id !== styleId && | ||||
|           code === (frameStylesCode[styleId] || frameStyles[styleId].join('\n'))) { | ||||
|         return true; | ||||
|       } | ||||
|     } | ||||
|   function defineProperty(obj, name, value) { | ||||
|     return Object.defineProperty(obj, name, {value, configurable: true}); | ||||
|   } | ||||
| 
 | ||||
|   function removeCSS(tabId, frameId, code) { | ||||
|     return browser.tabs.removeCSS(tabId, {frameId, code, matchAboutBlank: true}) | ||||
|       .catch(onError); | ||||
|   function sameArrays(a, b, fn) { | ||||
|     return a.length === b.length && a.every((el, i) => fn ? fn(el, b[i]) : el === b[i]); | ||||
|   } | ||||
| 
 | ||||
|   function isEmpty(obj) { | ||||
|     for (const k in obj) { | ||||
|       return false; | ||||
|     } | ||||
|     return true; | ||||
|   } | ||||
|   //endregion
 | ||||
| })(); | ||||
|  |  | |||
|  | @ -23,10 +23,6 @@ | |||
|   } | ||||
| 
 | ||||
|   function requestStyles(options, callback = applyStyles) { | ||||
|     if (!chrome.app && document instanceof XMLDocument) { | ||||
|       chrome.runtime.sendMessage({method: 'styleViaAPI', action: 'styleApply'}); | ||||
|       return; | ||||
|     } | ||||
|     var matchUrl = location.href; | ||||
|     if (!matchUrl.match(/^(http|file|chrome|ftp)/)) { | ||||
|       // dynamic about: and javascript: iframes don't have an URL yet
 | ||||
|  | @ -63,17 +59,6 @@ | |||
|       return; | ||||
|     } | ||||
| 
 | ||||
|     if (!chrome.app && document instanceof XMLDocument && request.method !== 'ping') { | ||||
|       request.action = request.method; | ||||
|       request.method = 'styleViaAPI'; | ||||
|       request.styles = null; | ||||
|       if (request.style) { | ||||
|         request.style.sections = null; | ||||
|       } | ||||
|       chrome.runtime.sendMessage(request); | ||||
|       return; | ||||
|     } | ||||
| 
 | ||||
|     switch (request.method) { | ||||
|       case 'styleDeleted': | ||||
|         removeStyle(request); | ||||
|  |  | |||
|  | @ -150,6 +150,13 @@ function sendMessage(msg, callback) { | |||
|     - enabled by passing a second param | ||||
|   */ | ||||
|   const {tabId, frameId} = msg; | ||||
|   if (tabId >= 0 && FIREFOX) { | ||||
|     // FF: reroute all tabs messages to styleViaAPI
 | ||||
|     const msgForBG = BG === window ? msg : BG.deepCopy(msg); | ||||
|     const sender = {tab: {id: tabId}, frameId}; | ||||
|     const task = BG.styleViaAPI.process(msgForBG, sender); | ||||
|     return callback ? task.then(callback) : task; | ||||
|   } | ||||
|   const fn = tabId >= 0 ? chrome.tabs.sendMessage : chrome.runtime.sendMessage; | ||||
|   const args = tabId >= 0 ? [tabId, msg, {frameId}] : [msg]; | ||||
|   if (callback) { | ||||
|  |  | |||
|  | @ -16,6 +16,7 @@ | |||
|     "webNavigation", | ||||
|     "contextMenus", | ||||
|     "storage", | ||||
|     "declarativeContent", | ||||
|     "<all_urls>" | ||||
|   ], | ||||
|   "background": { | ||||
|  | @ -43,13 +44,6 @@ | |||
|     } | ||||
|   }, | ||||
|   "content_scripts": [ | ||||
|     { | ||||
|       "matches": ["<all_urls>"], | ||||
|       "run_at": "document_start", | ||||
|       "all_frames": true, | ||||
|       "match_about_blank": true, | ||||
|       "js": ["content/apply.js"] | ||||
|     }, | ||||
|     { | ||||
|       "matches": ["http://userstyles.org/*", "https://userstyles.org/*"], | ||||
|       "run_at": "document_start", | ||||
|  |  | |||
		Loading…
	
		Reference in New Issue
	
	Block a user