2018-01-01 17:02:49 +00:00
|
|
|
/* global API_METHODS usercss saveStyle getStyles chromeLocal cachedStyles */
|
2017-09-18 03:34:12 +00:00
|
|
|
'use strict';
|
|
|
|
|
2018-01-01 17:02:49 +00:00
|
|
|
(() => {
|
|
|
|
|
2018-07-09 06:08:13 +00:00
|
|
|
API_METHODS.saveUsercss = style => save(style, false);
|
2018-01-07 13:36:30 +00:00
|
|
|
API_METHODS.saveUsercssUnsafe = style => save(style, true);
|
2018-01-01 17:02:49 +00:00
|
|
|
API_METHODS.buildUsercss = build;
|
|
|
|
API_METHODS.installUsercss = install;
|
2018-01-10 18:56:14 +00:00
|
|
|
API_METHODS.parseUsercss = parse;
|
|
|
|
API_METHODS.findUsercss = find;
|
2017-12-10 07:04:13 +00:00
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2017-09-18 03:34:12 +00:00
|
|
|
function buildMeta(style) {
|
|
|
|
if (style.usercssData) {
|
|
|
|
return Promise.resolve(style);
|
|
|
|
}
|
2018-09-25 13:11:54 +00:00
|
|
|
|
|
|
|
// allow sourceCode to be normalized
|
|
|
|
const {sourceCode} = style;
|
|
|
|
delete style.sourceCode;
|
|
|
|
|
|
|
|
return usercss.buildMeta(sourceCode)
|
|
|
|
.then(newStyle => Object.assign(newStyle, style));
|
2017-09-18 03:34:12 +00:00
|
|
|
}
|
|
|
|
|
2018-01-10 18:56:14 +00:00
|
|
|
function assignVars(style) {
|
|
|
|
if (style.reason === 'config' && style.id) {
|
|
|
|
return style;
|
|
|
|
}
|
|
|
|
const dup = find(style);
|
|
|
|
if (dup) {
|
|
|
|
style.id = dup.id;
|
|
|
|
if (style.reason !== 'config') {
|
|
|
|
// preserve style.vars during update
|
2018-09-25 13:11:54 +00:00
|
|
|
return usercss.assignVars(style, dup)
|
|
|
|
.then(() => style);
|
2018-01-10 18:56:14 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return style;
|
|
|
|
}
|
|
|
|
|
2018-08-18 20:17:20 +00:00
|
|
|
/**
|
|
|
|
* Parse the source and find the duplication
|
|
|
|
* @param _
|
|
|
|
* @param {String} _.sourceCode
|
|
|
|
* @param {Boolean=} _.checkDup
|
|
|
|
* @param {Boolean=} _.metaOnly
|
|
|
|
* @returns {Promise<{style, dup:Boolean?}>}
|
|
|
|
*/
|
|
|
|
function build({
|
|
|
|
sourceCode,
|
|
|
|
checkDup,
|
|
|
|
metaOnly,
|
|
|
|
}) {
|
|
|
|
const task = buildMeta({sourceCode});
|
|
|
|
return (metaOnly ? task : task.then(usercss.buildCode))
|
2018-01-07 13:36:30 +00:00
|
|
|
.then(style => ({
|
|
|
|
style,
|
2018-01-10 18:56:14 +00:00
|
|
|
dup: checkDup && find(style),
|
2018-01-07 13:36:30 +00:00
|
|
|
}));
|
2017-09-18 03:34:12 +00:00
|
|
|
}
|
|
|
|
|
2018-01-10 18:56:14 +00:00
|
|
|
// Parse the source, apply customizations, report fatal/syntax errors
|
|
|
|
function parse(style, allowErrors = false) {
|
2018-01-04 13:41:55 +00:00
|
|
|
// restore if stripped by getStyleWithNoCode
|
|
|
|
if (typeof style.sourceCode !== 'string') {
|
2018-01-01 17:02:49 +00:00
|
|
|
style.sourceCode = cachedStyles.byId.get(style.id).sourceCode;
|
|
|
|
}
|
|
|
|
return buildMeta(style)
|
2017-09-18 03:34:12 +00:00
|
|
|
.then(assignVars)
|
2018-01-10 18:56:14 +00:00
|
|
|
.then(style => usercss.buildCode(style, allowErrors));
|
|
|
|
}
|
|
|
|
|
|
|
|
function save(style, allowErrors = false) {
|
|
|
|
return parse(style, allowErrors)
|
2018-01-07 13:36:30 +00:00
|
|
|
.then(result =>
|
|
|
|
allowErrors ?
|
|
|
|
saveStyle(result.style).then(style => ({style, errors: result.errors})) :
|
|
|
|
saveStyle(result));
|
2017-09-18 03:34:12 +00:00
|
|
|
}
|
|
|
|
|
2018-01-09 16:13:37 +00:00
|
|
|
/**
|
|
|
|
* @param {Style|{name:string, namespace:string}} styleOrData
|
|
|
|
* @returns {Style}
|
|
|
|
*/
|
2018-01-10 18:56:14 +00:00
|
|
|
function find(styleOrData) {
|
2018-01-09 16:13:37 +00:00
|
|
|
if (styleOrData.id) return cachedStyles.byId.get(styleOrData.id);
|
|
|
|
const {name, namespace} = styleOrData.usercssData || styleOrData;
|
2018-01-07 13:36:30 +00:00
|
|
|
for (const dup of cachedStyles.list) {
|
|
|
|
const data = dup.usercssData;
|
|
|
|
if (!data) continue;
|
|
|
|
if (data.name === name &&
|
|
|
|
data.namespace === namespace) {
|
|
|
|
return dup;
|
|
|
|
}
|
2017-09-18 03:34:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-03 15:26:31 +00:00
|
|
|
function install({url, direct, downloaded, tab}, sender) {
|
|
|
|
tab = tab !== undefined ? tab : sender.tab;
|
2018-01-01 17:02:49 +00:00
|
|
|
url = url || tab.url;
|
2017-12-06 19:35:19 +00:00
|
|
|
if (direct && !downloaded) {
|
2017-11-24 16:33:50 +00:00
|
|
|
prefetchCodeForInstallation(tab.id, url);
|
|
|
|
}
|
2018-01-01 17:02:49 +00:00
|
|
|
return openURL({
|
2017-11-24 16:33:50 +00:00
|
|
|
url: '/install-usercss.html' +
|
|
|
|
'?updateUrl=' + encodeURIComponent(url) +
|
2017-11-25 17:24:15 +00:00
|
|
|
'&tabId=' + tab.id +
|
|
|
|
(direct ? '&direct=yes' : ''),
|
2017-10-01 12:34:29 +00:00
|
|
|
index: tab.index + 1,
|
|
|
|
openerTabId: tab.id,
|
2017-12-31 14:58:35 +00:00
|
|
|
currentWindow: null,
|
2018-01-01 17:02:49 +00:00
|
|
|
});
|
2017-09-24 03:39:04 +00:00
|
|
|
}
|
|
|
|
|
2017-11-24 16:33:50 +00:00
|
|
|
function prefetchCodeForInstallation(tabId, url) {
|
2017-12-10 07:04:13 +00:00
|
|
|
const key = TEMP_CODE_PREFIX + tabId;
|
|
|
|
tempCodeLastWriteDate = Date.now();
|
2017-11-24 16:33:50 +00:00
|
|
|
Promise.all([
|
|
|
|
download(url),
|
|
|
|
chromeLocal.setValue(key, {loading: true}),
|
|
|
|
]).then(([code]) => {
|
|
|
|
chromeLocal.setValue(key, code);
|
2017-12-10 07:04:13 +00:00
|
|
|
setTimeout(() => chromeLocal.remove(key), TEMP_CODE_CLEANUP_DELAY);
|
2017-11-24 16:33:50 +00:00
|
|
|
});
|
|
|
|
}
|
2017-09-18 03:34:12 +00:00
|
|
|
})();
|