5bc4206593
A few additional checks were needed to circumvent FF bugs so openURL was refactored to keep the code readable
137 lines
3.8 KiB
JavaScript
137 lines
3.8 KiB
JavaScript
/* global usercss saveStyle getStyles chromeLocal */
|
|
'use strict';
|
|
|
|
// eslint-disable-next-line no-var
|
|
var usercssHelper = (() => {
|
|
|
|
const TEMP_CODE_PREFIX = 'tempUsercssCode';
|
|
const TEMP_CODE_CLEANUP_DELAY = 60e3;
|
|
let tempCodeLastWriteDate = 0;
|
|
if (FIREFOX) {
|
|
// the temp code is created on direct installation of usercss URLs in FF
|
|
// and can be left behind in case the install page didn't open in time before
|
|
// the extension was updated/reloaded/disabled or the browser was closed
|
|
setTimeout(function poll() {
|
|
if (Date.now() - tempCodeLastWriteDate < TEMP_CODE_CLEANUP_DELAY) {
|
|
setTimeout(poll, TEMP_CODE_CLEANUP_DELAY);
|
|
return;
|
|
}
|
|
chrome.storage.local.get(null, storage => {
|
|
const leftovers = [];
|
|
for (const key in storage) {
|
|
if (key.startsWith(TEMP_CODE_PREFIX)) {
|
|
leftovers.push(key);
|
|
}
|
|
}
|
|
if (leftovers.length) {
|
|
chrome.storage.local.remove(leftovers);
|
|
}
|
|
});
|
|
}, TEMP_CODE_CLEANUP_DELAY);
|
|
}
|
|
|
|
function buildMeta(style) {
|
|
if (style.usercssData) {
|
|
return Promise.resolve(style);
|
|
}
|
|
try {
|
|
const {sourceCode} = style;
|
|
// allow sourceCode to be normalized
|
|
delete style.sourceCode;
|
|
return Promise.resolve(Object.assign(usercss.buildMeta(sourceCode), style));
|
|
} catch (e) {
|
|
return Promise.reject(e);
|
|
}
|
|
}
|
|
|
|
function buildCode(style) {
|
|
return usercss.buildCode(style);
|
|
}
|
|
|
|
function wrapReject(pending) {
|
|
return pending
|
|
.catch(err => new Error(Array.isArray(err) ? err.join('\n') : err.message || String(err)));
|
|
}
|
|
|
|
// Parse the source and find the duplication
|
|
function build({sourceCode, checkDup = false}, noReject) {
|
|
const pending = buildMeta({sourceCode})
|
|
.then(style => Promise.all([
|
|
buildCode(style),
|
|
checkDup && findDup(style)
|
|
]))
|
|
.then(([style, dup]) => ({style, dup}));
|
|
|
|
return noReject ? wrapReject(pending) : pending;
|
|
}
|
|
|
|
function save(style, noReject) {
|
|
const pending = buildMeta(style)
|
|
.then(assignVars)
|
|
.then(buildCode)
|
|
.then(saveStyle);
|
|
|
|
return noReject ? wrapReject(pending) : pending;
|
|
|
|
function assignVars(style) {
|
|
if (style.reason === 'config' && style.id) {
|
|
return style;
|
|
}
|
|
return findDup(style).then(dup => {
|
|
if (dup) {
|
|
style.id = dup.id;
|
|
if (style.reason !== 'config') {
|
|
// preserve style.vars during update
|
|
usercss.assignVars(style, dup);
|
|
}
|
|
}
|
|
return style;
|
|
});
|
|
}
|
|
}
|
|
|
|
function findDup(style) {
|
|
if (style.id) {
|
|
return getStyles({id: style.id}).then(s => s[0]);
|
|
}
|
|
return getStyles().then(styles =>
|
|
styles.find(target => {
|
|
if (!target.usercssData) {
|
|
return false;
|
|
}
|
|
return target.usercssData.name === style.usercssData.name &&
|
|
target.usercssData.namespace === style.usercssData.namespace;
|
|
})
|
|
);
|
|
}
|
|
|
|
function openInstallPage(tab, {url = tab.url, direct, downloaded} = {}) {
|
|
if (direct && !downloaded) {
|
|
prefetchCodeForInstallation(tab.id, url);
|
|
}
|
|
return wrapReject(openURL({
|
|
url: '/install-usercss.html' +
|
|
'?updateUrl=' + encodeURIComponent(url) +
|
|
'&tabId=' + tab.id +
|
|
(direct ? '&direct=yes' : ''),
|
|
index: tab.index + 1,
|
|
openerTabId: tab.id,
|
|
currentWindow: null,
|
|
}));
|
|
}
|
|
|
|
function prefetchCodeForInstallation(tabId, url) {
|
|
const key = TEMP_CODE_PREFIX + tabId;
|
|
tempCodeLastWriteDate = Date.now();
|
|
Promise.all([
|
|
download(url),
|
|
chromeLocal.setValue(key, {loading: true}),
|
|
]).then(([code]) => {
|
|
chromeLocal.setValue(key, code);
|
|
setTimeout(() => chromeLocal.remove(key), TEMP_CODE_CLEANUP_DELAY);
|
|
});
|
|
}
|
|
|
|
return {build, save, findDup, openInstallPage};
|
|
})();
|