Compare commits

..

7 Commits

Author SHA1 Message Date
eight04
c13be76553 Bump web-ext 2021-12-12 23:43:42 +08:00
eight04
63e48099ad Pass eslint 2021-12-12 23:38:17 +08:00
eight04
92573307cd Build with preserve signature false 2021-12-12 23:25:00 +08:00
eight04
e456457fa6 Build 2021-12-12 23:21:42 +08:00
eight04
a91dc72304 First approach 2021-12-12 23:14:31 +08:00
eight04
879362e0b7 Merge branch 'master' of https://github.com/openstyles/stylus into dev-rollup 2021-12-12 21:33:04 +08:00
eight04
f1f1e56e04 Change: move extension to dist folder 2021-12-12 21:32:50 +08:00
370 changed files with 23668 additions and 20397 deletions

22
.cjsescache Normal file
View File

@ -0,0 +1,22 @@
[
"node_modules/codemirror/addon/comment/comment.js",
"node_modules/codemirror/addon/dialog/dialog.js",
"node_modules/codemirror/addon/edit/closebrackets.js",
"node_modules/codemirror/addon/edit/matchbrackets.js",
"node_modules/codemirror/addon/fold/brace-fold.js",
"node_modules/codemirror/addon/fold/comment-fold.js",
"node_modules/codemirror/addon/fold/foldcode.js",
"node_modules/codemirror/addon/fold/foldgutter.js",
"node_modules/codemirror/addon/fold/indent-fold.js",
"node_modules/codemirror/addon/hint/css-hint.js",
"node_modules/codemirror/addon/hint/show-hint.js",
"node_modules/codemirror/addon/lint/lint.js",
"node_modules/codemirror/addon/scroll/annotatescrollbar.js",
"node_modules/codemirror/addon/search/matchesonscrollbar.js",
"node_modules/codemirror/addon/search/searchcursor.js",
"node_modules/codemirror/addon/selection/active-line.js",
"node_modules/codemirror/keymap/sublime.js",
"node_modules/codemirror/lib/codemirror.js",
"node_modules/codemirror/mode/css/css.js",
"node_modules/codemirror/mode/stylus/stylus.js"
]

View File

@ -1,2 +0,0 @@
vendor/
vendor-overwrites/

View File

@ -238,3 +238,7 @@ overrides:
- files: ["**/*worker*.js"] - files: ["**/*worker*.js"]
env: env:
worker: true worker: true
- files: ["**/*.mjs"]
parserOptions:
sourceType: module

View File

@ -24,9 +24,6 @@ If not, then provide details describing which page the feature will effect, e.g.
## Adding translations ## Adding translations
You can help us translate the extension on [Transifex](https://www.transifex.com/github-7/Stylus). You can help us translate the extension on [Transifex](https://www.transifex.com/github-7/Stylus).
Only the languages supported by the web store are allowed:
https://developer.chrome.com/docs/webstore/i18n/#localeTable
## Pull requests ## Pull requests

10
.gitignore vendored
View File

@ -1,8 +1,10 @@
*.zip
.DS_Store .DS_Store
.eslintcache pull_locales_login.rb
.transifexrc
.vscode .vscode
desktop.ini
node_modules/ node_modules/
yarn.lock yarn.lock
*.zip
*.log
*.swp
.eslintcache
.transifexrc

View File

@ -2,9 +2,9 @@
host = https://www.transifex.com host = https://www.transifex.com
[Stylus.messages] [Stylus.messages]
file_filter = _locales/<lang>/messages.json file_filter = dist/_locales/<lang>/messages.json
minimum_perc = 0 minimum_perc = 0
source_file = _locales/en/messages.json source_file = dist/_locales/en/messages.json
source_lang = en_US source_lang = en_US
type = CHROME type = CHROME

View File

@ -50,7 +50,7 @@ Copyright &copy; 2005-2014 [Jason Barnabe](jason.barnabe@gmail.com)
Current Stylus: Current Stylus:
Copyright &copy; 2017-2022 [Stylus Team](https://github.com/openstyles/stylus/graphs/contributors) Copyright &copy; 2017-2019 [Stylus Team](https://github.com/openstyles/stylus/graphs/contributors)
**[GNU GPLv3](./LICENSE)** **[GNU GPLv3](./LICENSE)**

File diff suppressed because it is too large Load Diff

View File

@ -1,379 +0,0 @@
{
"InaccessibleFileHint": {
"message": "Stylus không thể truy cập một số kiểu tập tin (chẳng hạn như PDF và JSON)."
},
"addStyleLabel": {
"message": "Viết bảng định kiểu mới"
},
"addStyleTitle": {
"message": "Thêm bảng định kiểu"
},
"alphaChannel": {
"message": "Độ mờ"
},
"appliesAdd": {
"message": "Thêm"
},
"appliesDisplay": {
"message": "Áp dụng với: $applies$",
"placeholders": {
"applies": {
"content": "$1"
}
}
},
"appliesDisplayTruncatedSuffix": {
"message": "và một số khác"
},
"appliesDomainOption": {
"message": "Các địa chỉ thuộc tên miền này"
},
"appliesHelp": {
"message": "Dùng tuỳ chọn \"Áp dụng với\" để giới hạn các địa chỉ cho đoạn mã này"
},
"appliesLabel": {
"message": "Áp dụng với"
},
"appliesLineWidgetLabel": {
"message": "Hiển thị thông tin \"Áp dụng với\""
},
"appliesLineWidgetWarning": {
"message": "Không hoạt động với CSS tối giản"
},
"appliesRegexpOption": {
"message": "URL khớp với biểu thức chính quy"
},
"appliesRemove": {
"message": "Xoá"
},
"appliesRemoveError": {
"message": "Không thể xoá mục \"Áp dụng với\" cuối cùng"
},
"appliesSpecify": {
"message": "Chỉ định"
},
"appliesToEverything": {
"message": "Tất cả"
},
"appliesUrlPrefixOption": {
"message": "URL bắt đầu bằng"
},
"author": {
"message": "Tác giả"
},
"backupButtons": {
"message": "Sao lưu"
},
"backupMessage": {
"message": "Để nhập tập tin sao lưu, kéo và thả nó vào trang này hoặc nhấp vào nút Nhập.\n\nĐể xuất một bản sao lưu tương thích với Stylus trước phiên bản 1.5.18, nhấp chuột phải hoặc nhấn giữ Shift khi nhấp chuột trái vào nút Xuất."
},
"bckpInstStyles": {
"message": "Xuất bảng định kiểu"
},
"checkForUpdate": {
"message": "Kiểm tra bản cập nhật mới"
},
"checkingForUpdate": {
"message": "Đang kiểm tra..."
},
"clickToUninstall": {
"message": "Nhấp để huỷ kích hoạt"
},
"cm_autoCloseBrackets": {
"message": "Tự động đóng ngoặc và nháy"
},
"cm_autoCloseBracketsTooltip": {
"message": "Tự động thêm dấu đóng tương ứng khi nhập một trong các dấu (, [, {, ' và \"."
},
"cm_autocompleteOnTyping": {
"message": "Tự động hoàn thành"
},
"cm_colorpicker": {
"message": "Bộ chọn màu cho màu CSS"
},
"cm_indentWithTabs": {
"message": "Lùi đầu dòng thông minh bằng tab"
},
"cm_lineWrapping": {
"message": "Gập dòng dài"
},
"cm_linter": {
"message": "Trình phân tích cú pháp"
},
"cm_matchHighlight": {
"message": "Làm nổi"
},
"cm_matchHighlightSelection": {
"message": "Chỉ vùng được chọn"
},
"cm_matchHighlightToken": {
"message": "Token nằm dưới con trỏ văn bản"
},
"cm_selectByTokens": {
"message": "Nhấp đúp để chọn token"
},
"cm_selectByTokensTooltip": {
"message": "Ví dụ về token: .foo-bar-2 #aabbcc 0.32 !important\nKhi tắt: Chọn từ (phân tách bằng dấu câu)."
},
"cm_smartIndent": {
"message": "Lùi đầu dòng thông minh"
},
"cm_tabSize": {
"message": "Chiều rộng tab"
},
"cm_theme": {
"message": "Chủ đề"
},
"colorpickerSwitchFormatTooltip": {
"message": "Đổi định dạng: HEX → RGB → HSL.\nNhấn giữ phím Shift khi nhấp để đảo thứ tự.\nPhím tắt: PgUp và PgDn."
},
"colorpickerTooltip": {
"message": "Mở bộ chọn màu"
},
"configOnChange": {
"message": "khi thay đổi"
},
"configOnChangeTooltip": {
"message": "Tự động lưu và áp dụng"
},
"configureStyle": {
"message": "Thiết lập"
},
"configureStyleOnHomepage": {
"message": "Thiết lập trên trang chủ"
},
"confirmCancel": {
"message": "Huỷ"
},
"confirmClose": {
"message": "Đóng"
},
"confirmDefault": {
"message": "Dùng mặc định"
},
"confirmDelete": {
"message": "Xoá"
},
"confirmDiscardChanges": {
"message": "Huỷ thay đổi?"
},
"confirmNo": {
"message": "Không"
},
"confirmSave": {
"message": "Lưu"
},
"confirmStop": {
"message": "Dừng"
},
"confirmYes": {
"message": "Có"
},
"connectingDropbox": {
"message": "Đang kết nối với Dropbox..."
},
"connectingDropboxNotAllowed": {
"message": "Tính năng kết nối với Dropbox chỉ khả dụng khi cài ứng dụng trực tiếp từ cửa hàng web"
},
"copied": {
"message": "Đã chép vào khay nhớ tạm"
},
"copy": {
"message": "Chép vào khay nhớ tạm"
},
"dateAbbrDay": {
"message": "$value$ ngày",
"placeholders": {
"value": {
"content": "$1"
}
}
},
"dateAbbrHour": {
"message": "$value$ giờ",
"placeholders": {
"value": {
"content": "$1"
}
}
},
"dateAbbrMonth": {
"message": "$value$ tháng",
"placeholders": {
"value": {
"content": "$1"
}
}
},
"dateAbbrYear": {
"message": "$value$ năm",
"placeholders": {
"value": {
"content": "$1"
}
}
},
"dateInstalled": {
"message": "Ngày cài đặt"
},
"dateUpdated": {
"message": "Ngày cập nhật"
},
"defaultTheme": {
"message": "mặc định"
},
"deleteStyleConfirm": {
"message": "Bạn có chắc chắn muốn xoá bảng định kiểu này không?"
},
"deleteStyleLabel": {
"message": "Xoá"
},
"disableAllStyles": {
"message": "Tắt tất cả bảng định kiểu"
},
"disableAllStylesOff": {
"message": "Bật tất cả bảng định kiểu"
},
"disableStyleLabel": {
"message": "Vô hiệu hoá"
},
"draftAction": {
"message": "Chọn \"Có\" để tải bản nháp hoặc \"Không\" để xoá nó đi."
},
"editDeleteText": {
"message": "Xoá"
},
"editGotoLine": {
"message": "Đi đến dòng (hoặc dòng:cột)"
},
"editStyleHeading": {
"message": "Sửa bảng định kiểu"
},
"editStyleLabel": {
"message": "Sửa"
},
"editStyleTitle": {
"message": "Sửa bảng định kiểu $stylename$",
"placeholders": {
"stylename": {
"content": "$1"
}
}
},
"editorSettings": {
"message": "Cài đặt trình soạn thảo"
},
"enableStyleLabel": {
"message": "Kích hoạt"
},
"exportCompatible": {
"message": "Xuất (chế độ tương thích)"
},
"exportLabel": {
"message": "Xuất"
},
"exportSavedSuccess": {
"message": "Lưu thành công"
},
"externalFeedback": {
"message": "Phản hồi"
},
"externalHomepage": {
"message": "Trang chủ"
},
"externalLink": {
"message": "Liên kết ngoài"
},
"externalSupport": {
"message": "Ủng hộ"
},
"genericAdd": {
"message": "Thêm"
},
"genericDescription": {
"message": "Mô tả"
},
"genericDisabledLabel": {
"message": "Vô hiệu hoá"
},
"genericEnabledLabel": {
"message": "Kích hoạt"
},
"genericError": {
"message": "Lỗi"
},
"genericHistoryLabel": {
"message": "Lịch sử"
},
"genericNext": {
"message": "Sau"
},
"genericPrevious": {
"message": "Trước"
},
"genericResetLabel": {
"message": "Đặt lại"
},
"genericSavedMessage": {
"message": "Đã lưu"
},
"genericTest": {
"message": "Thứ"
},
"genericTitle": {
"message": "Tiêu đề"
},
"genericUnknown": {
"message": "Không xác định"
},
"helpAlt": {
"message": "Trợ giúp"
},
"importLabel": {
"message": "Nhập"
},
"importReportLegendAdded": {
"message": "đã thêm"
},
"importReportLegendIdentical": {
"message": "Đã bỏ qua các tệp trùng lặp"
},
"importReportLegendInvalid": {
"message": "Đã bỏ qua các tệp không hợp lệ"
},
"importReportLegendUpdatedBoth": {
"message": "đã cập nhật siêu thông tin và mã"
},
"importReportLegendUpdatedCode": {
"message": "đã cập nhật mã"
},
"importReportLegendUpdatedMeta": {
"message": "đã cập nhật siêu thông tin"
},
"importReportTitle": {
"message": "Đã nhập xong"
},
"installUpdateFromLabel": {
"message": "Kiểm tra bản cập nhật mới"
},
"license": {
"message": "Giấy phép"
},
"linterCSSLintIncompatible": {
"message": "CSSLint không hỗ trợ bộ tiền xử lý $preprocessorname$",
"placeholders": {
"preprocessorname": {
"content": "$1"
}
}
},
"linterJSONError": {
"message": "Định dạng JSON không hợp lệ"
},
"styleEnabledLabel": {
"message": "Kích hoạt"
},
"styleSaveLabel": {
"message": "Lưu"
}
}

View File

@ -1,91 +0,0 @@
/* global prefs */
/* exported colorScheme */
'use strict';
const colorScheme = (() => {
const changeListeners = new Set();
const kSTATE = 'schemeSwitcher.enabled';
const kSTART = 'schemeSwitcher.nightStart';
const kEND = 'schemeSwitcher.nightEnd';
const SCHEMES = ['dark', 'light'];
const isDark = {
never: null,
dark: true,
light: false,
system: false,
time: false,
};
let isDarkNow = false;
prefs.subscribe(kSTATE, () => update());
prefs.subscribe([kSTART, kEND], (key, value) => {
updateTimePreferDark();
createAlarm(key, value);
}, {runNow: true});
chrome.alarms.onAlarm.addListener(({name}) => {
if (name === kSTART || name === kEND) {
updateTimePreferDark();
}
});
return {
SCHEMES,
onChange(listener) {
changeListeners.add(listener);
},
isDark: () => isDarkNow,
/** @param {StyleObj} _ */
shouldIncludeStyle({preferScheme: ps}) {
return prefs.get(kSTATE) === 'never' ||
!SCHEMES.includes(ps) ||
isDarkNow === (ps === 'dark');
},
updateSystemPreferDark(val) {
update('system', val);
return true;
},
};
function calcTime(key) {
const [h, m] = prefs.get(key).split(':');
return (h * 3600 + m * 60) * 1000;
}
function createAlarm(key, value) {
const date = new Date();
const [h, m] = value.split(':');
date.setHours(h, m, 0, 0);
if (date.getTime() < Date.now()) {
date.setDate(date.getDate() + 1);
}
chrome.alarms.create(key, {
when: date.getTime(),
periodInMinutes: 24 * 60,
});
}
function updateTimePreferDark() {
const now = Date.now() - new Date().setHours(0, 0, 0, 0);
const start = calcTime(kSTART);
const end = calcTime(kEND);
const val = start > end ?
now >= start || now < end :
now >= start && now < end;
update('time', val);
}
function update(type, val) {
if (type) {
if (isDark[type] === val) return;
isDark[type] = val;
}
val = isDark[prefs.get(kSTATE)];
if (isDarkNow !== val) {
isDarkNow = val;
for (const listener of changeListeners) {
listener(isDarkNow);
}
}
}
})();

View File

@ -1,92 +0,0 @@
/* global API */// msg.js
'use strict';
/**
* Common stuff that's loaded first so it's immediately available to all background scripts
*/
window.bgReady = {}; /* global bgReady */
bgReady.styles = new Promise(r => (bgReady._resolveStyles = r));
bgReady.all = new Promise(r => (bgReady._resolveAll = r));
const uuidIndex = Object.assign(new Map(), {
custom: {},
/** `obj` must have a unique `id`, a UUIDv4 `_id`, and Date.now() for `_rev`. */
addCustomId(obj, {get = () => obj, set}) {
Object.defineProperty(uuidIndex.custom, obj.id, {get, set});
},
});
/* exported addAPI */
function addAPI(methods) {
for (const [key, val] of Object.entries(methods)) {
const old = API[key];
if (old && Object.prototype.toString.call(old) === '[object Object]') {
Object.assign(old, val);
} else {
API[key] = val;
}
}
}
/* exported createCache */
/** Creates a FIFO limit-size map. */
function createCache({size = 1000, onDeleted} = {}) {
const map = new Map();
const buffer = Array(size);
let index = 0;
let lastIndex = 0;
return {
get(id) {
const item = map.get(id);
return item && item.data;
},
set(id, data) {
if (map.size === size) {
// full
map.delete(buffer[lastIndex].id);
if (onDeleted) {
onDeleted(buffer[lastIndex].id, buffer[lastIndex].data);
}
lastIndex = (lastIndex + 1) % size;
}
const item = {id, data, index};
map.set(id, item);
buffer[index] = item;
index = (index + 1) % size;
},
delete(id) {
const item = map.get(id);
if (!item) {
return false;
}
map.delete(item.id);
const lastItem = buffer[lastIndex];
lastItem.index = item.index;
buffer[item.index] = lastItem;
lastIndex = (lastIndex + 1) % size;
if (onDeleted) {
onDeleted(item.id, item.data);
}
return true;
},
clear() {
map.clear();
index = lastIndex = 0;
},
has: id => map.has(id),
*entries() {
for (const [id, item] of map) {
yield [id, item.data];
}
},
*values() {
for (const item of map.values()) {
yield item.data;
}
},
get size() {
return map.size;
},
};
}

View File

@ -1,87 +0,0 @@
/* global browserCommands */// background.js
/* global msg */
/* global prefs */
/* global CHROME URLS ignoreChromeError */// toolbox.js
'use strict';
chrome.management.getSelf(ext => {
const contextMenus = Object.assign({
'show-badge': {
title: 'menuShowBadge',
click: togglePref,
},
'disableAll': {
title: 'disableAllStyles',
click: browserCommands.styleDisableAll,
},
'open-manager': {
title: 'optionsOpenManager',
click: browserCommands.openManage,
},
'open-options': {
title: 'openOptions',
click: browserCommands.openOptions,
},
}, ext.installType === 'development' && {
'reload': {
title: 'reload',
click: browserCommands.reload,
},
}, CHROME && {
'editor.contextDelete': {
title: 'editDeleteText',
type: 'normal',
contexts: ['editable'],
documentUrlPatterns: [URLS.ownOrigin + '*'],
click: (info, tab) => {
msg.sendTab(tab.id, {method: 'editDeleteText'}, undefined, 'extension')
.catch(msg.ignoreError);
},
},
});
createContextMenus(Object.keys(contextMenus));
chrome.contextMenus.onClicked.addListener((info, tab) =>
contextMenus[info.menuItemId].click(info, tab));
function createContextMenus(ids) {
for (const id of ids) {
const item = Object.assign({id, contexts: ['browser_action']}, contextMenus[id]);
item.title = chrome.i18n.getMessage(item.title);
if (typeof prefs.defaults[id] === 'boolean') {
if (item.type) {
prefs.subscribe(id, togglePresence);
} else {
item.type = 'checkbox';
item.checked = prefs.get(id);
prefs.subscribe(id, CHROME >= 62 && CHROME <= 64 ? toggleCheckmarkBugged : toggleCheckmark);
}
}
delete item.click;
chrome.contextMenus.create(item, ignoreChromeError);
}
}
function toggleCheckmark(id, checked) {
chrome.contextMenus.update(id, {checked}, ignoreChromeError);
}
/** Circumvents the bug with disabling check marks in Chrome 62-64 */
async function toggleCheckmarkBugged(id) {
await browser.contextMenus.remove(id).catch(ignoreChromeError);
createContextMenus([id]);
}
/** @param {chrome.contextMenus.OnClickData} info */
function togglePref(info) {
prefs.set(info.menuItemId, info.checked);
}
function togglePresence(id, checked) {
if (checked) {
createContextMenus([id]);
} else {
chrome.contextMenus.remove(id, ignoreChromeError);
}
}
});

View File

@ -1,150 +0,0 @@
/* global addAPI */// common.js
/* global chromeLocal */// storage-util.js
/* global cloneError */// worker-util.js
/* global deepCopy */// toolbox.js
/* global prefs */
'use strict';
/*
Initialize a database. There are some problems using IndexedDB in Firefox:
https://www.reddit.com/r/firefox/comments/74wttb/note_to_firefox_webextension_developers_who_use/
Some of them are fixed in FF59:
https://www.reddit.com/r/firefox/comments/7ijuaq/firefox_59_webextensions_can_use_indexeddb_when/
*/
/* exported db */
const db = (() => {
let exec = async (...args) => (
exec = await tryUsingIndexedDB().catch(useChromeStorage)
)(...args);
const DB = 'stylish';
const FALLBACK = 'dbInChromeStorage';
const ID_AS_KEY = {[DB]: true};
const getStoreName = dbName => dbName === DB ? 'styles' : 'data';
const cache = {};
const proxies = {};
const proxyHandler = {
get: ({dbName}, cmd) =>
(...args) =>
(dbName === DB ? exec : cachedExec)(dbName, cmd, ...args),
};
/**
* @param {string} dbName
* @return {IDBObjectStore | {putMany: function(items:?[]):Promise<?[]>}}
*/
const getProxy = dbName => proxies[dbName] || (
(proxies[dbName] = new Proxy({dbName}, proxyHandler))
);
addAPI(/** @namespace API */ {
drafts: getProxy('drafts'),
/** Storage for big items that may exceed 8kB limit of chrome.storage.sync.
* To make an item syncable register it with uuidIndex.addCustomId. */
prefsDb: getProxy(prefs.STORAGE_KEY),
});
return {
styles: getProxy(DB),
};
async function cachedExec(dbName, cmd, a, b) {
const hub = cache[dbName] || (cache[dbName] = {});
const res = cmd === 'get' && a in hub ? hub[a] : await exec(...arguments);
if (cmd === 'get') {
hub[a] = deepCopy(res);
} else if (cmd === 'put') {
hub[ID_AS_KEY[dbName] ? a.id : b] = deepCopy(a);
} else if (cmd === 'delete') {
delete hub[a];
}
return res;
}
async function tryUsingIndexedDB() {
// we use chrome.storage.local fallback if IndexedDB doesn't save data,
// which, once detected on the first run, is remembered in chrome.storage.local
// note that accessing indexedDB may throw, https://github.com/openstyles/stylus/issues/615
if (typeof indexedDB === 'undefined') {
throw new Error('indexedDB is undefined');
}
switch (await chromeLocal.getValue(FALLBACK)) {
case true: throw null;
case false: break;
default: await testDB();
}
chromeLocal.setValue(FALLBACK, false);
return dbExecIndexedDB;
}
async function testDB() {
const id = `${performance.now()}.${Math.random()}.${Date.now()}`;
await dbExecIndexedDB(DB, 'put', {id});
const e = await dbExecIndexedDB(DB, 'get', id);
await dbExecIndexedDB(DB, 'delete', e.id); // throws if `e` or id is null
}
async function useChromeStorage(err) {
chromeLocal.setValue(FALLBACK, true);
if (err) {
chromeLocal.setValue(FALLBACK + 'Reason', cloneError(err));
console.warn('Failed to access indexedDB. Switched to storage API.', err);
}
await require(['/background/db-chrome-storage']); /* global createChromeStorageDB */
const BASES = {};
return (dbName, method, ...args) => (
BASES[dbName] || (
BASES[dbName] = createChromeStorageDB(dbName !== DB && `${dbName}-`)
)
)[method](...args);
}
async function dbExecIndexedDB(dbName, method, ...args) {
const mode = method.startsWith('get') ? 'readonly' : 'readwrite';
const storeName = getStoreName(dbName);
const store = (await open(dbName)).transaction([storeName], mode).objectStore(storeName);
const fn = method === 'putMany' ? putMany : storeRequest;
return fn(store, method, ...args);
}
function storeRequest(store, method, ...args) {
return new Promise((resolve, reject) => {
/** @type {IDBRequest} */
const request = store[method](...args);
request.onsuccess = () => resolve(request.result);
request.onerror = reject;
});
}
function putMany(store, _method, items) {
return Promise.all(items.map(item => storeRequest(store, 'put', item)));
}
function open(name) {
return new Promise((resolve, reject) => {
const request = indexedDB.open(name, 2);
request.onsuccess = e => resolve(create(e));
request.onerror = reject;
request.onupgradeneeded = create;
});
}
function create(event) {
/** @type IDBDatabase */
const idb = event.target.result;
const dbName = idb.name;
const sn = getStoreName(dbName);
if (!idb.objectStoreNames.contains(sn)) {
if (event.type === 'success') {
idb.close();
return new Promise(resolve => {
indexedDB.deleteDatabase(dbName).onsuccess = () => {
resolve(open(dbName));
};
});
}
idb.createObjectStore(sn, ID_AS_KEY[dbName] ? {
keyPath: 'id',
autoIncrement: true,
} : undefined);
}
return idb;
}
})();

View File

@ -1,158 +0,0 @@
/* global URLS stringAsRegExp */// toolbox.js
/* global usercssMan */
'use strict';
const usoApi = {};
(() => {
const pingers = {};
usoApi.pingback = (usoId, delay) => {
clearTimeout(pingers[usoId]);
delete pingers[usoId];
if (delay > 0) {
return new Promise(resolve => (pingers[usoId] = setTimeout(ping, delay, usoId, resolve)));
} else if (delay !== false) {
return ping(usoId);
}
};
/**
* Replicating USO-Archive format
* https://github.com/33kk/uso-archive/blob/flomaster/lib/uso.js
* https://github.com/33kk/uso-archive/blob/flomaster/lib/converters.js
*/
usoApi.toUsercss = async (data, {metaOnly = true, varsUrl} = {}) => {
const badKeys = {};
const newKeys = [];
const descr = JSON.stringify(data.description.trim());
const vars = (data.style_settings || []).map(makeVar, {badKeys, newKeys}).join('');
const sourceCode = `\
/* ==UserStyle==
@name ${data.name}
@namespace USO Archive
@version ${data.updated.replace(/-/g, '').replace(/[T:]/g, '.').slice(0, 14)}
@description ${/^"['`]|\\/.test(descr) ? descr : descr.slice(1, -1)}
@author ${(data.user || {}).name || '?'}
@license ${makeLicense(data.license)}${vars ? '\n@preprocessor uso' + vars : ''}`
.replace(/\*\//g, '*\\/') +
`==/UserStyle== */\n${newKeys[0] ? useNewKeys(data.css, badKeys) : data.css}`;
const {style} = await usercssMan.build({sourceCode, metaOnly});
usoApi.useVarsUrl(style, varsUrl);
return {style, badKeys, newKeys};
};
usoApi.useVarsUrl = (style, url) => {
if (!/\?ik-/.test(url)) {
return;
}
const cfg = {badKeys: {}, newKeys: []};
const {vars} = style.usercssData;
if (!vars) {
return;
}
for (let [key, val] of new URLSearchParams(url.split('?')[1])) {
if (!key.startsWith('ik-')) continue;
key = makeKey(key.slice(3), cfg);
const v = vars[key];
if (!v) continue;
if (v.options) {
let sel = val.startsWith('ik-') && optByName(v, makeKey(val.slice(3), cfg));
if (!sel) {
key += '-custom';
sel = optByName(v, key + '-dropdown');
if (sel) vars[key].value = val;
}
if (sel) v.value = sel.name;
} else {
v.value = val;
}
}
return true;
};
async function ping(id, resolve) {
await fetch(`${URLS.uso}styles/install/${id}?source=stylish-ch`);
if (resolve) resolve(true);
return true;
}
function makeKey(key, {badKeys, newKeys}) {
let res = badKeys[key];
if (!res) {
res = key.replace(/[^-\w]/g, '-');
res += newKeys.includes(res) ? '-' : '';
if (key !== res) {
badKeys[key] = res;
newKeys.push(res);
}
}
return res;
}
function makeLicense(s) {
return !s ? 'NO-REDISTRIBUTION' :
s === 'publicdomain' ? 'CC0-1.0' :
s.startsWith('ccby') ? `${s.toUpperCase().match(/(..)/g).join('-')}-4.0` :
s;
}
function makeVar({
label,
setting_type: type,
install_key: ik,
style_setting_options: opts,
}) {
const cfg = this;
let value, suffix;
ik = makeKey(ik, cfg);
label = JSON.stringify(label);
switch (type) {
case 'color':
value = opts[0].value;
break;
case 'text':
value = JSON.stringify(opts[0].value);
break;
case 'image': {
const ikCust = `${ik}-custom`;
opts.push({
label: 'Custom',
install_key: `${ikCust}-dropdown`,
value: `/*[[${ikCust}]]*/`,
});
suffix = `\n@advanced text ${ikCust} ${label.slice(0, -1)} (Custom)" "https://foo.com/123.jpg"`;
type = 'dropdown';
} // fallthrough
case 'dropdown':
value = '';
for (const o of opts) {
const def = o.default ? '*' : '';
const val = o.value;
const s = ` ${makeKey(o.install_key, cfg)} ${JSON.stringify(o.label + def)} <<<EOT${
val.includes('\n') ? '\n' : ' '}${val} EOT;\n`;
value = def ? s + value : value + s;
}
value = `{\n${value}}`;
break;
default:
value = '"ERROR: unknown type"';
}
return `\n@advanced ${type} ${ik} ${label} ${value}${suffix || ''}`;
}
function optByName(v, name) {
return v.options.find(o => o.name === name);
}
function useNewKeys(css, badKeys) {
const rxsKeys = stringAsRegExp(Object.keys(badKeys).join('\n'), '', true).replace(/\n/g, '|');
const rxUsoVars = new RegExp(`(/\\*\\[\\[)(${rxsKeys})(?=]]\\*/)`, 'g');
return css.replace(rxUsoVars, (s, a, key) => a + badKeys[key]);
}
})();

View File

@ -1,324 +0,0 @@
/* global API */// msg.js
'use strict';
// eslint-disable-next-line no-unused-expressions
/^\/styles\/(\d+)(\/([^/]*))?([?#].*)?$/.test(location.pathname) && (async () => {
if (window.INJECTED_USO === 1) return;
window.INJECTED_USO = 1;
const usoId = RegExp.$1;
const USO = 'https://userstyles.org';
const apiUrl = `${USO}/api/v1/styles/${usoId}`;
const md5Url = `https://update.userstyles.org/${usoId}.md5`;
const CLICK = [
['#install_stylish_style_button', onInstall],
['#update_stylish_style_button', onInstall],
['.customize_style_button', onCustomize],
['.uninstall_stylish_style_button', onUninstall],
];
const pageEventId = `${performance.now()}${Math.random()}`;
const contentEventId = pageEventId + ':';
const orphanEventId = chrome.runtime.id; // id won't be available in the orphaned script
const $ = (sel, base = document) => base.querySelector(sel);
const toggleListener = (isOn, ...args) => (isOn ? addEventListener : removeEventListener)(...args);
const togglePageListener = isOn => toggleListener(isOn, contentEventId, onPageEvent, true);
const mo = new MutationObserver(onMutation);
const observeColors = isOn =>
isOn ? mo.observe(document.body, {subtree: true, attributes: true, attributeFilter: ['value']})
: mo.disconnect();
let style, dup, md5, pageData, badKeys;
runInPage(inPageContext, pageEventId, contentEventId, usoId, apiUrl);
addEventListener(orphanEventId, orphanCheck, true);
addEventListener('click', onClick, true);
togglePageListener(true);
[md5, dup] = await Promise.all([
fetch(md5Url).then(r => r.text()),
API.styles.find({md5Url}, {installationUrl: `https://uso.kkx.one/style/${usoId}`})
.then(sendVarsToPage),
document.body || new Promise(resolve => addEventListener('load', resolve, {once: true})),
]);
if (!dup) {
sendStylishEvent('styleCanBeInstalledChrome');
} else if (dup.originalMd5 && dup.originalMd5 !== md5 || !dup.usercssData || !dup.md5Url) {
// allow update if 1) changed, 2) is a classic USO style, 3) is from USO-archive
sendStylishEvent('styleCanBeUpdatedChrome');
} else {
sendStylishEvent('styleAlreadyInstalledChrome');
}
async function onClick(e) {
for (const [sel, fn] of CLICK) {
const el = e.target.closest(sel);
if (!el) continue;
try {
el.disabled = true;
await fn(e);
} catch (e) {
alert(chrome.i18n.getMessage('styleInstallFailed', e.message || e));
} finally {
el.disabled = false;
}
}
}
function onCustomize() {
const ss = $('#style-settings');
const willShow = !ss || !ss.offsetHeight;
observeColors(willShow);
toggleListener(willShow, 'change', onChange);
}
async function onInstall(e) {
const {id} = dup;
e.stopPropagation();
if (!style) await buildStyle();
style = dup = await API.usercss.install(style, {
dup: {id},
vars: getPageVars(),
});
sendStylishEvent('styleInstalledChrome');
API.uso.pingback(id);
}
function onUninstall() {
const {id} = dup;
dup = style = false;
observeColors(false);
removeEventListener('change', onChange);
return API.styles.delete(id);
}
function onChange({target: el}) {
if (dup && el.matches('[name^="ik-"], [type=file]')) {
API.usercss.configVars(dup.id, getPageVars());
}
}
function onMutation(mutations) {
for (const {target: el} of mutations) {
if (el.style.display === 'none' &&
/^ik-/.test(el.name) &&
/^#[\da-f]{6}$/.test(el.value)) {
onChange({target: el});
}
}
}
function onPageEvent(e) {
pageData = e.detail;
togglePageListener(false);
}
async function buildStyle() {
if (!pageData) pageData = await (await fetch(apiUrl)).json();
({style, badKeys} = await API.uso.toUsercss(pageData, {varsUrl: dup.updateUrl}));
Object.assign(style, {
md5Url,
id: dup.id,
originalMd5: md5,
updateUrl: apiUrl,
});
}
function getPageVars() {
const {vars} = (style || dup).usercssData;
for (const el of document.querySelectorAll('[name^="ik-"]')) {
const name = el.name.slice(3); // dropping "ik-"
const ik = (badKeys || {})[name] || name;
const v = vars[ik] || false;
const isImage = el.type === 'radio';
if (v && (!isImage || el.checked)) {
const val = el.value;
const isFile = val === 'user-upload';
if (isImage && (isFile || val === 'user-url')) {
const el2 = $(`[type=${isFile ? 'file' : 'url'}]`, el.parentElement);
const ikCust = `${ik}-custom`;
v.value = `${ikCust}-dropdown`;
vars[ikCust].value = isFile ? getFileUriFromPage(el2) : el2.value;
} else {
v.value = v.type === 'select' ? val.replace(/^ik-/, '') : val;
}
}
}
return vars;
}
function getFileUriFromPage(el) {
togglePageListener(true);
sendPageEvent(el);
return pageData;
}
function runInPage(fn, ...args) {
const div = document.createElement('div');
div.attachShadow({mode: 'closed'})
.appendChild(document.createElement('script'))
.textContent = `(${fn})(${JSON.stringify(args).slice(1, -1)})`;
document.documentElement.appendChild(div).remove();
}
function sendPageEvent(data) {
dispatchEvent(data instanceof Node
? new MouseEvent(pageEventId, {relatedTarget: data})
: new CustomEvent(pageEventId, {detail: data}));
//* global cloneInto */// WARNING! Firefox requires cloning of an object `detail`
}
function sendStylishEvent(type) {
document.dispatchEvent(new Event(type));
}
function sendVarsToPage(style) {
if (style) {
const vars = (style.usercssData || {}).vars || `${style.updateUrl}`.split('?')[1];
if (vars) sendPageEvent('vars:' + JSON.stringify(vars));
}
return style || false;
}
function orphanCheck() {
if (chrome.runtime.id) return true;
removeEventListener(orphanEventId, orphanCheck, true);
removeEventListener('click', onClick, true);
removeEventListener('change', onChange);
sendPageEvent('quit');
observeColors(false);
togglePageListener(false);
}
})();
function inPageContext(eventId, eventIdHost, styleId, apiUrl) {
let done, orphaned, vars;
// `chrome` may be empty if no extensions use externally_connectable but USO needs it
if (!window.chrome) window.chrome = {runtime: {sendMessage: () => {}}};
const EXT_ID = 'fjnbnpbmkenffdnngjfgmeleoegfcffe';
const {defineProperty} = Object;
const {dispatchEvent, CustomEvent, removeEventListener} = window;
const apply = Map.call.bind(Map.apply);
const OVR = [
[chrome.runtime, 'sendMessage', (fn, me, args) => {
const [id, /*msg*/, opts, cb = opts] = args;
if (id !== EXT_ID) return apply(fn, me, args);
if (typeof cb !== 'function') return Promise.resolve(true);
cb(true);
}],
[Response.prototype, 'json', async (fn, me, args) => {
const res = await apply(fn, me, args);
try {
if (!done && me.url === apiUrl) {
done = true;
send(res);
setVars(res);
}
} catch (e) {}
return res;
}],
[window, 'fetch', (fn, me, args) =>
args[0] === `chrome-extension://${EXT_ID}/index.html`
? Promise.resolve(new Response('<!doctype html><html lang="en"></html>'))
: apply(fn, me, args),
],
];
OVR.forEach(([obj, name, caller], i) => {
const orig = obj[name];
const ovr = new Proxy(orig, {
apply(fn, me, args) {
if (orphaned) restore(obj, name, ovr, fn);
return (orphaned ? apply : caller)(fn, me, args);
},
});
defineProperty(obj, name, {value: ovr});
OVR[i] = [obj, name, ovr, orig]; // same args as restore()
});
/* We set `isInstalled` at page start intentionally not trying to replicate Stylish login events.
* This difference allows USO site to detect presence of Stylus (or another similar extension). */
window.isInstalled = true;
addEventListener(eventId, onCommand, true);
function onCommand(e) {
if (e.detail === 'quit') {
removeEventListener(eventId, onCommand, true);
OVR.forEach(restore);
done = orphaned = true;
} else if (/^vars:/.test(e.detail)) {
vars = JSON.parse(e.detail.slice(5));
} else if (e.relatedTarget) {
send(e.relatedTarget.uploadedData);
}
}
function restore(obj, name, ovr, orig) { // same order as OVR after patching
if (obj[name] === ovr) {
defineProperty(obj, name, {value: orig});
}
}
function send(data) {
dispatchEvent(new CustomEvent(eventIdHost, {__proto: null, detail: data}));
}
function setVars(json) {
const images = new Map();
const isNew = typeof vars === 'object';
const badKeys = {};
const newKeys = [];
const makeKey = ({install_key: key}) => {
let res = isNew ? badKeys[key] : key;
if (!res) {
res = key.replace(/[^-\w]/g, '-');
res += newKeys.includes(res) ? '-' : '';
if (key !== res) {
badKeys[key] = res;
newKeys.push(res);
}
}
return res;
};
if (!isNew) vars = new URLSearchParams(vars);
for (const ss of json.style_settings || []) {
const ik = makeKey(ss);
let value = isNew ? (vars[ik] || {}).value : vars.get('ik-' + ik);
if (value == null || !(ss.style_setting_options || [])[0]) {
continue;
}
if (ss.setting_type === 'image') {
let isListed;
for (const opt of ss.style_setting_options) {
isListed |= opt.default = (opt.install_key === value);
}
images.set(ik, {url: isNew && !isListed ? vars[`${ik}-custom`].value : value, isListed});
} else if (value.startsWith('ik-') || isNew && vars[ik].type === 'select') {
value = value.replace(/^ik-/, '');
const def = ss.style_setting_options.find(item => item.default);
if (!def || makeKey(def) !== value) {
if (def) def.default = false;
for (const item of ss.style_setting_options) {
if (makeKey(item) === value) {
item.default = true;
break;
}
}
}
} else {
const item = ss.style_setting_options[0];
if (item.value !== value && item.install_key === 'placeholder') {
item.value = value;
}
}
}
if (!images.size) return;
new MutationObserver((_, observer) => {
if (!document.getElementById('style-settings')) return;
observer.disconnect();
for (const [name, {url, isListed}] of images) {
const elRadio = document.querySelector(`input[name="ik-${name}"][value="user-url"]`);
const elUrl = elRadio && document.getElementById(elRadio.id.replace('url-choice', 'user-url'));
if (elUrl) {
elRadio.checked = !isListed;
elUrl.value = url;
}
}
}).observe(document, {childList: true, subtree: true});
}
}

View File

@ -58,12 +58,6 @@
"checkingForUpdate": { "checkingForUpdate": {
"message": "جارٍ البحث..." "message": "جارٍ البحث..."
}, },
"confirmDelete": {
"message": "حذف"
},
"confirmSave": {
"message": "حفظ"
},
"deleteStyleConfirm": { "deleteStyleConfirm": {
"message": "هل تريد بالتأكيد حذف هذا النمط؟" "message": "هل تريد بالتأكيد حذف هذا النمط؟"
}, },
@ -76,9 +70,6 @@
"disableStyleLabel": { "disableStyleLabel": {
"message": "تعطيل" "message": "تعطيل"
}, },
"editDeleteText": {
"message": "حذف"
},
"editStyleHeading": { "editStyleHeading": {
"message": "تعديل النمط" "message": "تعديل النمط"
}, },
@ -96,11 +87,8 @@
"enableStyleLabel": { "enableStyleLabel": {
"message": "تمكين" "message": "تمكين"
}, },
"genericAdd": { "findStylesForSite": {
"message": "إضافة" "message": "البحث عن المزيد من الأنماط لموقع الويب هذا"
},
"genericEnabledLabel": {
"message": "ممكّن"
}, },
"helpAlt": { "helpAlt": {
"message": "مساعدة" "message": "مساعدة"
@ -117,9 +105,6 @@
"openManage": { "openManage": {
"message": "إدارة الأنماط المثبتة" "message": "إدارة الأنماط المثبتة"
}, },
"optionsSyncUrl": {
"message": "عنوان URL"
},
"sectionAdd": { "sectionAdd": {
"message": "إضافة قسم آخر" "message": "إضافة قسم آخر"
}, },
@ -129,9 +114,6 @@
"sectionRemove": { "sectionRemove": {
"message": "إزالة القسم" "message": "إزالة القسم"
}, },
"sections": {
"message": "الأقسام"
},
"styleCancelEditLabel": { "styleCancelEditLabel": {
"message": "رجوع للإدارة" "message": "رجوع للإدارة"
}, },

View File

@ -52,6 +52,9 @@
"backupButtons": { "backupButtons": {
"message": "Резервни копия" "message": "Резервни копия"
}, },
"backupMessage": {
"message": "Изберете файл или го влачете до страницата."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Изнасяне на стилове" "message": "Изнасяне на стилове"
}, },
@ -112,9 +115,6 @@
"confirmOK": { "confirmOK": {
"message": "Добре" "message": "Добре"
}, },
"confirmSave": {
"message": "Запазване"
},
"confirmStop": { "confirmStop": {
"message": "Спиране" "message": "Спиране"
}, },
@ -171,15 +171,12 @@
"exportLabel": { "exportLabel": {
"message": "Изнасяне" "message": "Изнасяне"
}, },
"genericAdd": { "findStylesForSite": {
"message": "Добавяне" "message": "Още стилове за този сайт"
}, },
"genericDisabledLabel": { "genericDisabledLabel": {
"message": "Изключено" "message": "Изключено"
}, },
"genericEnabledLabel": {
"message": "Включено"
},
"genericHistoryLabel": { "genericHistoryLabel": {
"message": "Хронология" "message": "Хронология"
}, },
@ -263,6 +260,9 @@
"manageFaviconsGray": { "manageFaviconsGray": {
"message": "Сиви" "message": "Сиви"
}, },
"manageFaviconsHelp": {
"message": "Разширението използва външна услуга https://www.google.com/s2/favicons"
},
"manageFilters": { "manageFilters": {
"message": "Филтри" "message": "Филтри"
}, },
@ -299,9 +299,6 @@
"openManage": { "openManage": {
"message": "Управление" "message": "Управление"
}, },
"openOptions": {
"message": "Настройки"
},
"openStylesManager": { "openStylesManager": {
"message": "Управление на стиловете" "message": "Управление на стиловете"
}, },
@ -368,9 +365,6 @@
"optionsSubheading": { "optionsSubheading": {
"message": "Още настройки" "message": "Още настройки"
}, },
"optionsSyncUrl": {
"message": "Адрес"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "При внасянето на резервни копия от стари версии или от Стайлиш направете ръчна проверка за обновления, за да сте сигурни, че стиловете са актуални." "message": "При внасянето на резервни копия от стари версии или от Стайлиш направете ръчна проверка за обновления, за да сте сигурни, че стиловете са актуални."
}, },
@ -407,9 +401,6 @@
"sectionRemove": { "sectionRemove": {
"message": "Премахване на отдела" "message": "Премахване на отдела"
}, },
"sections": {
"message": "Отдели"
},
"shortcuts": { "shortcuts": {
"message": "Клавишни комбинации" "message": "Клавишни комбинации"
}, },
@ -463,6 +454,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Брой на неприложените отдели поради неправилно използване на регулярни изрази" "message": "Брой на неприложените отдели поради неправилно използване на регулярни изрази"
}, },
"styleRegexpTestButton": {
"message": "Тест на регулярния израз"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Съвпадащи подпрозорци" "message": "Съвпадащи подпрозорци"
}, },

View File

@ -67,6 +67,9 @@
"backupButtons": { "backupButtons": {
"message": "Zálohovat" "message": "Zálohovat"
}, },
"backupMessage": {
"message": "Vyberte soubor nebo ho přetáhněte na tuto stránku."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Exportovat styly" "message": "Exportovat styly"
}, },
@ -272,6 +275,15 @@
"findStyles": { "findStyles": {
"message": "Najít styly" "message": "Najít styly"
}, },
"findStylesForSite": {
"message": "Najít styly pro tento web"
},
"findStylesInline": {
"message": "Zobrazit zde"
},
"findStylesInlineTooltip": {
"message": "Zobrazit výsledky vyhledávání v tomto okně."
},
"genericAdd": { "genericAdd": {
"message": "Přidat" "message": "Přidat"
}, },
@ -461,7 +473,7 @@
"message": "Zešednutí" "message": "Zešednutí"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus používá externí službu https://icons.duckduckgo.com" "message": "Stylus používá externí službu https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtry" "message": "Filtry"
@ -505,9 +517,6 @@
"menuShowBadge": { "menuShowBadge": {
"message": "Zobrazit počet aktivních stylů" "message": "Zobrazit počet aktivních stylů"
}, },
"meta_invalidCheckboxDefault": {
"message": "Neplatný @var checkbox: hodnota musí být 0 nebo 1"
},
"meta_invalidNumber": { "meta_invalidNumber": {
"message": "Očekáváno číslo" "message": "Očekáváno číslo"
}, },
@ -553,9 +562,6 @@
"openManage": { "openManage": {
"message": "Spravovat" "message": "Spravovat"
}, },
"openOptions": {
"message": "Možnosti"
},
"openStylesManager": { "openStylesManager": {
"message": "Otevřít správce stylů" "message": "Otevřít správce stylů"
}, },
@ -625,9 +631,6 @@
"optionsSubheading": { "optionsSubheading": {
"message": "Další možnosti" "message": "Další možnosti"
}, },
"optionsSyncUrl": {
"message": "URL adresa"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "Importujete-li zálohy stylů ze starší verze nebo z rozšíření Stylish, proveďte jednorázovou ruční kontrolu aktualizací ve správci stylů, aby byly všechny styly aktuální." "message": "Importujete-li zálohy stylů ze starší verze nebo z rozšíření Stylish, proveďte jednorázovou ruční kontrolu aktualizací ve správci stylů, aby byly všechny styly aktuální."
}, },
@ -742,9 +745,6 @@
"sectionRestore": { "sectionRestore": {
"message": "Obnovit odstraněnou sekci" "message": "Obnovit odstraněnou sekci"
}, },
"sections": {
"message": "Sekce"
},
"shortcuts": { "shortcuts": {
"message": "Zkratky" "message": "Zkratky"
}, },
@ -844,6 +844,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Počet sekcí nepoužitých kvůli nesprávnému použití „regexp()“" "message": "Počet sekcí nepoužitých kvůli nesprávnému použití „regexp()“"
}, },
"styleRegexpTestButton": {
"message": "Otestovat RegExp"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Odpovídající listy" "message": "Odpovídající listy"
}, },
@ -912,9 +915,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "Stylus může přistupovat k file:// URL pouze při povolení odpovídající možnosti pro rozšíření Stylus ve správci chrome://extensions." "message": "Stylus může přistupovat k file:// URL pouze při povolení odpovídající možnosti pro rozšíření Stylus ve správci chrome://extensions."
}, },
"unreachableMozSiteHintOldFF": {
"message": "Pouze Firefox 59 a novější může být nakonfigurován tak, aby rozšíření typu WebExtensions mohla přidávat styly na stránky chráněné CSP (Content Security Policy) jako je tato."
},
"unzipStyles": { "unzipStyles": {
"message": "Rozbalování stylů…" "message": "Rozbalování stylů…"
}, },
@ -974,6 +974,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Nahradit výchozí šablonu pro nové Usercss styly aktuálním kódem?" "message": "Nahradit výchozí šablonu pro nové Usercss styly aktuálním kódem?"
}, },
"usercssReplaceTemplateName": {
"message": "Prázdné @name nahrazuje výchozí šablonu"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Sem vložte kód…" "message": "Sem vložte kód…"
}, },

View File

@ -58,6 +58,9 @@
"author": { "author": {
"message": "Forfatter" "message": "Forfatter"
}, },
"backupMessage": {
"message": "Vælg en fil eller træk og slip til denne side."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Eksportér stil" "message": "Eksportér stil"
}, },
@ -93,8 +96,5 @@
}, },
"cm_keyMap": { "cm_keyMap": {
"message": "Tastegenveje" "message": "Tastegenveje"
},
"genericAdd": {
"message": "Tilføj"
} }
} }

View File

@ -68,7 +68,7 @@
"message": "Datensicherung" "message": "Datensicherung"
}, },
"backupMessage": { "backupMessage": {
"message": "Um die Backupdatei zu importieren, ziehe sie in diese Seite oder klicke auf die Import-Schaltfläche.\n\nZum Exportieren einer mit Stylus 1.5.18 (und älter) kompatiblen Backupdatei, rechtsklicke oder Shift-klicke auf die Export-Schaltfläche." "message": "Wähle eine Datei aus oder ziehe die Datei auf diese Seite. (Drag and Drop)"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "Styles exportieren" "message": "Styles exportieren"
@ -223,23 +223,9 @@
"disableAllStyles": { "disableAllStyles": {
"message": "Alle Styles deaktivieren" "message": "Alle Styles deaktivieren"
}, },
"disableAllStylesOff": {
"message": "Styles sind ausgeschaltet"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "Deaktivieren" "message": "Deaktivieren"
}, },
"draftAction": {
"message": "Wähle \"Ja\", um diesen Entwurf zu laden oder \"Nein\", um ihn zu verwerfen."
},
"draftTitle": {
"message": "Wiederherstellung ungespeicherter Entwürfe, erstellt vor $date$",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "Ziehe die Backup Datei zum importieren an irgendeinen Ort auf dieser Seite." "message": "Ziehe die Backup Datei zum importieren an irgendeinen Ort auf dieser Seite."
}, },
@ -266,9 +252,6 @@
} }
} }
}, },
"editorSettings": {
"message": "Editor Einstellungen"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "Aktivieren" "message": "Aktivieren"
}, },
@ -278,9 +261,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "Aktuelle URL ausschließen" "message": "Aktuelle URL ausschließen"
}, },
"exportCompatible": {
"message": "Exportieren (Kompatibilitätsmodus)"
},
"exportLabel": { "exportLabel": {
"message": "Exportieren" "message": "Exportieren"
}, },
@ -313,6 +293,15 @@
"findStyles": { "findStyles": {
"message": "Styles finden" "message": "Styles finden"
}, },
"findStylesForSite": {
"message": "Weitere Styles für diese Seite finden"
},
"findStylesInline": {
"message": "Ergebnisse hier anzeigen"
},
"findStylesInlineTooltip": {
"message": "Suchergebnisse in diesem Fenster anzeigen."
},
"genericAdd": { "genericAdd": {
"message": "Hinzufügen" "message": "Hinzufügen"
}, },
@ -346,9 +335,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "Gespeichert" "message": "Gespeichert"
}, },
"genericSize": {
"message": "Größe"
},
"genericTitle": { "genericTitle": {
"message": "Name" "message": "Name"
}, },
@ -358,9 +344,6 @@
"gettingStyles": { "gettingStyles": {
"message": "Empfange alle Styles..." "message": "Empfange alle Styles..."
}, },
"headerResizerHint": {
"message": "Halte Shift gedrückt, um nur diese Art der Benutzeroberfläche (z.B. Editor, Manager, Installer) zu verändern"
},
"helpAlt": { "helpAlt": {
"message": "Hilfe" "message": "Hilfe"
}, },
@ -535,7 +518,7 @@
"message": "Ausgegraut" "message": "Ausgegraut"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus nutzt hierzu den externen Dienst https://icons.duckduckgo.com" "message": "Stylus nutzt hierzu den externen Dienst https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filter" "message": "Filter"
@ -546,9 +529,6 @@
"manageMaxTargets": { "manageMaxTargets": {
"message": "Anzahl der \"Gilt für\" Elemente" "message": "Anzahl der \"Gilt für\" Elemente"
}, },
"manageMinColumnWidth": {
"message": "Minimale Spaltenbreite (in Pixeln. 9999 deaktiviert den Mehrspalten-Modus)"
},
"manageNewStyleAsUsercss": { "manageNewStyleAsUsercss": {
"message": "als UserCSS" "message": "als UserCSS"
}, },
@ -798,15 +778,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "Erweitert" "message": "Erweitert"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "Nach Systemeinstellung"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "Bei Nacht:"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "Deaktiviert. Die hell / dunkel Einstellung in Styles wird ignoriert."
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "\"Löschen\" im Editor-Kontextmenü hinzufügen" "message": "\"Löschen\" im Editor-Kontextmenü hinzufügen"
}, },
@ -816,17 +787,11 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "Style wirkt sich auch auf iframes der anvisierten (obersten) Domain aus.\nIframe-spezifisches CSS ist dann wie folgt möglich:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }" "message": "Style wirkt sich auch auf iframes der anvisierten (obersten) Domain aus.\nIframe-spezifisches CSS ist dann wie folgt möglich:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }"
}, },
"optionsAdvancedExposeStyleName": {
"message": "Stylename anzeigen"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "Zeigt den Stylenamen auf der Seite an, um das Debuggen von Styles in den Devtools zu erleichtern. Bitte lade die Tabs neu, um die neue Einstellung zu übernehmen."
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "Schreibe neuen Style als UserCSS" "message": "Schreibe neuen Style als UserCSS"
}, },
"optionsAdvancedPatchCsp": { "optionsAdvancedPatchCsp": {
"message": "<code>CSP</code> abändern, um externe Ressourcen zu erlauben" "message": "<code>CSP</code>abändern, um externe Ressourcen zu erlauben"
}, },
"optionsAdvancedPatchCspNote": { "optionsAdvancedPatchCspNote": {
"message": "Aktivieren, wenn Styles Bilder oder Schriftarten enthalten, die aufgrund strenger <code>CSP</code> (<code>Content-Security-Policy</code>) Regeln mancher Seiten nicht laden.\n\nDas Aktivieren wird <code>CSP</code>Beschränkungen lockern, um für den Style erforderliche Ressourcen zu laden. Diese Option ist für fortgeschrittene Benutzer gedacht, die sich den möglichen Sicherheitsrisiken bewusst sind und die die Verantwortung dafür tragen, die nachgeladenen Inhalte selbst zu überwachen. Informiere dich über \"CSS-basierte Angriffe\" um mehr zu erfahren.\n\nBeachte außerdem, dass diese Option nicht garantiert funktioniert, falls eine andere installierte Erweiterung die Netzwerkantwort (CSP-header) zuerst abändert." "message": "Aktivieren, wenn Styles Bilder oder Schriftarten enthalten, die aufgrund strenger <code>CSP</code> (<code>Content-Security-Policy</code>) Regeln mancher Seiten nicht laden.\n\nDas Aktivieren wird <code>CSP</code>Beschränkungen lockern, um für den Style erforderliche Ressourcen zu laden. Diese Option ist für fortgeschrittene Benutzer gedacht, die sich den möglichen Sicherheitsrisiken bewusst sind und die die Verantwortung dafür tragen, die nachgeladenen Inhalte selbst zu überwachen. Informiere dich über \"CSS-basierte Angriffe\" um mehr zu erfahren.\n\nBeachte außerdem, dass diese Option nicht garantiert funktioniert, falls eine andere installierte Erweiterung die Netzwerkantwort (CSP-header) zuerst abändert."
@ -861,9 +826,6 @@
"optionsHeading": { "optionsHeading": {
"message": "Optionen" "message": "Optionen"
}, },
"optionsIconAuto": {
"message": "An Hell- / Dunkelmodus angleichen"
},
"optionsIconDark": { "optionsIconDark": {
"message": "Dunkle Browser-Themes" "message": "Dunkle Browser-Themes"
}, },
@ -886,7 +848,7 @@
"message": "Optionen zurücksetzen" "message": "Optionen zurücksetzen"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "Klicke auf einer beliebigen Stylus Seite auf das Stylus-Symbol in der Werkzeugleiste des Browsers, dann klicke auf \"Styles finden\"" "message": "Stylus UI-Theme suchen"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "Mehr Optionen" "message": "Mehr Optionen"
@ -903,9 +865,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "Nichts" "message": "Nichts"
}, },
"optionsSyncPassword": {
"message": "Passwort"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "Verbunden" "message": "Verbunden"
}, },
@ -949,9 +908,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Jetzt synchronisieren" "message": "Jetzt synchronisieren"
}, },
"optionsSyncUsername": {
"message": "Benutzername"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "Nach dem Importieren von Styles aus einer alten Version oder von Stylish ist eine einmalige manuelle Updatesuche in der Verwaltung nötig. Dies stellt sicher, dass alle Styles auf dem aktuellsten Stand sind." "message": "Nach dem Importieren von Styles aus einer alten Version oder von Stylish ist eine einmalige manuelle Updatesuche in der Verwaltung nötig. Dies stellt sicher, dass alle Styles auf dem aktuellsten Stand sind."
}, },
@ -994,9 +950,6 @@
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "Klicke, um Tastenkürzel zu sehen" "message": "Klicke, um Tastenkürzel zu sehen"
}, },
"popupManageSiteStyles": {
"message": "Styles dieser Seite verwalten"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "Shift+Klick oder Rechtsklick öffnet den Stylemanager mit Filter für Styles der aktuellen Seite" "message": "Shift+Klick oder Rechtsklick öffnet den Stylemanager mit Filter für Styles der aktuellen Seite"
}, },
@ -1018,21 +971,6 @@
"prefShowBadge": { "prefShowBadge": {
"message": "Anzahl der aktiven Styles auf der aktuellen Seite" "message": "Anzahl der aktiven Styles auf der aktuellen Seite"
}, },
"preferScheme": {
"message": "Hell- / Dunkelmodus Vorrang"
},
"preferSchemeAlways": {
"message": "Derzeit ignoriert (Style wird immer angewendet), weil der allgemeine Hell- / Dunkelmodus nicht aktiv ist"
},
"preferSchemeDark": {
"message": "Dunkel"
},
"preferSchemeLight": {
"message": "Hell"
},
"preferSchemeNone": {
"message": "Keines (immer aktiv)"
},
"previewLabel": { "previewLabel": {
"message": "Echtzeitvorschau" "message": "Echtzeitvorschau"
}, },
@ -1061,7 +999,7 @@
"message": "Lese Styles..." "message": "Lese Styles..."
}, },
"reload": { "reload": {
"message": "Neu laden" "message": "Stylus Addon neu laden"
}, },
"replace": { "replace": {
"message": "Ersetzen" "message": "Ersetzen"
@ -1072,15 +1010,9 @@
"replaceWith": { "replaceWith": {
"message": "Ersetzen durch" "message": "Ersetzen durch"
}, },
"restoreTemplate": {
"message": "Standard-Template wiederherstellen.\n\n(Die gerade geöffneten Editorseiten werden nicht verändert.)"
},
"retrieveBckp": { "retrieveBckp": {
"message": "Styles importieren" "message": "Styles importieren"
}, },
"saveAsTemplate": {
"message": "Als Template speichern"
},
"search": { "search": {
"message": "Suche" "message": "Suche"
}, },
@ -1121,7 +1053,7 @@
"message": "Wöchentliche Installationen" "message": "Wöchentliche Installationen"
}, },
"searchStyleQueryHint": { "searchStyleQueryHint": {
"message": "Stylenamen durchsuchen (Groß-Kleinschreibung wird beachtet, sobald ein Großbuchstabe benutzt wird):\nHaus Baum Sonne - sucht nach allen Wörtern in beliebiger Reihenfolge\n\"Baum Haus\" - sucht exakt diesen Ausdruck (ohne Anführungszeichen)\n/foo.*bar/i - Regulärer Ausdruck ohne Leerzeichen (nutze stattdessen \\s)" "message": "Stylenamen ohne Beachtung der Groß-/Kleinschreibung suchen:\nMehrere Suchworte - alle Wörter in beliebiger Reihenfolge\n\"Bestimmte Phrase\" - genau diese Phrase ohne Anführungszeichen\n2020 - zeigt auch Styles, die 2020 aktualisiert wurden"
}, },
"searchStylesAll": { "searchStylesAll": {
"message": "Alles" "message": "Alles"
@ -1150,18 +1082,12 @@
"sections": { "sections": {
"message": "Bereiche" "message": "Bereiche"
}, },
"settings": {
"message": "Einstellungen"
},
"shortcuts": { "shortcuts": {
"message": "Tastenkürzel" "message": "Tastenkürzel"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "Eine Tastenkombination definieren" "message": "Eine Tastenkombination definieren"
}, },
"shortcutsNoteFF": {
"message": "In Firefox 66+ kannst du die eingebaute Tastenkürzelverwaltung selbst öffnen:\n1) Rechtsklicke das Stylus-Symbol in der Werkzeugleiste und wähle \"Erweiterung verwalten\". Alternativ kannst du about:addons über das Hauptmenü oder per Tastenkürzel Strg-Umschalt-A öffnen.\n2) Klicke auf der geöffneten Seite (about:addons) auf das Zahnrad oben rechts.\n3) Wähle \"Tastenkombinationen von Erweiterungen verwalten\".\n\nHier kannst du alle Tastenkürzel individuell anpassen."
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "neueste zuerst" "message": "neueste zuerst"
}, },
@ -1207,30 +1133,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "Aktiviert" "message": "Aktiviert"
}, },
"styleExcludeLabel": {
"message": "Benutzerdefinierte ausgeschlossene URLs"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "Import vom Mozilla Format fehlgeschlagen" "message": "Import vom Mozilla Format fehlgeschlagen"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "Mozilla-Format Code einfügen" "message": "Mozilla-Format Code einfügen"
}, },
"styleIncludeLabel": {
"message": "Benutzerdefinierte Ziel-URLs"
},
"styleInjectionImportance": {
"message": "Style als wichtig markieren"
},
"styleInjectionOrder": {
"message": "Applikationsreihenfolge der Styles"
},
"styleInjectionOrderHint": {
"message": "Ziehe den Style mit der Maus an die gewünschte Stelle, um die Applikationsreihenfolge zu ändern. Styles weiter unten können die oberen überschreiben."
},
"styleInjectionOrderHint_prio": {
"message": "Die unten als wichtig gekennzeichneten Styles werden immer zuletzt angewendet, sodass sie auch neu installierte Styles überschreiben können. Klicke auf das Zeichen des Styles, um die Wichtigkeit umzuschalten."
},
"styleInstall": { "styleInstall": {
"message": "\"$stylename$\" mit Stylus installieren?", "message": "\"$stylename$\" mit Stylus installieren?",
"placeholders": { "placeholders": {
@ -1270,15 +1178,6 @@
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "Der Style wurde aufgrund ungültiger RegExp nicht angewandt." "message": "Der Style wurde aufgrund ungültiger RegExp nicht angewandt."
}, },
"styleNotAppliedSchemeDark": {
"message": "Der Style wird nur im dunklen Modus angewendet"
},
"styleNotAppliedSchemeLight": {
"message": "Der Style wird nur im hellen Modus angewendet"
},
"stylePreferSchemeLabel": {
"message": "Dunkler/Heller Modus"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "Einige RegExp konnten nicht kompiliert werden." "message": "Einige RegExp konnten nicht kompiliert werden."
}, },
@ -1288,6 +1187,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Anzahl der Bereiche, welche aufgrund nicht korrekt verwendeter RegExp nicht angewendet wurden" "message": "Anzahl der Bereiche, welche aufgrund nicht korrekt verwendeter RegExp nicht angewendet wurden"
}, },
"styleRegexpTestButton": {
"message": "RegExp testen"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Zutreffende Tabs" "message": "Zutreffende Tabs"
}, },
@ -1309,9 +1211,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "Speichern" "message": "Speichern"
}, },
"styleSettings": {
"message": "Style Einstellungen"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "Das Mozilla-Format des Codes kann mit Stylish für Firefox verwendet werden und bei userstyles.org eingereicht werden." "message": "Das Mozilla-Format des Codes kann mit Stylish für Firefox verwendet werden und bei userstyles.org eingereicht werden."
}, },
@ -1333,7 +1232,7 @@
"message": "Stylus funktioniert nicht auf Seiten wie diesen." "message": "Stylus funktioniert nicht auf Seiten wie diesen."
}, },
"stylusUnavailableForURLdetails": { "stylusUnavailableForURLdetails": {
"message": "Als Sicherheitsvorkehrung verbietet der Browser es Erweiterungen, seine eigenen Seiten (wie z.B. chrome://version oder about:addons) und eigene Seiten anderer Erweiterungen zu verändern. Auch die jeweils browser-eigene Erweiterungsgalerie (wie Chrome Web Store oder addons.mozilla.org) kann nicht verändert werden." "message": "Als Sicherheitsvorkehrung verbietet der Browser es Erweiterungen, seine eingebauten Seiten (wie chrome://version, dem neuen Standard-Tab ab Chrome 61, about:addons, usw.), sowie Seiten anderer Erweiterungen zu beeinflussen. Jeder Browser beschränkt auch den Zugriff auf seine eigene Erweiterungsgalerie (wie Chrome Web Store oder addons.mozilla.org)."
}, },
"syncDropboxDeprecated": { "syncDropboxDeprecated": {
"message": "Dropbox Import / Export wurde durch einen fortschrittlicheren Mechanismus auf der Optionsseite ersetzt." "message": "Dropbox Import / Export wurde durch einen fortschrittlicheren Mechanismus auf der Optionsseite ersetzt."
@ -1341,16 +1240,8 @@
"syncError": { "syncError": {
"message": "Synchronisation fehlgeschlagen" "message": "Synchronisation fehlgeschlagen"
}, },
"syncErrorLock": {
"message": "Die Datenbank wird bereits verwendet. Die Sperre wird um $TIME$ aufgehoben.",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "Synchronisation erfolglos. Du wurdest ausgeloggt.\nVersuche, dich in den Stylus Einstellungen wieder einzuloggen." "message": "Synchronisation fehlgeschlagen.\nVersuche, dich in den Optionen neu einzuloggen:\nKlicke erst \"Trennen\", dann \"Verbinden\"."
}, },
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "Der Wert kann nicht gespeichert werden. Versuche, die Textmenge zu reduzieren." "message": "Der Wert kann nicht gespeichert werden. Versuche, die Textmenge zu reduzieren."
@ -1441,6 +1332,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Ersetze das vorgegebene Template für neue UserCSS styles mit dem vorliegenden Code?" "message": "Ersetze das vorgegebene Template für neue UserCSS styles mit dem vorliegenden Code?"
}, },
"usercssReplaceTemplateName": {
"message": "Ein leeres @name ersetzt das vorgegebene Template"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Quelltext hier eingeben..." "message": "Quelltext hier eingeben..."
}, },

View File

@ -64,6 +64,9 @@
"backupButtons": { "backupButtons": {
"message": "Δημιουργήστε αντίγραφο ασφαλείας" "message": "Δημιουργήστε αντίγραφο ασφαλείας"
}, },
"backupMessage": {
"message": "Επιλέξτε ένα αρχείο ή σύρετέ το σε αυτήν τη σελίδα"
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Εξαγωγή στυλ" "message": "Εξαγωγή στυλ"
}, },
@ -298,6 +301,9 @@
"findStyles": { "findStyles": {
"message": "Εύρεση στυλ" "message": "Εύρεση στυλ"
}, },
"findStylesForSite": {
"message": "Αναζήτηση περισσότερων στυλ για αυτή την ιστοσελίδα"
},
"genericAdd": { "genericAdd": {
"message": "Προσθήκη" "message": "Προσθήκη"
}, },
@ -409,6 +415,9 @@
"linterResetMessage": { "linterResetMessage": {
"message": "Για αναίρεση μιας κατά λάθος επαναφοράς, πατήστε Ctrl-Z (ή Cmd-Z) στο πλαίσιο κειμένου" "message": "Για αναίρεση μιας κατά λάθος επαναφοράς, πατήστε Ctrl-Z (ή Cmd-Z) στο πλαίσιο κειμένου"
}, },
"manageFaviconsHelp": {
"message": "Το Stylus χρησιμοποιεί μία εξωτερική υπηρεσία https://www.google.com/s2/favicons"
},
"manageFilters": { "manageFilters": {
"message": "Φίλτρα" "message": "Φίλτρα"
}, },
@ -529,9 +538,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Συγχρονισμός τώρα" "message": "Συγχρονισμός τώρα"
}, },
"optionsSyncUrl": {
"message": "διεύθυνση URL"
},
"optionsUpdateInterval": { "optionsUpdateInterval": {
"message": "Διάστημα αυτόματης ενημέρωσης των στυλ σε ώρες (0 για απενεργοποίηση)" "message": "Διάστημα αυτόματης ενημέρωσης των στυλ σε ώρες (0 για απενεργοποίηση)"
}, },
@ -604,9 +610,6 @@
"sectionRemove": { "sectionRemove": {
"message": "Αφαίρεση ενότητας" "message": "Αφαίρεση ενότητας"
}, },
"sections": {
"message": "Ενότητες"
},
"shortcuts": { "shortcuts": {
"message": "Συντομεύσεις" "message": "Συντομεύσεις"
}, },
@ -677,6 +680,9 @@
"syncError": { "syncError": {
"message": "Ο συγχρονισμός απέτυχε" "message": "Ο συγχρονισμός απέτυχε"
}, },
"syncErrorRelogin": {
"message": "Ο συγχρονισμός απέτυχε.\nΠροσπαθήστε να συνδεθείτε ξανά στις επιλογές Stylus:\nκάντε κλικ στο 'αποσύνδεση' πρώτα και μετά στο 'σύνδεση'."
},
"toggleStyle": { "toggleStyle": {
"message": "Αλλαγή στυλ" "message": "Αλλαγή στυλ"
}, },

View File

@ -93,8 +93,8 @@
"description": "Heading for backup" "description": "Heading for backup"
}, },
"backupMessage": { "backupMessage": {
"message": "To import the backup file, drag'n'drop it into this page or click the Import button.\n\nTo export a compatible backup for Stylus older than 1.5.18, right-click or shift-click the Export button.", "message": "Select a file or drag and drop to this page.",
"description": "Text for Backup section's (i) in the manager" "description": "Message for backup"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "Export styles" "message": "Export styles"
@ -186,10 +186,6 @@
"message": "Theme", "message": "Theme",
"description": "Label for the style editor's CSS theme." "description": "Label for the style editor's CSS theme."
}, },
"cm_arrowKeysTraverse": {
"message": "Arrow keys ↑↓ traverse sections",
"description": "Label for the option in the editor."
},
"colorpickerPaletteHint": { "colorpickerPaletteHint": {
"message": "Right-click a swatch to cycle through its source lines" "message": "Right-click a swatch to cycle through its source lines"
}, },
@ -340,29 +336,12 @@
}, },
"disableAllStyles": { "disableAllStyles": {
"message": "Turn all styles off", "message": "Turn all styles off",
"description": "Label for the checkbox that turns all styles off." "description": "Label for the checkbox that turns all enabled styles off."
},
"disableAllStylesOff": {
"message": "Styles are turned off",
"description": "Label for the checkbox that turns all styles off when it's checked."
}, },
"disableStyleLabel": { "disableStyleLabel": {
"message": "Disable", "message": "Disable",
"description": "Label for the button to disable a style" "description": "Label for the button to disable a style"
}, },
"draftTitle": {
"message": "Draft recovery, created $date$",
"placeholders": {
"date": {
"content": "$1"
}
},
"description": "Title of the modal displayed in the editor when an unsaved draft is found, the $date$ looks like '1 hour ago' in user's current UI language"
},
"draftAction": {
"message": "Choose 'Yes' to load this draft or 'No' to discard it.",
"description": "Displayed in the editor after the browser/extension crashed"
},
"dragDropMessage": { "dragDropMessage": {
"message": "Drop your backup file anywhere on this page to import.", "message": "Drop your backup file anywhere on this page to import.",
"description": "Drag'n'drop message" "description": "Drag'n'drop message"
@ -396,8 +375,11 @@
}, },
"description": "Title of the page for editing styles" "description": "Title of the page for editing styles"
}, },
"editorSettings": { "editorCodeLabel": {
"message": "Editor settings" "message": "Code"
},
"editorSettingLabel": {
"message": "Settings"
}, },
"enableStyleLabel": { "enableStyleLabel": {
"message": "Enable", "message": "Enable",
@ -413,9 +395,6 @@
"message": "Export", "message": "Export",
"description": "Label for the button to export a style ('edit' page) or all styles ('manage' page)" "description": "Label for the button to export a style ('edit' page) or all styles ('manage' page)"
}, },
"exportCompatible": {
"message": "Export (compatible mode)"
},
"exportSavedSuccess": { "exportSavedSuccess": {
"message": "File saved with success" "message": "File saved with success"
}, },
@ -459,6 +438,18 @@
"message": "Find styles", "message": "Find styles",
"description": "Text for a link that gets a list of styles for the current site" "description": "Text for a link that gets a list of styles for the current site"
}, },
"findStylesForSite": {
"message": "Find more styles for this site",
"description": "Text for a link that gets a list of styles for the current site"
},
"findStylesInline": {
"message": "Inline",
"description": "Text for a checkbox that opens search results 'inline' (within the Stylus popup window)"
},
"findStylesInlineTooltip": {
"message": "Display search results inside this window.",
"description": "Text for a checkbox that displays search results within the Stylus popup."
},
"genericAdd": { "genericAdd": {
"message": "Add", "message": "Add",
"description": "Used in various places for an action that adds something" "description": "Used in various places for an action that adds something"
@ -502,13 +493,6 @@
"message": "Saved", "message": "Saved",
"description": "Used in various parts of the UI to indicate that something was saved" "description": "Used in various parts of the UI to indicate that something was saved"
}, },
"genericSize": {
"message": "Size"
},
"genericTest": {
"message": "Test",
"description": "Label for the action that runs some test e.g. opens the regexp tester panel in the editor"
},
"genericTitle": { "genericTitle": {
"message": "Title", "message": "Title",
"description": "Used in various parts of the UI to indicate the title of something" "description": "Used in various parts of the UI to indicate the title of something"
@ -520,10 +504,6 @@
"gettingStyles": { "gettingStyles": {
"message": "Getting all styles..." "message": "Getting all styles..."
}, },
"headerResizerHint": {
"message": "Hold Shift to resize only in this type of UI, i.e. editor, manager, installer",
"description": "Tooltip for the header panel resizer"
},
"helpAlt": { "helpAlt": {
"message": "Help", "message": "Help",
"description": "Alternate text for help buttons" "description": "Alternate text for help buttons"
@ -622,6 +602,18 @@
"message": "Update style", "message": "Update style",
"description": "Label for update button" "description": "Label for update button"
}, },
"installPreferSchemeLabel": {
"message": "The style should be applied:"
},
"installPreferSchemeNone": {
"message": "Always"
},
"installPreferSchemeDark": {
"message": "In Dark Mode"
},
"installPreferSchemeLight": {
"message": "In Light Mode"
},
"installUpdate": { "installUpdate": {
"message": "Install update", "message": "Install update",
"description": "Label for the button to install an update for a single style" "description": "Label for the button to install an update for a single style"
@ -751,7 +743,7 @@
"description": "Label for the checkbox that toggles grayed out mode of applies-to favicons in the new UI on manage page" "description": "Label for the checkbox that toggles grayed out mode of applies-to favicons in the new UI on manage page"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus uses an external service https://icons.duckduckgo.com", "message": "Stylus uses an external service https://www.google.com/s2/favicons",
"description": "Label for the checkbox that toggles applies-to favicons in the new UI on manage page" "description": "Label for the checkbox that toggles applies-to favicons in the new UI on manage page"
}, },
"manageFilters": { "manageFilters": {
@ -766,9 +758,6 @@
"message": "Number of applies-to items", "message": "Number of applies-to items",
"description": "Label for the numeric input box to limit max number of applies-to targets in the new UI on manage page" "description": "Label for the numeric input box to limit max number of applies-to targets in the new UI on manage page"
}, },
"manageMinColumnWidth": {
"message": "Minimum column width (in pixels; 9999 disables multi-column mode)"
},
"manageNewStyleAsUsercss": { "manageNewStyleAsUsercss": {
"message": "as Usercss", "message": "as Usercss",
"description": "VERY SHORT label for the checkbox next to the 'Write new style' button in the style manager" "description": "VERY SHORT label for the checkbox next to the 'Write new style' button in the style manager"
@ -1079,17 +1068,14 @@
"message": "Exposes the top site domain in each iframe.\nEnables writing iframe-specific CSS like this:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }", "message": "Exposes the top site domain in each iframe.\nEnables writing iframe-specific CSS like this:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }",
"description": "Add attribute to iframe; make sure to include the double $$ in the css example, or the `$=` will be omitted in the displayed text." "description": "Add attribute to iframe; make sure to include the double $$ in the css example, or the `$=` will be omitted in the displayed text."
}, },
"optionsAdvancedExposeStyleName": {
"message": "Expose style name"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "Exposes the style name in the page to facilitate debugging of styles in devtools. Please reload the tab(s) to apply the new setting."
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "Write new style as usercss" "message": "Write new style as usercss"
}, },
"optionsAdvancedAutoSwitchScheme": {
"message": "Toggle Light/Dark Mode styles automatically"
},
"optionsAdvancedAutoSwitchSchemeNever": { "optionsAdvancedAutoSwitchSchemeNever": {
"message": "Disabled. The dark/light setting in styles is ignored." "message": "Never"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": { "optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "By system preference" "message": "By system preference"
@ -1140,9 +1126,6 @@
"message": "Options", "message": "Options",
"description": "Heading for options section on manage page." "description": "Heading for options section on manage page."
}, },
"optionsIconAuto": {
"message": "Match the Dark/Light mode"
},
"optionsIconDark": { "optionsIconDark": {
"message": "Dark browser themes" "message": "Dark browser themes"
}, },
@ -1165,7 +1148,7 @@
"message": "Reset options" "message": "Reset options"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "Click Stylus icon in the browser toolbar on any Stylus page including this one, then click 'Find styles'" "message": "Find a Stylus UI theme"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "More Options", "message": "More Options",
@ -1285,10 +1268,6 @@
"message": "Click to see available hotkeys", "message": "Click to see available hotkeys",
"description": "Tooltip displayed when hovering the right edge of the extension popup" "description": "Tooltip displayed when hovering the right edge of the extension popup"
}, },
"popupManageSiteStyles": {
"message": "Manage site styles",
"description": "Item in the dropdown menu for the 'Manage' button in the popup that opens manager with styles applicable for current site."
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "Shift-click or right-click opens manager with styles applicable for current site", "message": "Shift-click or right-click opens manager with styles applicable for current site",
"description": "Tooltip for the 'Manage' button in the popup." "description": "Tooltip for the 'Manage' button in the popup."
@ -1313,21 +1292,6 @@
"message": "Styles before commands", "message": "Styles before commands",
"description": "Label for the checkbox controlling section order in the popup." "description": "Label for the checkbox controlling section order in the popup."
}, },
"preferScheme": {
"message": "Dark/Light mode preference"
},
"preferSchemeAlways": {
"message": "Currently ignored (the style always applies) because the global Dark/Light mode is disabled"
},
"preferSchemeDark": {
"message": "Dark"
},
"preferSchemeLight": {
"message": "Light"
},
"preferSchemeNone": {
"message": "None (always applied)"
},
"prefShowBadge": { "prefShowBadge": {
"message": "Number of styles active for the current site", "message": "Number of styles active for the current site",
"description": "Label for the checkbox controlling toolbar badge text." "description": "Label for the checkbox controlling toolbar badge text."
@ -1366,8 +1330,8 @@
"message": "Reading styles..." "message": "Reading styles..."
}, },
"reload": { "reload": {
"message": "Reload", "message": "Reload Stylus extension",
"description": "Context menu to reload the extension when installed in developer mode" "description": "Context menu reload"
}, },
"replace": { "replace": {
"message": "Replace", "message": "Replace",
@ -1381,18 +1345,12 @@
"message": "Replace with", "message": "Replace with",
"description": "Label before the replace-with input field in the editor shown on Ctrl-H etc." "description": "Label before the replace-with input field in the editor shown on Ctrl-H etc."
}, },
"restoreTemplate": {
"message": "Restore the default template.\n\n(The currently open editor pages won't change.)"
},
"retrieveBckp": { "retrieveBckp": {
"message": "Import styles" "message": "Import styles"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "Dropbox Import" "message": "Dropbox Import"
}, },
"saveAsTemplate": {
"message": "Save as template"
},
"search": { "search": {
"message": "Search", "message": "Search",
"description": "Label before the search input field in the editor shown on Ctrl-F" "description": "Label before the search input field in the editor shown on Ctrl-F"
@ -1444,7 +1402,7 @@
"description": "Text for label that shows the number of times a search result was installed during last week" "description": "Text for label that shows the number of times a search result was installed during last week"
}, },
"searchStyleQueryHint": { "searchStyleQueryHint": {
"message": "Search style names (case-sensitively if an uppercase letter is used):\nsome words - all these words in any order\n\"some phrase\" - this exact phrase without quotes\n/foo.*bar/i - regular expression without spaces (use \\s instead)", "message": "Search style names case-insensitively:\nsome words - all words in any order\n\"some phrase\" - this exact phrase without quotes\n2020 - a year like this also shows styles updated in 2020",
"description": "Tooltip shown for the text input in the popup's inline style finder" "description": "Tooltip shown for the text input in the popup's inline style finder"
}, },
"searchStylesAll": { "searchStylesAll": {
@ -1491,10 +1449,6 @@
"message": "Sections", "message": "Sections",
"description": "Header for the table of contents block listing style section names in the left panel of the classic editor" "description": "Header for the table of contents block listing style section names in the left panel of the classic editor"
}, },
"settings": {
"message": "Settings",
"description": "Generic label/title for settings"
},
"shortcuts": { "shortcuts": {
"message": "Shortcuts", "message": "Shortcuts",
"description": "Go to shortcut configuration" "description": "Go to shortcut configuration"
@ -1502,9 +1456,6 @@
"shortcutsNote": { "shortcutsNote": {
"message": "Define keyboard shortcuts" "message": "Define keyboard shortcuts"
}, },
"shortcutsNoteFF": {
"message": "In Firefox 66+ you can open the built-in shortcuts UI manually:\n1) right-click Stylus icon in the toolbar and choose 'Manage'\n(alternatively, open about:addons via the main menu or Ctrl-Shift-A),\n2) in the page that opens click the cog wheel icon in the top right corner,\n3) choose 'Manage extension shortcuts'.\n\nYou can also customize the shortcuts here."
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "newest first", "message": "newest first",
"description": "Text added to indicate that sorting a date would add the newest entries at the top" "description": "Text added to indicate that sorting a date would add the newest entries at the top"
@ -1637,6 +1588,10 @@
"message": "Number of sections not applied due to incorrect usage of 'regexp()'", "message": "Number of sections not applied due to incorrect usage of 'regexp()'",
"description": "Tooltip in the popup for styles that were applied only partially" "description": "Tooltip in the popup for styles that were applied only partially"
}, },
"styleRegexpTestButton": {
"message": "RegExp test",
"description": "RegExp test button label in the editor shown when applies-to list has a regexp value"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Matching tabs", "message": "Matching tabs",
"description": "RegExp test report: label for the fully matching expressions" "description": "RegExp test report: label for the fully matching expressions"
@ -1665,10 +1620,6 @@
"message": "Save", "message": "Save",
"description": "Label for save button for style editing" "description": "Label for save button for style editing"
}, },
"styleSettings": {
"message": "Style settings",
"description": "Label/title for style settings dialog"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "The Mozilla format of the code can be submitted to userstyles.org and used with the classic Stylish for Firefox", "message": "The Mozilla format of the code can be submitted to userstyles.org and used with the classic Stylish for Firefox",
"description": "Help info for the Mozilla format header section that converts the code to/from Mozilla format" "description": "Help info for the Mozilla format header section that converts the code to/from Mozilla format"
@ -1698,6 +1649,9 @@
"message": "As a security precaution, the browser prohibits extensions from affecting its built-in pages (like chrome://version, the standard new tab page as of Chrome 61, about:addons, and so on) as well as other extensions' pages. Each browser also restricts access to its own extensions gallery (like Chrome Web Store or AMO).", "message": "As a security precaution, the browser prohibits extensions from affecting its built-in pages (like chrome://version, the standard new tab page as of Chrome 61, about:addons, and so on) as well as other extensions' pages. Each browser also restricts access to its own extensions gallery (like Chrome Web Store or AMO).",
"description": "Sub-note in the toolbar pop-up when on a URL Stylus can't affect" "description": "Sub-note in the toolbar pop-up when on a URL Stylus can't affect"
}, },
"styleOriginLabel": {
"message": "Style origin"
},
"styleUpdateUrlLabel": { "styleUpdateUrlLabel": {
"message": "Update URL" "message": "Update URL"
}, },
@ -1707,21 +1661,6 @@
"styleIncludeLabel": { "styleIncludeLabel": {
"message": "Custom included sites" "message": "Custom included sites"
}, },
"styleInjectionImportance": {
"message": "Toggle style's importance"
},
"styleInjectionOrder": {
"message": "Style injection order",
"description": "Tooltip for the button in the manager to open the dialog and also the title of this dialog"
},
"styleInjectionOrderHint": {
"message": "Drag'n'drop a style to change its position. Styles are injected sequentially in the order shown below so a style further down the list can override the earlier styles.",
"description": "Hint in the injection order dialog in the manager"
},
"styleInjectionOrderHint_prio": {
"message": "Important styles listed below are always injected last so they can override any newly installed styles. Click the style's mark to toggle its importance.",
"description": "Hint at the bottom of the injection order dialog in the manager"
},
"styleExcludeLabel": { "styleExcludeLabel": {
"message": "Custom excluded sites" "message": "Custom excluded sites"
}, },
@ -1859,6 +1798,10 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Replace the default template for new Usercss styles with the current code?" "message": "Replace the default template for new Usercss styles with the current code?"
}, },
"usercssReplaceTemplateName": {
"message": "Empty @name replaces the default template",
"description": "The text shown after @name when creating a new Usercss style"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Insert code here...", "message": "Insert code here...",
"description": "The code placeholder comment in a new style created by clicking 'Write style' in the popup" "description": "The code placeholder comment in a new style created by clicking 'Write style' in the popup"

View File

@ -67,6 +67,9 @@
"backupButtons": { "backupButtons": {
"message": "Respaldo" "message": "Respaldo"
}, },
"backupMessage": {
"message": "Al exportar hará una copia de respaldo de TODOS los estilos instalados. Para restaurar una copia de respaldo, importe el archivo o arrástrelo a esta página."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Exportar estilos" "message": "Exportar estilos"
}, },
@ -310,6 +313,15 @@
"findStyles": { "findStyles": {
"message": "Buscar estilos" "message": "Buscar estilos"
}, },
"findStylesForSite": {
"message": "Buscar más estilos para este sitio"
},
"findStylesInline": {
"message": "Dentro"
},
"findStylesInlineTooltip": {
"message": "Muestra los resultados de búsqueda dentro de esta ventana."
},
"genericAdd": { "genericAdd": {
"message": "Añadir" "message": "Añadir"
}, },
@ -514,7 +526,7 @@
"message": "Atenuado" "message": "Atenuado"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus usa un servicio externo https://icons.duckduckgo.com" "message": "Stylus usa un servicio externo https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtros" "message": "Filtros"
@ -849,6 +861,9 @@
"optionsResetButton": { "optionsResetButton": {
"message": "Restablecer opciones" "message": "Restablecer opciones"
}, },
"optionsStylusThemes": {
"message": "Buscar un tema de Stylus"
},
"optionsSubheading": { "optionsSubheading": {
"message": "Más opciones" "message": "Más opciones"
}, },
@ -997,6 +1012,9 @@
"readingStyles": { "readingStyles": {
"message": "Leyendo los estilos..." "message": "Leyendo los estilos..."
}, },
"reload": {
"message": "Recargar extensión Stylus"
},
"replace": { "replace": {
"message": "Reemplazar" "message": "Reemplazar"
}, },
@ -1051,6 +1069,9 @@
"searchResultWeeklyCount": { "searchResultWeeklyCount": {
"message": "Instalaciones semanales" "message": "Instalaciones semanales"
}, },
"searchStyleQueryHint": {
"message": "La búsqueda de nombres de estilo no distingue mayúsculas de minúsculas:\nalgunas palabras: todas las palabras en cualquier orden\n\"una frase\": esta frase exacta (sin comillas)\n2020: al escribir un año, se muestran estilos que se actualizaron ese año"
},
"searchStylesAll": { "searchStylesAll": {
"message": "Todos" "message": "Todos"
}, },
@ -1133,7 +1154,7 @@
"message": "Has realizado cambios en este estilo sin guardarlos." "message": "Has realizado cambios en este estilo sin guardarlos."
}, },
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "Habilitado" "message": "Activado"
}, },
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "No se pudo importar desde el formato Mozilla" "message": "No se pudo importar desde el formato Mozilla"
@ -1192,6 +1213,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Número de secciones no aplicadas debido a un uso incorrecto de 'regexp()'" "message": "Número de secciones no aplicadas debido a un uso incorrecto de 'regexp()'"
}, },
"styleRegexpTestButton": {
"message": "Probar regexp"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Pestañas coincidentes" "message": "Pestañas coincidentes"
}, },
@ -1245,6 +1269,9 @@
"syncError": { "syncError": {
"message": "No se pudo sincronizar" "message": "No se pudo sincronizar"
}, },
"syncErrorRelogin": {
"message": "No se pudo sincronizar.\nIntenta volver a iniciar sesión en las opciones de Stylus:\nprimero haz clic en 'desconectar', y luego en 'conectar'."
},
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "El valor no se puede guardar. Trate de reducir la longitud del texto." "message": "El valor no se puede guardar. Trate de reducir la longitud del texto."
}, },
@ -1334,6 +1361,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "¿Reemplazar con el código actual la plantilla predeterminada para nuevos estilos UserCSS?" "message": "¿Reemplazar con el código actual la plantilla predeterminada para nuevos estilos UserCSS?"
}, },
"usercssReplaceTemplateName": {
"message": "@name vacío reemplaza la plantilla predeterminada"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Inserte el código aquí..." "message": "Inserte el código aquí..."
}, },

View File

@ -67,6 +67,9 @@
"backupButtons": { "backupButtons": {
"message": "Varunda" "message": "Varunda"
}, },
"backupMessage": {
"message": "Vali fail või lohista see siia lehele."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Ekspordi stiilid" "message": "Ekspordi stiilid"
}, },
@ -290,6 +293,15 @@
"findStyles": { "findStyles": {
"message": "Leia stiile" "message": "Leia stiile"
}, },
"findStylesForSite": {
"message": "Leia sellele saidile veel stiile"
},
"findStylesInline": {
"message": "Aknasisene"
},
"findStylesInlineTooltip": {
"message": "Näita otsingu tulemusi selles aknas."
},
"genericAdd": { "genericAdd": {
"message": "Lisa" "message": "Lisa"
}, },
@ -485,7 +497,7 @@
"message": "Tee halliks" "message": "Tee halliks"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus kasutab välist teenust https://icons.duckduckgo.com" "message": "Stylus kasutab välist teenust https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtrid" "message": "Filtrid"
@ -865,9 +877,6 @@
"sectionRestore": { "sectionRestore": {
"message": "Taasta eemaldatud jaotis" "message": "Taasta eemaldatud jaotis"
}, },
"sections": {
"message": "Jaotised"
},
"shortcuts": { "shortcuts": {
"message": "Otseteed" "message": "Otseteed"
}, },
@ -970,6 +979,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "\"regexp()\" vale kasutuse tõttu rakendamata jaotiste arv" "message": "\"regexp()\" vale kasutuse tõttu rakendamata jaotiste arv"
}, },
"styleRegexpTestButton": {
"message": "Regulaaravaldise katsetus"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Vastavad kaardid" "message": "Vastavad kaardid"
}, },
@ -1044,9 +1056,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "Stylus saab ligi pääseda file:// URLidele ainult siis, kui märgistad vastava kasti Stylus laiendusel chrome://extensions lehel" "message": "Stylus saab ligi pääseda file:// URLidele ainult siis, kui märgistad vastava kasti Stylus laiendusel chrome://extensions lehel"
}, },
"unreachableMozSiteHintOldFF": {
"message": "Ainult Firefox 59 ja uuem on võimalik seadistada lubama WebExtensionsil lisada stiilielemente CSP-kaitstud saitidele nagu see."
},
"unzipStyles": { "unzipStyles": {
"message": "Stiilide lahtipakkimine..." "message": "Stiilide lahtipakkimine..."
}, },
@ -1106,6 +1115,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Asendad vaikimisi malli uutes kasutajacss stiilides praeguse koodiga?" "message": "Asendad vaikimisi malli uutes kasutajacss stiilides praeguse koodiga?"
}, },
"usercssReplaceTemplateName": {
"message": "Tühi @name asendab vaikimisi malli"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Sisesta kood siia..." "message": "Sisesta kood siia..."
}, },

View File

@ -52,12 +52,6 @@
"checkingForUpdate": { "checkingForUpdate": {
"message": "Tarkistetaan..." "message": "Tarkistetaan..."
}, },
"confirmDelete": {
"message": "Poista"
},
"confirmSave": {
"message": "Tallenna"
},
"deleteStyleConfirm": { "deleteStyleConfirm": {
"message": "Oletko varma että haluat poistaa tämän tyylin?" "message": "Oletko varma että haluat poistaa tämän tyylin?"
}, },
@ -70,9 +64,6 @@
"disableStyleLabel": { "disableStyleLabel": {
"message": "Poista Käytöstä" "message": "Poista Käytöstä"
}, },
"editDeleteText": {
"message": "Poista"
},
"editStyleHeading": { "editStyleHeading": {
"message": "Muokkaa Tyyliä" "message": "Muokkaa Tyyliä"
}, },
@ -90,11 +81,8 @@
"enableStyleLabel": { "enableStyleLabel": {
"message": "Aktivoi" "message": "Aktivoi"
}, },
"genericAdd": { "findStylesForSite": {
"message": "Lisää" "message": "Hae lisää tyylejä tälle sivustolle"
},
"genericEnabledLabel": {
"message": "Aktivoitu"
}, },
"helpAlt": { "helpAlt": {
"message": "Apu" "message": "Apu"
@ -129,9 +117,6 @@
"sectionRemove": { "sectionRemove": {
"message": "Poista osio" "message": "Poista osio"
}, },
"sections": {
"message": "Osiot"
},
"styleBadRegexp": { "styleBadRegexp": {
"message": "Regexp ei kelpaa." "message": "Regexp ei kelpaa."
}, },

View File

@ -68,7 +68,7 @@
"message": "Sauvegarde" "message": "Sauvegarde"
}, },
"backupMessage": { "backupMessage": {
"message": "Pour importer le fichier de sauvegarder, glissez-déposez-le sur dans page ou cliquez sur le bouton Importer.\n\nPour export une sauvegarde compatible avec une version de Stylus plus ancienne que 1.5.18, effectuez un clic droit ou un appuie sur la touche Maj + clic sur le bouton Exporter." "message": "Sélectionner un fichier ou le glisser-déposer sur cette page"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "Exporter des styles" "message": "Exporter des styles"
@ -199,28 +199,6 @@
"copy": { "copy": {
"message": "Copier dans le presse-papier" "message": "Copier dans le presse-papier"
}, },
"customNameHint": {
"message": "Entrez un nom personnalisé ici pour renommer le style dans l'IU sans casser les mises à jour"
},
"customNameResetHint": {
"message": "Arrête d'utiliser le nom personnalisé, change vers le propre nom du style"
},
"dateAbbrDay": {
"message": "$value$j",
"placeholders": {
"value": {
"content": "$1"
}
}
},
"dateAbbrYear": {
"message": "$value$a",
"placeholders": {
"value": {
"content": "$1"
}
}
},
"dateInstalled": { "dateInstalled": {
"message": "Date d'installation" "message": "Date d'installation"
}, },
@ -245,23 +223,9 @@
"disableAllStyles": { "disableAllStyles": {
"message": "Désactiver tous les styles" "message": "Désactiver tous les styles"
}, },
"disableAllStylesOff": {
"message": "Les styles sont désactivés"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "Désactiver" "message": "Désactiver"
}, },
"draftAction": {
"message": "Choisissez « Oui » pour charger ce brouillon ou « Non » pour l'abandonner."
},
"draftTitle": {
"message": "Sauvegarde du brouillon, créée $date$",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "Glisser votre fichier de sauvegarde nimporte où sur cette page pour limporter." "message": "Glisser votre fichier de sauvegarde nimporte où sur cette page pour limporter."
}, },
@ -288,9 +252,6 @@
} }
} }
}, },
"editorSettings": {
"message": "Paramètres de léditeur"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "Activer" "message": "Activer"
}, },
@ -300,9 +261,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "Exclure lURL actuelle" "message": "Exclure lURL actuelle"
}, },
"exportCompatible": {
"message": "Export (mode de compatibilité)"
},
"exportLabel": { "exportLabel": {
"message": "Exporter" "message": "Exporter"
}, },
@ -338,6 +296,15 @@
"findStyles": { "findStyles": {
"message": "Trouver des styles" "message": "Trouver des styles"
}, },
"findStylesForSite": {
"message": "Rechercher d'autres styles pour ce site"
},
"findStylesInline": {
"message": "Dans la popup"
},
"findStylesInlineTooltip": {
"message": "Affiche les résultats de recherche dans cette fenêtre"
},
"genericAdd": { "genericAdd": {
"message": "Ajouter" "message": "Ajouter"
}, },
@ -377,9 +344,6 @@
"gettingStyles": { "gettingStyles": {
"message": "Récupération de tous les styles…" "message": "Récupération de tous les styles…"
}, },
"headerResizerHint": {
"message": "Maintenir la touche Maj pour redimensionner uniquement dans ce type d'interface, par exemple, éditeur, gestionnaire, installateur"
},
"helpAlt": { "helpAlt": {
"message": "Aide" "message": "Aide"
}, },
@ -401,12 +365,6 @@
"importLabel": { "importLabel": {
"message": "Importer" "message": "Importer"
}, },
"importPreprocessor": {
"message": "Les styles avec un <code>@preprocessor</code> ne fonctionnera pas en mode classique. Vous pouvez changer l'éditeur en mode Usercss : 1) Ouvrez le gestionnaire de styles, 2) Activer l'option \"en tant que Usercss\", 3) Cliquez \"écrire un nouveau style\"\n\nImporter de toute façon ?"
},
"importPreprocessorTitle": {
"message": "Problème potentiel à cause de @preprocessor"
},
"importReplaceLabel": { "importReplaceLabel": {
"message": "Remplacer le style" "message": "Remplacer le style"
}, },
@ -475,18 +433,9 @@
"linkGetHelp": { "linkGetHelp": {
"message": "Consulter l'aide" "message": "Consulter l'aide"
}, },
"linkGetShareStyles": {
"message": "Obtenir et partager les styles"
},
"linkGetShareStylesInfo": {
"message": "Le nouveau userstyles.world géré par la communauté est créé par les auteurs d'userstyles dans le but de remplacer userstyles.org, qui a été tellement lent et inconscient durant l'année dernière que beaucoup d'auteurs ont arrêté de mettre à jour leurs styles."
},
"linkGetStyles": { "linkGetStyles": {
"message": "Obtenir des styles" "message": "Obtenir des styles"
}, },
"linkGetStylesInfo": {
"message": "Ce site d'archive à été créé par un membre de la communauté userstyle pour archiver le lent et inconscient site userstyles.org. Cette archive est mise à jour approximativement chaque jour."
},
"linkTranslate": { "linkTranslate": {
"message": "Traduire" "message": "Traduire"
}, },
@ -554,7 +503,7 @@
"message": "Grisés" "message": "Grisés"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus utilise le service externe https://icons.duckduckgo.com" "message": "Stylus utilise le service externe https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtres" "message": "Filtres"
@ -760,17 +709,6 @@
} }
} }
}, },
"meta_unknownMetaTypo": {
"message": "Peut-être @$keyOk$? Métadonnée inconnue: @$keyErr$",
"placeholders": {
"keyErr": {
"content": "$1"
},
"keyOk": {
"content": "$2"
}
}
},
"meta_unknownPreprocessor": { "meta_unknownPreprocessor": {
"message": "@preprocessor inconnu: $preprocessor$", "message": "@preprocessor inconnu: $preprocessor$",
"placeholders": { "placeholders": {
@ -796,9 +734,6 @@
"noStylesForSite": { "noStylesForSite": {
"message": "Aucun style n'est installé pour ce site." "message": "Aucun style n'est installé pour ce site."
}, },
"numberedLine": {
"message": "Ligne :"
},
"openManage": { "openManage": {
"message": "Gestion" "message": "Gestion"
}, },
@ -808,15 +743,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "Avancé" "message": "Avancé"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "Par préférence système"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "Par horaire nocturne :"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "Désactivé. Le paramètre clair/obscur dans les styles est ignoré."
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "Ajouter « Supprimer » dans le menu contextuel de lextension" "message": "Ajouter « Supprimer » dans le menu contextuel de lextension"
}, },
@ -826,24 +752,9 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "Expose le domaine principal dans chaque iframe.\nPermet décrire du CSS spécifique aux iframe tel que :\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }" "message": "Expose le domaine principal dans chaque iframe.\nPermet décrire du CSS spécifique aux iframe tel que :\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }"
}, },
"optionsAdvancedExposeStyleName": {
"message": "Afficher le nom du style"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "Afficher le nom du style dans la page pour faciliter le débogage des styles dans les outils de développement. Veuillez recharger le ou les onglets pour appliquer ce nouveau paramètre."
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "Écrire un nouveau style en tant que usercss" "message": "Écrire un nouveau style en tant que usercss"
}, },
"optionsAdvancedPatchCsp": {
"message": "Corrige <code>CSP</code> pour permettre les ressources de style"
},
"optionsAdvancedPatchCspNote": {
"message": "Activez cela si les styles comportent des images ou des polices qui échoue à charger sur les sites avec une politique de sécurité de contenus stricte <code>CSP</code> (<code>Content-Security-Policy</code>).\n\nActiver cette option permettra d'alléger les restrictions du <code>CSP</code>, permettant les styles essentiels de charger. Cette option n'est entendue que pour les utilisateurs expérimentés qui comprennent les potentielles implications de sécurité que cela introduit, et accepte la responsabilité de surveiller le contenu qu'ils autorisent. Veuillez lire les attaques basé sur le CSS pour plus dinformation.\n\nAussi sachez, que cette option en particulier n'est pas garanti de fonctionner si une autre extension modifie la réponse réseau avant."
},
"optionsAdvancedStyleViaXhr": {
"message": "Mode d'injection instantanée"
},
"optionsBadgeDisabled": { "optionsBadgeDisabled": {
"message": "Couleur d'arrière plan si désactivé" "message": "Couleur d'arrière plan si désactivé"
}, },
@ -904,9 +815,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "Aucun" "message": "Aucun"
}, },
"optionsSyncPassword": {
"message": "Mot de passe"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "Connecté" "message": "Connecté"
}, },
@ -941,18 +849,12 @@
} }
} }
}, },
"optionsSyncStatusRelogin": {
"message": "Session expirée, veuillez vous reconnecter à nouveau."
},
"optionsSyncStatusSyncing": { "optionsSyncStatusSyncing": {
"message": "Synchronisation..." "message": "Synchronisation..."
}, },
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Synchroniser maintenant" "message": "Synchroniser maintenant"
}, },
"optionsSyncUsername": {
"message": "Nom dutilisateur"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "Quand vous importez des sauvegardes de style dune ancienne version ou de Stylish, faites une vérification manuellement pour vous assurez que tous les styles sont à jour." "message": "Quand vous importez des sauvegardes de style dune ancienne version ou de Stylish, faites une vérification manuellement pour vous assurez que tous les styles sont à jour."
}, },
@ -1001,9 +903,6 @@
"popupMenuButtonTooltip": { "popupMenuButtonTooltip": {
"message": "Actions" "message": "Actions"
}, },
"popupOpenEditInPopup": {
"message": "Utiliser une simple fenêtre (pas d'omnibox)"
},
"popupOpenEditInWindow": { "popupOpenEditInWindow": {
"message": "Ouvrir l'éditeur dans une nouvelle fenêtre" "message": "Ouvrir l'éditeur dans une nouvelle fenêtre"
}, },
@ -1016,41 +915,17 @@
"prefShowBadge": { "prefShowBadge": {
"message": "Afficher le nombre de styles actifs pour le site actuel sur le boutton Stylus" "message": "Afficher le nombre de styles actifs pour le site actuel sur le boutton Stylus"
}, },
"preferSchemeDark": {
"message": "Obscur"
},
"preferSchemeLight": {
"message": "Clair"
},
"preferSchemeNone": {
"message": "Aucun (quand appliqué)"
},
"previewLabel": { "previewLabel": {
"message": "Prévisualiser en direct" "message": "Prévisualiser en direct"
}, },
"previewTooltip": { "previewTooltip": {
"message": "Applique temporairement les changements sans sauvegarder.\nSauvegarde le style pour rendre les changements permanents." "message": "Applique temporairement les changements sans sauvegarder.\nSauvegarde le style pour rendre les changements permanents."
}, },
"publish": {
"message": "Publier"
},
"publishPush": {
"message": "Pousser la mise à jour"
},
"publishReconnect": {
"message": "Essayez de vous déconnecter et de publier à nouveau"
},
"publishStyle": {
"message": "Publier le style"
},
"publishUsw": {
"message": "Utiliser <userstyles.world>"
},
"readingStyles": { "readingStyles": {
"message": "Lecture des styles…" "message": "Lecture des styles…"
}, },
"reload": { "reload": {
"message": "Recharger" "message": "Recharger l´extension Stylus"
}, },
"replace": { "replace": {
"message": "Remplacer" "message": "Remplacer"
@ -1067,18 +942,12 @@
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "Importer depuis Dropbox" "message": "Importer depuis Dropbox"
}, },
"saveAsTemplate": {
"message": "Sauvegarder comme modèle"
},
"search": { "search": {
"message": "Rechercher" "message": "Rechercher"
}, },
"searchCaseSensitive": { "searchCaseSensitive": {
"message": "Sensible à la casse" "message": "Sensible à la casse"
}, },
"searchGlobalStyles": {
"message": "Chercher aussi des styles globaux."
},
"searchNumberOfResults": { "searchNumberOfResults": {
"message": "Nombre de correspondances" "message": "Nombre de correspondances"
}, },
@ -1094,12 +963,6 @@
"searchResultNoneFound": { "searchResultNoneFound": {
"message": "Aucun style trouvé pour ce site" "message": "Aucun style trouvé pour ce site"
}, },
"searchResultNotMatching": {
"message": "Ce style est installé mais il ne s'applique pas au lien du site actuel."
},
"searchResultNotMatchingNote": {
"message": "Essaye de demander lauteur pour ajouter le lien l'URL.\n\nVous pouvez aussi ouvrir le style dans le gestionnaire et l'éditer soit-même,\nmais soyez avertis que cela désactive les mises à jour automatiques pour ce style."
},
"searchResultRating": { "searchResultRating": {
"message": "Note" "message": "Note"
}, },
@ -1109,21 +972,6 @@
"searchResultWeeklyCount": { "searchResultWeeklyCount": {
"message": "Installations hebdomadaires" "message": "Installations hebdomadaires"
}, },
"searchStylesAll": {
"message": "Tout"
},
"searchStylesCode": {
"message": "Code CSS"
},
"searchStylesMatchUrl": {
"message": "Par lien"
},
"searchStylesMeta": {
"message": "Meta-données"
},
"searchStylesName": {
"message": "Nom"
},
"sectionAdd": { "sectionAdd": {
"message": "Ajouter une section" "message": "Ajouter une section"
}, },
@ -1133,9 +981,6 @@
"sectionRestore": { "sectionRestore": {
"message": "Restaurer la section supprimée" "message": "Restaurer la section supprimée"
}, },
"settings": {
"message": "Paramètres"
},
"shortcuts": { "shortcuts": {
"message": "Raccourcis" "message": "Raccourcis"
}, },
@ -1169,9 +1014,6 @@
"styleBeautify": { "styleBeautify": {
"message": "Embellir " "message": "Embellir "
}, },
"styleBeautifyHint": {
"message": "Astuce : Cliquez-droit le bouton \"Beautify\" ou utilisez le raccourci clavier définit ci-dessous pour beautify sans montrer le panneau."
},
"styleBeautifyIndentConditional": { "styleBeautifyIndentConditional": {
"message": "Indenter @media, @supports" "message": "Indenter @media, @supports"
}, },
@ -1187,9 +1029,6 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "Activé" "message": "Activé"
}, },
"styleExcludeLabel": {
"message": "Personnalisation des sites exclus"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "Échec de l'importation depuis le format Mozilla" "message": "Échec de l'importation depuis le format Mozilla"
}, },
@ -1232,21 +1071,9 @@
"styleMozillaFormatHeading": { "styleMozillaFormatHeading": {
"message": "Format Mozilla" "message": "Format Mozilla"
}, },
"styleName": {
"message": "Nom du style"
},
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "Le style n'a pu s'appliquer en raison d'une utilisation erronée de 'regexp()'" "message": "Le style n'a pu s'appliquer en raison d'une utilisation erronée de 'regexp()'"
}, },
"styleNotAppliedSchemeDark": {
"message": "Ce style est uniquement appliqué en mode sombre"
},
"styleNotAppliedSchemeLight": {
"message": "Ce style est uniquement appliqué en mode clair"
},
"stylePreferSchemeLabel": {
"message": "Mode clair/sombre"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "Quelques règles 'regexp()' qui nont pas pu être compilées" "message": "Quelques règles 'regexp()' qui nont pas pu être compilées"
}, },
@ -1256,6 +1083,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Nombre de sections non appliquées à cause dune utilisation incorrecte de 'regexp()'" "message": "Nombre de sections non appliquées à cause dune utilisation incorrecte de 'regexp()'"
}, },
"styleRegexpTestButton": {
"message": "Test dexpression rationnelle"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Onglets correspondants" "message": "Onglets correspondants"
}, },
@ -1277,9 +1107,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "Enregistrer" "message": "Enregistrer"
}, },
"styleSettings": {
"message": "Paramètres du style"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "Le code au format Mozilla peut être utilisé dans Stylish for Firefox et envoyé à userstyles.org." "message": "Le code au format Mozilla peut être utilisé dans Stylish for Firefox et envoyé à userstyles.org."
}, },
@ -1297,9 +1124,6 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "Le style a été changé en dehors de léditeur. Voulez-vous recharger le style ?" "message": "Le style a été changé en dehors de léditeur. Voulez-vous recharger le style ?"
}, },
"styleUpdateUrlLabel": {
"message": "URL de mise à jour"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "Stylus ne fonctionne pas sur les pages de ce genre" "message": "Stylus ne fonctionne pas sur les pages de ce genre"
}, },
@ -1312,17 +1136,6 @@
"syncDropboxStyles": { "syncDropboxStyles": {
"message": "Exporter vers Dropbox" "message": "Exporter vers Dropbox"
}, },
"syncError": {
"message": "Synchronisation échouée"
},
"syncErrorLock": {
"message": "Cette base de données est déjà utilisée. Le verrouillage expirera le $TIME$",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "La valeur ne peut pas être sauvegardée. Essayez de réduire la quantité de texte." "message": "La valeur ne peut pas être sauvegardée. Essayez de réduire la quantité de texte."
}, },
@ -1347,12 +1160,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "Stylus peut accéder aux URL file:// uniquement si vous cochez la case correspondante pour lextension Stylus sur la page chrome://extensions." "message": "Stylus peut accéder aux URL file:// uniquement si vous cochez la case correspondante pour lextension Stylus sur la page chrome://extensions."
}, },
"unreachableMozSiteHint": {
"message": "Dans Firefox 60 et plus récent vous devez retirer ce domaine de <extensions.webextensions.restrictedDomains> dans <about:config>."
},
"unreachableMozSiteHintOldFF": {
"message": "Seulement dans Firefox 59 et plus récent que les extensions-web peuvent être configurés pour autoriser de rajouter des éléments de style sur les sites protégés CSP comme celui-ci."
},
"unzipStyles": { "unzipStyles": {
"message": "Décompression des styles…" "message": "Décompression des styles…"
}, },
@ -1412,6 +1219,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Remplacer le modèle par défaut pour les nouveaux styles Usercss par le code actuel ?" "message": "Remplacer le modèle par défaut pour les nouveaux styles Usercss par le code actuel ?"
}, },
"usercssReplaceTemplateName": {
"message": "Un @name vide remplace le modèle par défaut"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Insérer le code ici..." "message": "Insérer le code ici..."
}, },

View File

@ -67,6 +67,9 @@
"backupButtons": { "backupButtons": {
"message": "גיבוי" "message": "גיבוי"
}, },
"backupMessage": {
"message": "בחר קובץ או גרור ושחרר אותו בדף זה."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "ייצא עיצובים" "message": "ייצא עיצובים"
}, },
@ -293,6 +296,15 @@
"findStyles": { "findStyles": {
"message": "מצא עיצובים" "message": "מצא עיצובים"
}, },
"findStylesForSite": {
"message": "מצא סגנונות נוספים לאתר זה"
},
"findStylesInline": {
"message": "מוטבע"
},
"findStylesInlineTooltip": {
"message": "הצג תוצאות חיפוש בחלון זה."
},
"genericAdd": { "genericAdd": {
"message": "הוספה" "message": "הוספה"
}, },
@ -484,6 +496,9 @@
"manageFaviconsGray": { "manageFaviconsGray": {
"message": "האפרת האייקונים" "message": "האפרת האייקונים"
}, },
"manageFaviconsHelp": {
"message": "Stylus משתמש בשירות חיצוני https://www.google.com/s2/favicons"
},
"manageFilters": { "manageFilters": {
"message": "מסננים" "message": "מסננים"
}, },
@ -688,9 +703,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "סנכרן כעת" "message": "סנכרן כעת"
}, },
"optionsSyncUrl": {
"message": "כתובת אתר"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "בעת ייבוא גיבויים לסגנון מגרסה ישנה או מ־Stylish, בדוק באופן חד פעמי אם יש עדכונים באופן ידני במנהל הסגנונות כדי לוודא שכל הסגנונות מעודכנים." "message": "בעת ייבוא גיבויים לסגנון מגרסה ישנה או מ־Stylish, בדוק באופן חד פעמי אם יש עדכונים באופן ידני במנהל הסגנונות כדי לוודא שכל הסגנונות מעודכנים."
}, },
@ -754,6 +766,9 @@
"readingStyles": { "readingStyles": {
"message": "קורא עיצובים..." "message": "קורא עיצובים..."
}, },
"reload": {
"message": "טען מחדש את Stylus"
},
"replace": { "replace": {
"message": "החלף" "message": "החלף"
}, },
@ -811,9 +826,6 @@
"sectionRestore": { "sectionRestore": {
"message": "שחזר סעיף שהוסר" "message": "שחזר סעיף שהוסר"
}, },
"sections": {
"message": "סעיפים"
},
"shortcuts": { "shortcuts": {
"message": "קיצורי מקשים" "message": "קיצורי מקשים"
}, },
@ -907,6 +919,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "מספר המקטעים שלא הוחלו עקב שימוש שגוי ב־'regexp()'" "message": "מספר המקטעים שלא הוחלו עקב שימוש שגוי ב־'regexp()'"
}, },
"styleRegexpTestButton": {
"message": "בדוק RegExp"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "כרטיסיות תואמות" "message": "כרטיסיות תואמות"
}, },

View File

@ -67,6 +67,9 @@
"backupButtons": { "backupButtons": {
"message": "Biztonsági mentés" "message": "Biztonsági mentés"
}, },
"backupMessage": {
"message": "Válassz ki egy fájlt vagy húzd erre az oldalra!"
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Stílusok exportálása" "message": "Stílusok exportálása"
}, },
@ -287,6 +290,15 @@
"findStyles": { "findStyles": {
"message": "Stílusok keresése" "message": "Stílusok keresése"
}, },
"findStylesForSite": {
"message": "További stílusok keresése ehhez az oldalhoz"
},
"findStylesInline": {
"message": "Helyben"
},
"findStylesInlineTooltip": {
"message": "Keresési eredmények megjelenítése ebben az ablakban."
},
"genericAdd": { "genericAdd": {
"message": "Hozzáadás" "message": "Hozzáadás"
}, },
@ -476,7 +488,7 @@
"message": "Megjelenítés szürkítve" "message": "Megjelenítés szürkítve"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "A Stylus külső szolgáltatást használ (https://icons.duckduckgo.com)" "message": "A Stylus külső szolgáltatást használ (https://www.google.com/s2/favicons)"
}, },
"manageFilters": { "manageFilters": {
"message": "Szűrők" "message": "Szűrők"
@ -710,9 +722,6 @@
"openManage": { "openManage": {
"message": "Kezelés" "message": "Kezelés"
}, },
"openOptions": {
"message": "Beállítások"
},
"openStylesManager": { "openStylesManager": {
"message": "Stíluskezelő megnyitása" "message": "Stíluskezelő megnyitása"
}, },
@ -966,9 +975,6 @@
"sectionRestore": { "sectionRestore": {
"message": "Eltávolított szekció visszaállítása" "message": "Eltávolított szekció visszaállítása"
}, },
"sections": {
"message": "Szekciók"
},
"shortcuts": { "shortcuts": {
"message": "Gyorsbillentyűk" "message": "Gyorsbillentyűk"
}, },
@ -1071,6 +1077,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "A nem alkalmazott szeckiók száma helytelen 'regexp()' használat miatt" "message": "A nem alkalmazott szeckiók száma helytelen 'regexp()' használat miatt"
}, },
"styleRegexpTestButton": {
"message": "RegExp tesztelése"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Illeszkedő fülek" "message": "Illeszkedő fülek"
}, },
@ -1142,9 +1151,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "A Stylus csak akkor képes hozzáférni a file:// URL-ekhez, ha engedélyezed az erre vonatkozó beállítást a Stylus kiegészítőre a chrome://extensions oldalon." "message": "A Stylus csak akkor képes hozzáférni a file:// URL-ekhez, ha engedélyezed az erre vonatkozó beállítást a Stylus kiegészítőre a chrome://extensions oldalon."
}, },
"unreachableMozSiteHintOldFF": {
"message": "Csak a Firefox 59 vagy annál újabb állítható be az ilyen CSP-védett oldalak stílusának átszabása WebExtensions kiegészítőkön keresztül."
},
"unzipStyles": { "unzipStyles": {
"message": "Stílusok kibontása..." "message": "Stílusok kibontása..."
}, },
@ -1204,6 +1210,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Le legyen cserélve az alapértelmezett sablon az új Usercss stílusokhoz a jelenlegi kóddal?" "message": "Le legyen cserélve az alapértelmezett sablon az új Usercss stílusokhoz a jelenlegi kóddal?"
}, },
"usercssReplaceTemplateName": {
"message": "Az üres @name lecseréli az alapértelmezett sablont"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Írj kódot ide…" "message": "Írj kódot ide…"
}, },

View File

@ -1,9 +1,6 @@
{ {
"InaccessibleFileHint": {
"message": "Stylus non può accedere ad alcuni tipi di file (ad es. pdf & JSON)"
},
"addStyleLabel": { "addStyleLabel": {
"message": "Scrivi un nuovo stile" "message": "Scrivi nuovo stile"
}, },
"addStyleTitle": { "addStyleTitle": {
"message": "Aggiungi stili" "message": "Aggiungi stili"
@ -62,7 +59,7 @@
"message": "Autore" "message": "Autore"
}, },
"backupMessage": { "backupMessage": {
"message": "Per importare il file di backup, selezionalo e rilascialo in questa pagina oppure utilizza il bottone Importa\n\nPer esportare un backup compatibile con Stylus versione 1.5.18 e oltre, selezionalo col tasto destro oppure shift-click il pulsante Esporta" "message": "Seleziona un file o trascinalo in questa pagina."
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "Esporta stili" "message": "Esporta stili"
@ -127,9 +124,6 @@
"cm_theme": { "cm_theme": {
"message": "Tema" "message": "Tema"
}, },
"colorpickerSwitchFormatTooltip": {
"message": "Cambia formato: HEX -> RGB -> HSL.\nShift-click per invertire la direzione.\nAnche con i tasti PgUp (PaginaSu), PgDn (PaginaGiù)."
},
"colorpickerTooltip": { "colorpickerTooltip": {
"message": "Apri selettore colore" "message": "Apri selettore colore"
}, },
@ -166,18 +160,6 @@
"confirmYes": { "confirmYes": {
"message": "Si" "message": "Si"
}, },
"connectingDropboxNotAllowed": {
"message": "La connessione a Dropbox è disponibile soltanto nelle app installate direttamente dal webstore"
},
"copied": {
"message": "Copiato negli appunti"
},
"copy": {
"message": "Copia negli appunti"
},
"customNameHint": {
"message": "Inserisci qui un nome personalizzato per rinominare lo stile nell'interfaccia utente (UI) senza rompere i suoi aggiornamenti"
},
"dateInstalled": { "dateInstalled": {
"message": "Data installazione" "message": "Data installazione"
}, },
@ -257,6 +239,12 @@
"findStyles": { "findStyles": {
"message": "Trova stili" "message": "Trova stili"
}, },
"findStylesForSite": {
"message": "Trova più stili per questo sito"
},
"findStylesInlineTooltip": {
"message": "Visualizza risultati in questa finestra."
},
"genericAdd": { "genericAdd": {
"message": "Aggiungi" "message": "Aggiungi"
}, },
@ -411,6 +399,9 @@
"liveReloadLabel": { "liveReloadLabel": {
"message": "Ricaricamento live" "message": "Ricaricamento live"
}, },
"manageFaviconsHelp": {
"message": "Stylus utilizza un servizio esterno https://www.google.com/s2/favicons"
},
"manageFilters": { "manageFilters": {
"message": "Filtri" "message": "Filtri"
}, },
@ -497,9 +488,6 @@
"openManage": { "openManage": {
"message": "Gestisci gli stili installati" "message": "Gestisci gli stili installati"
}, },
"openOptions": {
"message": "Opzioni"
},
"openStylesManager": { "openStylesManager": {
"message": "Apri gestore stili" "message": "Apri gestore stili"
}, },
@ -638,9 +626,6 @@
"sectionRemove": { "sectionRemove": {
"message": "Rimuovi sezione" "message": "Rimuovi sezione"
}, },
"sections": {
"message": "Sezioni"
},
"shortcuts": { "shortcuts": {
"message": "Scorciatoie" "message": "Scorciatoie"
}, },
@ -728,6 +713,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Numero di sezioni non applicato a causa dell'uso errato di 'regexp()'" "message": "Numero di sezioni non applicato a causa dell'uso errato di 'regexp()'"
}, },
"styleRegexpTestButton": {
"message": "Test RegExp"
},
"styleRegexpTestInvalid": { "styleRegexpTestInvalid": {
"message": "Regexp invalida ignorata" "message": "Regexp invalida ignorata"
}, },
@ -766,9 +754,6 @@
"unreachableContentScript": { "unreachableContentScript": {
"message": "Impossibile comunicare con la pagina. Prova a ricaricare la scheda." "message": "Impossibile comunicare con la pagina. Prova a ricaricare la scheda."
}, },
"unreachableMozSiteHintOldFF": {
"message": "Solo Firefox 59 e successivi possono essere configurati per consentire alle WebExtensions di aggiungere elementi degli stili su siti CSP-protected come questo."
},
"updateAllCheckSucceededNoUpdate": { "updateAllCheckSucceededNoUpdate": {
"message": "Nessun aggiornamento trovato" "message": "Nessun aggiornamento trovato"
}, },

View File

@ -68,7 +68,7 @@
"message": "バックアップ" "message": "バックアップ"
}, },
"backupMessage": { "backupMessage": {
"message": "バックアップファイルをインポートする場合、このページにドラッグ&ドロップするか、インポートボタンをクリックしてください。\nまた、Stylus 1.5.18 より古い形式でバックアップしたい場合は、エクスポートボタンを 右クリック もしくは shift-クリック してください。" "message": "ファイルを選択するか、このページにドラッグ&ドロップしてください。"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "スタイルをエクスポート" "message": "スタイルをエクスポート"
@ -226,23 +226,9 @@
"disableAllStyles": { "disableAllStyles": {
"message": "すべてのスタイルをオフにする" "message": "すべてのスタイルをオフにする"
}, },
"disableAllStylesOff": {
"message": "全スタイルをOFFにする"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "無効化" "message": "無効化"
}, },
"draftAction": {
"message": "このデータをロードするならば「はい」を、破棄するならば「いいえ」を選択してください。"
},
"draftTitle": {
"message": "$date$に作成されたデータ",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "このページの任意の場所にバックアップファイルをドロップしてインポートします。" "message": "このページの任意の場所にバックアップファイルをドロップしてインポートします。"
}, },
@ -269,9 +255,6 @@
} }
} }
}, },
"editorSettings": {
"message": "エディタ設定"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "有効化" "message": "有効化"
}, },
@ -281,9 +264,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "現在のURLを除外" "message": "現在のURLを除外"
}, },
"exportCompatible": {
"message": "エクスポート(互換モード)"
},
"exportLabel": { "exportLabel": {
"message": "エクスポート" "message": "エクスポート"
}, },
@ -322,6 +302,15 @@
"findStyles": { "findStyles": {
"message": "スタイルを検索" "message": "スタイルを検索"
}, },
"findStylesForSite": {
"message": "このサイト用のスタイルを検索"
},
"findStylesInline": {
"message": "結果を下に表示"
},
"findStylesInlineTooltip": {
"message": "検索結果をこのウィンドウに表示"
},
"genericAdd": { "genericAdd": {
"message": "追加" "message": "追加"
}, },
@ -355,12 +344,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "保存しました" "message": "保存しました"
}, },
"genericSize": {
"message": "サイズ"
},
"genericTest": {
"message": "テスト"
},
"genericTitle": { "genericTitle": {
"message": "タイトル" "message": "タイトル"
}, },
@ -370,9 +353,6 @@
"gettingStyles": { "gettingStyles": {
"message": "全スタイルを取得中..." "message": "全スタイルを取得中..."
}, },
"headerResizerHint": {
"message": "この種のUIエディタ、管理、インストーラの中では、Shiftを押しながらリサイズしてください"
},
"helpAlt": { "helpAlt": {
"message": "ヘルプ" "message": "ヘルプ"
}, },
@ -550,7 +530,7 @@
"message": "ファビコンをグレー表示" "message": "ファビコンをグレー表示"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylusは外部サービスを使用します https://icons.duckduckgo.com" "message": "Stylusは外部サービスを使用します https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "フィルター" "message": "フィルター"
@ -561,9 +541,6 @@
"manageMaxTargets": { "manageMaxTargets": {
"message": "適用先欄の表示件数" "message": "適用先欄の表示件数"
}, },
"manageMinColumnWidth": {
"message": "最小カラム幅 (ピクセル単位。9999にすると、マルチカラムモードを無効にします)"
},
"manageNewStyleAsUsercss": { "manageNewStyleAsUsercss": {
"message": "Usercssとして" "message": "Usercssとして"
}, },
@ -813,15 +790,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "上級者向け" "message": "上級者向け"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "システム設定による"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "夜間時間による"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "無効。スタイルの暗い/明るい設定は無視されます。"
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "エディタのコンテキストメニューに「削除」を追加する" "message": "エディタのコンテキストメニューに「削除」を追加する"
}, },
@ -831,12 +799,6 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "各iframeにトップサイトドメインを公開します。\n次のようにiframe用のCSSを書くことができるようになります:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }" "message": "各iframeにトップサイトドメインを公開します。\n次のようにiframe用のCSSを書くことができるようになります:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }"
}, },
"optionsAdvancedExposeStyleName": {
"message": "スタイル名を公開"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "開発ツールでスタイルのデバッグを容易にするため、ページ内にスタイル名を公開します。タブをリロードして新しい設定を適用してください。"
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "新しいスタイルを usercss として作成します" "message": "新しいスタイルを usercss として作成します"
}, },
@ -882,9 +844,6 @@
"optionsHeading": { "optionsHeading": {
"message": "オプション" "message": "オプション"
}, },
"optionsIconAuto": {
"message": "暗い/明るいモード設定に合わせます"
},
"optionsIconDark": { "optionsIconDark": {
"message": "暗いブラウザのテーマ" "message": "暗いブラウザのテーマ"
}, },
@ -907,7 +866,7 @@
"message": "オプションをリセット" "message": "オプションをリセット"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "このページを含む任意のStylusのページで、ブラウザのツールバー内のStylusアイコンをクリックして、「スタイルを見つける」をクリックします" "message": "StylusのUIテーマを見つける"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "その他のオプション" "message": "その他のオプション"
@ -924,9 +883,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "未選択" "message": "未選択"
}, },
"optionsSyncPassword": {
"message": "パスワード"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "接続状態" "message": "接続状態"
}, },
@ -970,9 +926,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "すぐに同期" "message": "すぐに同期"
}, },
"optionsSyncUsername": {
"message": "ユーザ名"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "旧バージョンまたはStylishからスタイルのバックアップをインポートする場合は、スタイルマネージャで一度手動で更新チェックを行い、すべてのスタイルが確実に更新されるようにしてください。" "message": "旧バージョンまたはStylishからスタイルのバックアップをインポートする場合は、スタイルマネージャで一度手動で更新チェックを行い、すべてのスタイルが確実に更新されるようにしてください。"
}, },
@ -1015,9 +968,6 @@
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "クリックすると使用可能なホットキーを表示します" "message": "クリックすると使用可能なホットキーを表示します"
}, },
"popupManageSiteStyles": {
"message": "サイトのスタイルを管理"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "シフトを押しながらクリック または 右クリック で、現在のサイトに適用可能なスタイルの管理画面を開きます" "message": "シフトを押しながらクリック または 右クリック で、現在のサイトに適用可能なスタイルの管理画面を開きます"
}, },
@ -1039,21 +989,6 @@
"prefShowBadge": { "prefShowBadge": {
"message": "現在のサイトで有効なスタイルの数を表示" "message": "現在のサイトで有効なスタイルの数を表示"
}, },
"preferScheme": {
"message": "暗い/明るいモード設定"
},
"preferSchemeAlways": {
"message": "グローバルな暗い/明るいモード設定が無効になっているため、現在は無視されます(スタイルは常に適用されます)"
},
"preferSchemeDark": {
"message": "暗い"
},
"preferSchemeLight": {
"message": "明るい"
},
"preferSchemeNone": {
"message": "なし(常に適用)"
},
"previewLabel": { "previewLabel": {
"message": "自動プレビュー" "message": "自動プレビュー"
}, },
@ -1082,7 +1017,7 @@
"message": "スタイルを読み込み中..." "message": "スタイルを読み込み中..."
}, },
"reload": { "reload": {
"message": "リロード" "message": "Stylus拡張をリロードする"
}, },
"replace": { "replace": {
"message": "置換" "message": "置換"
@ -1093,18 +1028,12 @@
"replaceWith": { "replaceWith": {
"message": "次に置換" "message": "次に置換"
}, },
"restoreTemplate": {
"message": "デフォルトのテンプレートに戻す。\n\n現在エディタで開いているものは変化しません。"
},
"retrieveBckp": { "retrieveBckp": {
"message": "スタイルをインポート" "message": "スタイルをインポート"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "Dropboxからインポート" "message": "Dropboxからインポート"
}, },
"saveAsTemplate": {
"message": "テンプレートとして保存"
},
"search": { "search": {
"message": "検索" "message": "検索"
}, },
@ -1145,7 +1074,7 @@
"message": "週間インストール数" "message": "週間インストール数"
}, },
"searchStyleQueryHint": { "searchStyleQueryHint": {
"message": "スタイル名を検索します (大文字を使用すると、大文字小文字を区別して検索します) :\n複数の単語 - すべての単語を (順不同で) 含む\n\"複数の単語からなるフレーズ\" - 正確なフレーズを (引用符は除いて) 探す\n/foo.*bar/i - 正規表現 (空白は使用不可。空白を使いたい場合は、代わりに \\s を使ってください)" "message": "スタイル名を大文字小文字の区別なく検索します:\n複数の単語 - 全単語を順番の区別なく検索します\n\"複数の単語からなるフレーズ\" - 正確なフレーズを (引用符は除いて) 検索します\n2020 - このように数値 (年) を指定すると、2020年に更新されたスタイルも結果に表示します"
}, },
"searchStylesAll": { "searchStylesAll": {
"message": "すべて" "message": "すべて"
@ -1180,18 +1109,12 @@
"sections": { "sections": {
"message": "セクション" "message": "セクション"
}, },
"settings": {
"message": "設定"
},
"shortcuts": { "shortcuts": {
"message": "ショートカット" "message": "ショートカット"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "キーボードショートカットを定義する" "message": "キーボードショートカットを定義する"
}, },
"shortcutsNoteFF": {
"message": "Firefox 66+ では、ビルトインのショートカットUIを手動で開けます。\n1ツールバーのStylusアイコンを右クリックして、「管理」を選択します。\nもしくは、メインメニューでabout:addonsを開くかCtrl-Shift-Aを押します\n2開いたページの右上隅にある歯車のアイコンをクリックします。\n3「拡張機能のショートカットの管理」を選択します。\n\nまた、ショートカットをここで設定することもできます。"
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "新しい順" "message": "新しい順"
}, },
@ -1237,30 +1160,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "有効" "message": "有効"
}, },
"styleExcludeLabel": {
"message": "含めないサイトを設定"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "Mozilla形式のインポートに失敗しました" "message": "Mozilla形式のインポートに失敗しました"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "Mozilla形式のコードを貼り付ける" "message": "Mozilla形式のコードを貼り付ける"
}, },
"styleIncludeLabel": {
"message": "含めるサイトを設定"
},
"styleInjectionImportance": {
"message": "スタイルの重要度を設定/解除する"
},
"styleInjectionOrder": {
"message": "スタイルの適用順"
},
"styleInjectionOrderHint": {
"message": "ドラッグ&ドロップでスタイルの位置を変更できます。スタイルは表示の順番に適用されるため、リストのより下にあるスタイルは、上のものを上書きできます。"
},
"styleInjectionOrderHint_prio": {
"message": "以下に表示される重要なスタイルは、常に最後に挿入されるため、新しくインストールしたスタイルを上書きできます。スタイルをクリックして重要度を設定/解除してください。"
},
"styleInstall": { "styleInstall": {
"message": "「$stylename$」を Stylus にインストールしますか?", "message": "「$stylename$」を Stylus にインストールしますか?",
"placeholders": { "placeholders": {
@ -1300,15 +1205,6 @@
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "'regexp()' の誤った使用のためスタイルを適用できませんでした" "message": "'regexp()' の誤った使用のためスタイルを適用できませんでした"
}, },
"styleNotAppliedSchemeDark": {
"message": "スタイルはダークモード時のみ適用されます"
},
"styleNotAppliedSchemeLight": {
"message": "スタイルはライトモード時のみ適用されます"
},
"stylePreferSchemeLabel": {
"message": "暗い/明るい モード"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "いくつかの 'regexp()' のルールはコンパイルできませんでした" "message": "いくつかの 'regexp()' のルールはコンパイルできませんでした"
}, },
@ -1318,6 +1214,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "'regexp()' の誤った使用のためいくつかのセクションを適用できませんでした" "message": "'regexp()' の誤った使用のためいくつかのセクションを適用できませんでした"
}, },
"styleRegexpTestButton": {
"message": "正規表現をテスト"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "一致するタブ" "message": "一致するタブ"
}, },
@ -1339,9 +1238,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "保存" "message": "保存"
}, },
"styleSettings": {
"message": "スタイル設定"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "Mozilla形式のコードは、userstyles.org に投稿することができ、また従来の Stylish for Firefox でも使用できます" "message": "Mozilla形式のコードは、userstyles.org に投稿することができ、また従来の Stylish for Firefox でも使用できます"
}, },
@ -1359,9 +1255,6 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "このスタイルはエディタの外部で変更されました。このスタイルをリロードしますか?" "message": "このスタイルはエディタの外部で変更されました。このスタイルをリロードしますか?"
}, },
"styleUpdateUrlLabel": {
"message": "更新URL"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "このページではStylusは動作しません。" "message": "このページではStylusは動作しません。"
}, },
@ -1377,16 +1270,8 @@
"syncError": { "syncError": {
"message": "同期に失敗" "message": "同期に失敗"
}, },
"syncErrorLock": {
"message": "データベースが使用中です。\nロック状態は$TIME$に解除されます。",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "同期に失敗しました。ログアウトした可能性があります。\nStylusのオプション画面で再ログインしてください。" "message": "同期に失敗しました。\nStylusのオプション画面で、再ログインしてください :\n「切断」をクリック後、「接続」をクリック"
}, },
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "値を保存できません。テキストの量を減らしてください。" "message": "値を保存できません。テキストの量を減らしてください。"
@ -1477,6 +1362,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "現在のコードで、新しいUsercssスタイルのデフォルト・テンプレートを置き換えますか" "message": "現在のコードで、新しいUsercssスタイルのデフォルト・テンプレートを置き換えますか"
}, },
"usercssReplaceTemplateName": {
"message": "@name が指定されていません"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "ここにコードを挿入..." "message": "ここにコードを挿入..."
}, },

View File

@ -67,6 +67,9 @@
"backupButtons": { "backupButtons": {
"message": "백업" "message": "백업"
}, },
"backupMessage": {
"message": "파일을 선택하거나, 이 페이지로 파일을 드래그앤드롭하세요."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "스타일 내보내기" "message": "스타일 내보내기"
}, },
@ -133,9 +136,6 @@
"cm_theme": { "cm_theme": {
"message": "테마" "message": "테마"
}, },
"colorpickerPaletteHint": {
"message": "견본을 우클릭하여 소스 코드와 번갈아 가며 확인"
},
"colorpickerSwitchFormatTooltip": { "colorpickerSwitchFormatTooltip": {
"message": "16 진수-> RGB -> HSL 순서로 형식을 변환합니다.\nShift+클릭을 이용해 역순으로 변환할 수 있습니다.\nPgUp(PageUp), PgDn(PageDown) 키도 사용 가능합니다." "message": "16 진수-> RGB -> HSL 순서로 형식을 변환합니다.\nShift+클릭을 이용해 역순으로 변환할 수 있습니다.\nPgUp(PageUp), PgDn(PageDown) 키도 사용 가능합니다."
}, },
@ -302,15 +302,21 @@
"findStyles": { "findStyles": {
"message": "스타일 찾기" "message": "스타일 찾기"
}, },
"findStylesForSite": {
"message": "현재 사이트에 맞는 스타일 찾기"
},
"findStylesInline": {
"message": "팝업창 내에서"
},
"findStylesInlineTooltip": {
"message": "이 창에서 검색 결과 표시"
},
"genericAdd": { "genericAdd": {
"message": "추가" "message": "추가"
}, },
"genericClone": { "genericClone": {
"message": "복제" "message": "복제"
}, },
"genericDescription": {
"message": "설명"
},
"genericDisabledLabel": { "genericDisabledLabel": {
"message": "비활성화됨" "message": "비활성화됨"
}, },
@ -439,18 +445,9 @@
"linkGetHelp": { "linkGetHelp": {
"message": "도움말" "message": "도움말"
}, },
"linkGetShareStyles": {
"message": "스타일 주고받기"
},
"linkGetShareStylesInfo": {
"message": "커뮤니티에 의해 새롭게 운영되는 userstyles.world 사이트는 userstyle 투고자들이 userstyle.org를 대체하기 위해 만들었습니다. userstyle.org는 지난 몇 년간 많은 투고자들이 스타일을 업데이트하는 것을 멈춰서 느리고 둔감해졌습니다."
},
"linkGetStyles": { "linkGetStyles": {
"message": "스타일 찾기" "message": "스타일 찾기"
}, },
"linkGetStylesInfo": {
"message": "본 아카이브 사이트는 느리고 응답이 늦는 userstyles.org를 백업하기 위해 Userstyle 커뮤니티 회원에 의해 제작되었습니다. 하루에 약 한 번 아카이브 내 콘텐츠가 갱신됩니다."
},
"linkStylusWiki": { "linkStylusWiki": {
"message": "위키" "message": "위키"
}, },
@ -521,7 +518,7 @@
"message": "회색으로 표시" "message": "회색으로 표시"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus는 https://icons.duckduckgo.com 외부 서비스를 이용합니다" "message": "Stylus는 https://www.google.com/s2/favicons 외부 서비스를 이용합니다"
}, },
"manageFilters": { "manageFilters": {
"message": "필터" "message": "필터"
@ -727,17 +724,6 @@
} }
} }
}, },
"meta_unknownMetaTypo": {
"message": "혹시 @$keyOk$입니까? 알 수 없는 메타데이터: @$keyErr$",
"placeholders": {
"keyErr": {
"content": "$1"
},
"keyOk": {
"content": "$2"
}
}
},
"meta_unknownPreprocessor": { "meta_unknownPreprocessor": {
"message": "알 수 없는 @preprocessor $preprocessor$", "message": "알 수 없는 @preprocessor $preprocessor$",
"placeholders": { "placeholders": {
@ -796,15 +782,9 @@
"optionsAdvancedPatchCsp": { "optionsAdvancedPatchCsp": {
"message": "스타일 자원을 허용하도록 <code>CSP</code>패치하기" "message": "스타일 자원을 허용하도록 <code>CSP</code>패치하기"
}, },
"optionsAdvancedPatchCspNote": {
"message": "엄격한 <code>콘텐츠 보안 정책</code>(<code>Content-Security-Policy; CSP</code>)이 적용된 사이트에서 사진이나 글꼴이 포함된 스타일을 불러올 수 없는 경우 이 설정을 켜십시오.\n\n설정을 켜면 <code>CSP</code> 제한이 풀려, 필수적인 스타일 콘텐츠를 불러올 수 있도록 합니다. 이 설정은 잠재적인 보안 영향을 이해하고, 콘텐츠 모니터링 허용 시 발생하는 문제에 대한 책임을 질 수 있는 사용자에게만 권장됩니다. 자세한 정보를 알고 싶다면 CSS 기반 공격에 대해 알아 보십시오.\n\n또, 이 설정은 설치된 다른 확장 프로그램이 네트워크 응답을 먼저 수정하는 경우 작동하지 않을 수도 있으므로 이 점에 유의하십시오."
},
"optionsAdvancedStyleViaXhr": { "optionsAdvancedStyleViaXhr": {
"message": "즉시 주입 모드" "message": "즉시 주입 모드"
}, },
"optionsAdvancedStyleViaXhrNote": {
"message": "만약 브라우징 도중 스타일이 적용되지 않은 콘텐츠가 깜빡거리며 나오는 경우(FOUC), 특히 다크 테마를 사용하는 중 눈에 두드러지게 깜빡이는 경우 이 설정을 켜십시오.\n\n기술적으로 Chrome/Chromium이 확장 프로그램과의 비동기 통신을 미루기 때문에 발생하는 문제이며, 페이지를 더 빨리 불러오려는 시도이지만 보통은 무의미하고 스타일이 늦게 적용되도록 하는 요인으로 작용합니다. 확장 프로그램에는 이를 회피하기 위한 동기형 API가 제공되지 않으므로, Stylus는 \"비권장\" 상태인 동기형 XMLHttpRquest 웹 API를 사용하여 스타일을 가져오는 설정을 제공합니다. 페이지를 서버에서 내려받는 몇 밀리초 내에 이런 과정이 끝나므로 로드 속도에는 악영향이 없습니다.\n\n그럼에도 불구하고 Chromium 브라우저에서는 개발자 도구의 콘솔에 경고를 출력할 것입니다. 경고를 우클릭하여 숨기면 이후에는 경고가 표시되지 않습니다."
},
"optionsBadgeDisabled": { "optionsBadgeDisabled": {
"message": "비활성화 시의 배경 색" "message": "비활성화 시의 배경 색"
}, },
@ -856,6 +836,9 @@
"optionsResetButton": { "optionsResetButton": {
"message": "설정 초기화" "message": "설정 초기화"
}, },
"optionsStylusThemes": {
"message": "스타일러스 UI 테마 찾기"
},
"optionsSubheading": { "optionsSubheading": {
"message": "추가 옵션" "message": "추가 옵션"
}, },
@ -983,27 +966,12 @@
"previewTooltip": { "previewTooltip": {
"message": "변경 사항은 저장되지 않고 일시적으로 적용됩니다.\n영구적으로 변경하려면 스타일을 저장하세요." "message": "변경 사항은 저장되지 않고 일시적으로 적용됩니다.\n영구적으로 변경하려면 스타일을 저장하세요."
}, },
"publish": {
"message": "배포"
},
"publishPush": {
"message": "업데이트 푸시"
},
"publishReconnect": {
"message": "연결을 해제한 뒤 배포를 다시 시도해보세요"
},
"publishRetry": {
"message": "Stylus에서는 계속 이 스타일을 배포하려고 하고 있습니다. 하지만 인증 창이 뜨지 않는 경우 다시 시도할 수 있습니다. 지금 다시 시도하시겠습니까?"
},
"publishStyle": {
"message": "스타일 배포"
},
"publishUsw": {
"message": "<userstyles.world> 사용"
},
"readingStyles": { "readingStyles": {
"message": "스타일 읽어들이는 중..." "message": "스타일 읽어들이는 중..."
}, },
"reload": {
"message": "Stylus 확장 프로그램 리로드"
},
"replace": { "replace": {
"message": "바꾸기" "message": "바꾸기"
}, },
@ -1064,9 +1032,6 @@
"searchStylesCode": { "searchStylesCode": {
"message": "CSS 코드" "message": "CSS 코드"
}, },
"searchStylesHelp": {
"message": "</> 또는 <Ctrl-F> 키를 누르면 검색창에 포커스합니다.\nDefault mode is plain text search for all space-separated terms in any order.\n정확한 단어 검색: 검색어를 큰따옴표로 감쌉니다. 예: <\".header ~ div\">\n정규표현식: 빗금 및 플래그 문자를 포함합니다. 예: </body.*?\\ba\\b/i>\nURL로 찾기: 완전히 특정된 URL에 적용되는 스타일을 검색합니다. 예: https://www.example.org/\n메타데이터로 찾기: 이름, \"적용 대상\" 지정자, 설치 URL, 업데이트 URL 등 usercss 스타일의 모든 메타데이터 블록을 기반으로 검색합니다."
},
"searchStylesMatchUrl": { "searchStylesMatchUrl": {
"message": "URL별" "message": "URL별"
}, },
@ -1181,9 +1146,6 @@
"styleMozillaFormatHeading": { "styleMozillaFormatHeading": {
"message": "Mozilla 형식" "message": "Mozilla 형식"
}, },
"styleName": {
"message": "스타일 이름"
},
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "잘못된 'regexp()'의 사용으로 인해 스타일을 적용할 수 없습니다" "message": "잘못된 'regexp()'의 사용으로 인해 스타일을 적용할 수 없습니다"
}, },
@ -1196,6 +1158,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "잘못된 'regexp()' 사용으로 인해 몇몇 섹션을 적용할 수 없습니다." "message": "잘못된 'regexp()' 사용으로 인해 몇몇 섹션을 적용할 수 없습니다."
}, },
"styleRegexpTestButton": {
"message": "정규 표현식 테스트"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "일치하는 탭 목록" "message": "일치하는 탭 목록"
}, },
@ -1273,12 +1238,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "Stylus가 file:// URL에 접근할 수 있도록 하려면 chrome://extensions 페이지에서 파일 URL에 대한 액세스를 허용해야 합니다." "message": "Stylus가 file:// URL에 접근할 수 있도록 하려면 chrome://extensions 페이지에서 파일 URL에 대한 액세스를 허용해야 합니다."
}, },
"unreachableMozSiteHint": {
"message": "Firefox 60 이후의 버전에서는 <about:config> 내의 <extensions.webextensions.restrictedDomains>에서 이 도메인을 제거해야 합니다. "
},
"unreachableMozSiteHintOldFF": {
"message": "Firefox 59 버전 이상에서만 이처럼 CSP 보호 정책이 적용된 사이트에 웹 확장 기능으로 스타일 요소를 추가할 수 있도록 설정 가능합니다."
},
"unzipStyles": { "unzipStyles": {
"message": "스타일 압축 푸는 중..." "message": "스타일 압축 푸는 중..."
}, },
@ -1338,6 +1297,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "현재 코드로 새 Usercss 스타일의 기본 템플릿을 대체하시겠습니까?" "message": "현재 코드로 새 Usercss 스타일의 기본 템플릿을 대체하시겠습니까?"
}, },
"usercssReplaceTemplateName": {
"message": "@name을 비우면 기본 템플릿이 대체됩니다."
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "코드를 여기 입력하세요" "message": "코드를 여기 입력하세요"
}, },

View File

@ -62,13 +62,13 @@
"message": "Alle updates toepassen" "message": "Alle updates toepassen"
}, },
"author": { "author": {
"message": "Maker" "message": "Auteur"
}, },
"backupButtons": { "backupButtons": {
"message": "Back-up" "message": "Back-up"
}, },
"backupMessage": { "backupMessage": {
"message": "Sleep het back-upbestand naar deze pagina of klik op de knop Importeren om het te importeren.\n\nRechtsklik of shift-klik op de knop Exporteren om een compatibele back-up voor Stylus ouder dan 1.5.18 te exporteren." "message": "Selecteer een bestand of sleep het naar deze pagina."
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "Stijlen exporteren" "message": "Stijlen exporteren"
@ -98,7 +98,7 @@
"message": "Automatisch aanvullen tijdens typen" "message": "Automatisch aanvullen tijdens typen"
}, },
"cm_colorpicker": { "cm_colorpicker": {
"message": "Kleurkiezers voor CSS-kleuren" "message": "Kleurkiezer voor CSS-kleuren"
}, },
"cm_indentWithTabs": { "cm_indentWithTabs": {
"message": "Tabs met slimme inspringing gebruiken" "message": "Tabs met slimme inspringing gebruiken"
@ -245,23 +245,9 @@
"disableAllStyles": { "disableAllStyles": {
"message": "Alle stijlen uitschakelen" "message": "Alle stijlen uitschakelen"
}, },
"disableAllStylesOff": {
"message": "Stijlen zijn uitgeschakeld"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "Uitschakelen" "message": "Uitschakelen"
}, },
"draftAction": {
"message": "Klik op Ja om dit concept te laden, of op Nee om het te verwerpen."
},
"draftTitle": {
"message": "Conceptherstel; gemaakt: $date$",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "Sleep uw back-upbestand naar deze pagina om het te importeren." "message": "Sleep uw back-upbestand naar deze pagina om het te importeren."
}, },
@ -288,9 +274,6 @@
} }
} }
}, },
"editorSettings": {
"message": "Editor-instellingen"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "Inschakelen" "message": "Inschakelen"
}, },
@ -300,9 +283,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "De huidige URL uitsluiten" "message": "De huidige URL uitsluiten"
}, },
"exportCompatible": {
"message": "Exporteren (compatibele modus)"
},
"exportLabel": { "exportLabel": {
"message": "Exporteren" "message": "Exporteren"
}, },
@ -338,6 +318,12 @@
"findStyles": { "findStyles": {
"message": "Stijlen zoeken" "message": "Stijlen zoeken"
}, },
"findStylesForSite": {
"message": "Meer stijlen voor deze website zoeken"
},
"findStylesInlineTooltip": {
"message": "Zoekresultaten binnen dit venster weergeven"
},
"genericAdd": { "genericAdd": {
"message": "Toevoegen" "message": "Toevoegen"
}, },
@ -371,12 +357,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "Opgeslagen" "message": "Opgeslagen"
}, },
"genericSize": {
"message": "Grootte"
},
"genericTest": {
"message": "Testen"
},
"genericTitle": { "genericTitle": {
"message": "Titel" "message": "Titel"
}, },
@ -386,9 +366,6 @@
"gettingStyles": { "gettingStyles": {
"message": "Alle stijlen ophalen..." "message": "Alle stijlen ophalen..."
}, },
"headerResizerHint": {
"message": "Houd Shift ingedrukt om alleen in dit type UI de grootte te wijzigen, m.a.w. editor, beheerder, installer"
},
"helpAlt": { "helpAlt": {
"message": "Hulp" "message": "Hulp"
}, },
@ -563,7 +540,7 @@
"message": "Niet beschikbaar" "message": "Niet beschikbaar"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus gebruikt een externe dienst: https://icons.duckduckgo.com" "message": "Stylus gebruikt een externe dienst: https://www.google.com/s2/favicons"
}, },
"manageHeading": { "manageHeading": {
"message": "Geïnstalleerde stijlen" "message": "Geïnstalleerde stijlen"
@ -571,9 +548,6 @@
"manageMaxTargets": { "manageMaxTargets": {
"message": "Aantal Van toepassing op-items" "message": "Aantal Van toepassing op-items"
}, },
"manageMinColumnWidth": {
"message": "Minimale kolombreedte (in pixels; 9999 schakelt de modus met meerdere kolommen uit)"
},
"manageNewStyleAsUsercss": { "manageNewStyleAsUsercss": {
"message": "als Usercss" "message": "als Usercss"
}, },
@ -823,15 +797,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "Geavanceerd" "message": "Geavanceerd"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "Systeemvoorkeur volgen"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "Door nachttijd:"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "Uitgeschakeld. De instelling voor donker/licht in stijlen wordt genegeerd."
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "Verwijderen toevoegen in contextmenu van editor" "message": "Verwijderen toevoegen in contextmenu van editor"
}, },
@ -841,12 +806,6 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "Legt het bovenliggende domein van de website bloot in elk iframe.\nSchakelt schrijven in voor iframe-specifieke CSS, zoals:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }" "message": "Legt het bovenliggende domein van de website bloot in elk iframe.\nSchakelt schrijven in voor iframe-specifieke CSS, zoals:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }"
}, },
"optionsAdvancedExposeStyleName": {
"message": "Stijlnaam tonen"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "Toont de stijlnaam in de pagina om het opsporen van stijlfouten in devtools mogelijk te maken. Vernieuw de tabbladen om de nieuwe instelling toe te passen."
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "Nieuwe stijl schrijven als usercss" "message": "Nieuwe stijl schrijven als usercss"
}, },
@ -889,9 +848,6 @@
"optionsHeading": { "optionsHeading": {
"message": "Opties" "message": "Opties"
}, },
"optionsIconAuto": {
"message": "Volgens de Donkere/Lichte modus"
},
"optionsIconDark": { "optionsIconDark": {
"message": "Donkere browserthemas" "message": "Donkere browserthemas"
}, },
@ -914,7 +870,7 @@
"message": "Standaardwaarden" "message": "Standaardwaarden"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "Klik op een willekeurige Stylus-pagina (waaronder deze) op het Stylus-pictogram in de browserwerkbalk, en klik daarna op Stijlen zoeken." "message": "Een UI-thema voor Stylus zoeken"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "Meer opties" "message": "Meer opties"
@ -931,9 +887,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "Geen" "message": "Geen"
}, },
"optionsSyncPassword": {
"message": "Wachtwoord"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "Gekoppeld" "message": "Gekoppeld"
}, },
@ -977,9 +930,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Nu synchroniseren" "message": "Nu synchroniseren"
}, },
"optionsSyncUsername": {
"message": "Gebruikersnaam"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "Als u stijlen vanuit een oudere versie of vanuit Stylish importeert, controleer dan eenmalig op updates in de stijlbeheerder om er zeker van te zijn dat alle stijlen zijn bijgewerkt." "message": "Als u stijlen vanuit een oudere versie of vanuit Stylish importeert, controleer dan eenmalig op updates in de stijlbeheerder om er zeker van te zijn dat alle stijlen zijn bijgewerkt."
}, },
@ -1022,9 +972,6 @@
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "Klik om beschikbare sneltoetsen te tonen" "message": "Klik om beschikbare sneltoetsen te tonen"
}, },
"popupManageSiteStyles": {
"message": "Websitestijlen beheren"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "Shift-klik of rechtsklik opent de beheerder met stijlen die op de huidige website van toepassing zijn" "message": "Shift-klik of rechtsklik opent de beheerder met stijlen die op de huidige website van toepassing zijn"
}, },
@ -1046,21 +993,6 @@
"prefShowBadge": { "prefShowBadge": {
"message": "Aantal actieve stijlen voor de huidige website" "message": "Aantal actieve stijlen voor de huidige website"
}, },
"preferScheme": {
"message": "Voorkeur voor Donkere/Lichte modus"
},
"preferSchemeAlways": {
"message": "Momenteel genegeerd (de stijl is altijd van toepassing), omdat de globale Donkere/Lichte modus is uitgeschakeld"
},
"preferSchemeDark": {
"message": "Donker"
},
"preferSchemeLight": {
"message": "Licht"
},
"preferSchemeNone": {
"message": "Geen (altijd toegepast)"
},
"previewLabel": { "previewLabel": {
"message": "Live-voorbeeld" "message": "Live-voorbeeld"
}, },
@ -1089,7 +1021,7 @@
"message": "Stijlen lezen..." "message": "Stijlen lezen..."
}, },
"reload": { "reload": {
"message": "Opnieuw laden" "message": "Stylus-extensie herladen"
}, },
"replace": { "replace": {
"message": "Vervangen" "message": "Vervangen"
@ -1100,18 +1032,12 @@
"replaceWith": { "replaceWith": {
"message": "Vervangen door" "message": "Vervangen door"
}, },
"restoreTemplate": {
"message": "De standaardsjabloon herstellen.\n\n(De momenteel geopende editorpaginas worden niet gewijzigd.)"
},
"retrieveBckp": { "retrieveBckp": {
"message": "Stijlen importeren" "message": "Stijlen importeren"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "Dropbox - Importeren" "message": "Dropbox - Importeren"
}, },
"saveAsTemplate": {
"message": "Opslaan als sjabloon"
},
"search": { "search": {
"message": "Zoeken" "message": "Zoeken"
}, },
@ -1152,7 +1078,7 @@
"message": "Wekelijks aantal installaties" "message": "Wekelijks aantal installaties"
}, },
"searchStyleQueryHint": { "searchStyleQueryHint": {
"message": "Stijlnamen doorzoeken (hoofdlettergevoelig als een hoofdletter wordt gebruikt):\nenkele woorden - al deze woorden in willekeurige volgorde\n\"bepaalde woordgroep\" - deze exacte woordgroep zonder aanhalingstekens\n/foo.*bar/i - reguliere expressie zonder spaties (gebruik in plaats daarvan \\s)" "message": "Stijlnamen niet hoofdlettergevoelig doorzoeken:\nbepaalde woorden - alle woorden in willekeurige volgorde\n\"bepaalde woordgroep\" - deze exacte woordgroep zonder aanhalingstekens\n2020 - een jaar als dit toont ook stijlen die in 2020 zijn bijgewerkt"
}, },
"searchStylesAll": { "searchStylesAll": {
"message": "Alles" "message": "Alles"
@ -1184,18 +1110,12 @@
"sections": { "sections": {
"message": "Secties" "message": "Secties"
}, },
"settings": {
"message": "Instellingen"
},
"shortcuts": { "shortcuts": {
"message": "Sneltoetsen" "message": "Sneltoetsen"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "Sneltoetsen definiëren" "message": "Sneltoetsen definiëren"
}, },
"shortcutsNoteFF": {
"message": "In Firefox 66+ kunt u de ingebouwde UI voor sneltoetsen handmatig openen:\n1) klik met rechts op het Stylus-pictogram in de werkbalk en kies Extensie beheren\n(als alternatief kunt u about:addons openen via het hoofdmenu of Ctrl-Shift-A),\n2) klik in de pagina die wordt geopend op het tandwielpictogram in de rechterbovenhoek,\n3) kies Extensiesneltoetsen beheren.\n\nHier kunt u ook de sneltoetsen aanpassen."
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "nieuwste eerst" "message": "nieuwste eerst"
}, },
@ -1241,30 +1161,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "Ingeschakeld" "message": "Ingeschakeld"
}, },
"styleExcludeLabel": {
"message": "Eigen uitgesloten websites"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "Importeren vanuit Mozilla-opmaak mislukt" "message": "Importeren vanuit Mozilla-opmaak mislukt"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "Plak de code in Mozilla-opmaak" "message": "Plak de code in Mozilla-opmaak"
}, },
"styleIncludeLabel": {
"message": "Eigen opgenomen websites"
},
"styleInjectionImportance": {
"message": "Urgentie van stijl bepalen"
},
"styleInjectionOrder": {
"message": "Stijlinjectievolgorde"
},
"styleInjectionOrderHint": {
"message": "Versleep een stijl om de positie ervan te wijzigen. Stijlen worden geïnjecteerd op basis van de onderstaande volgorde, dus een stijl die lager op de lijst staat, kan de eerdere stijlen vervangen."
},
"styleInjectionOrderHint_prio": {
"message": "Belangrijke stijlen hieronder worden altijd als laatste geïnjecteerd, zodat ze eventuele pas geïnstalleerde stijlen kunnen vervangen. Klik op het sterpictogram van een stijl om de urgentie ervan te bepalen."
},
"styleInstall": { "styleInstall": {
"message": "$stylename$ installeren in Stylus?", "message": "$stylename$ installeren in Stylus?",
"placeholders": { "placeholders": {
@ -1307,15 +1209,6 @@
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "Stijl is niet toegepast vanwege onjuist gebruik van regexp()" "message": "Stijl is niet toegepast vanwege onjuist gebruik van regexp()"
}, },
"styleNotAppliedSchemeDark": {
"message": "Deze stijl wordt alleen in Donkere modus toegepast"
},
"styleNotAppliedSchemeLight": {
"message": "Deze stijl wordt alleen in Lichte modus toegepast"
},
"stylePreferSchemeLabel": {
"message": "Donkere/Lichte modus"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "Sommige regexp()-regels konden niet worden gecompileerd." "message": "Sommige regexp()-regels konden niet worden gecompileerd."
}, },
@ -1325,6 +1218,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Aantal secties dat niet wordt toegepast vanwege onjuist gebruik van regexp()" "message": "Aantal secties dat niet wordt toegepast vanwege onjuist gebruik van regexp()"
}, },
"styleRegexpTestButton": {
"message": "RegExp-test"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Overeenkomende tabbladen" "message": "Overeenkomende tabbladen"
}, },
@ -1341,14 +1237,11 @@
"message": "Niet volledig overeenkomend, dus overgeslagen" "message": "Niet volledig overeenkomend, dus overgeslagen"
}, },
"styleRegexpTestTitle": { "styleRegexpTestTitle": {
"message": "Lijst met overeenkomende geopende tabbladen (klik op de URL om de focus op het tabblad te leggen)" "message": "Lijst van overeenkomende geopende tabbladen (klik op de URL om de focus op het tabblad te leggen)"
}, },
"styleSaveLabel": { "styleSaveLabel": {
"message": "Opslaan" "message": "Opslaan"
}, },
"styleSettings": {
"message": "Stijlinstellingen"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "De Mozilla-opmaak van de code kan bij userstyles.org worden ingediend en met het klassieke Stylish voor Firefox worden gebruikt." "message": "De Mozilla-opmaak van de code kan bij userstyles.org worden ingediend en met het klassieke Stylish voor Firefox worden gebruikt."
}, },
@ -1366,9 +1259,6 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "Deze stijl is buiten de editor om gewijzigd. Wilt u de stijl opnieuw laden?" "message": "Deze stijl is buiten de editor om gewijzigd. Wilt u de stijl opnieuw laden?"
}, },
"styleUpdateUrlLabel": {
"message": "Update-URL"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "Stylus werkt niet op paginas als deze." "message": "Stylus werkt niet op paginas als deze."
}, },
@ -1384,16 +1274,8 @@
"syncError": { "syncError": {
"message": "Synchronisatie mislukt" "message": "Synchronisatie mislukt"
}, },
"syncErrorLock": {
"message": "De database is al in gebruik. De vergrendeling vervalt om $TIME$.",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "Synchronisatie mislukt. U bent afgemeld.\nProbeer u opnieuw aan te melden in de Stylus-opties." "message": "Synchronisatie mislukt.\nProbeer u opnieuw aan te melden in de Stylus-opties:\nklik eerst op ontkoppelen, daarna op koppelen."
}, },
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "De waarde kan niet worden opgeslagen. Probeer de hoeveelheid tekst te verminderen." "message": "De waarde kan niet worden opgeslagen. Probeer de hoeveelheid tekst te verminderen."
@ -1484,6 +1366,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Wilt u de standaardsjabloon voor nieuwe Usercss-stijlen vervangen door de huidige code?" "message": "Wilt u de standaardsjabloon voor nieuwe Usercss-stijlen vervangen door de huidige code?"
}, },
"usercssReplaceTemplateName": {
"message": "Lege @name vervangt de standaardsjabloon"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Voer hier code in..." "message": "Voer hier code in..."
}, },

View File

@ -71,7 +71,7 @@
"message": "Kopia zapasowa" "message": "Kopia zapasowa"
}, },
"backupMessage": { "backupMessage": {
"message": "Aby zaimportować plik kopii zapasowej, przeciągnij go i upuść na tej stronie lub kliknij przycisk Importuj.\n\nAby wyeksportować kompatybilną kopię zapasową Stylusa w wersji starszej niż 1.5.18, kliknij prawym przyciskiem myszy lub kliknij z wciśniętym klawiszem Shift przycisk Eksportuj." "message": "Wybierz plik lub przeciągnij i upuść go na tę stronę."
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "Eksportuj style" "message": "Eksportuj style"
@ -264,23 +264,9 @@
"disableAllStyles": { "disableAllStyles": {
"message": "Wyłącz wszystkie style" "message": "Wyłącz wszystkie style"
}, },
"disableAllStylesOff": {
"message": "Style są wyłączone"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "Wyłącz" "message": "Wyłącz"
}, },
"draftAction": {
"message": "Wybierz 'Tak', aby załadować tę wersję roboczą lub 'Nie', aby ją odrzucić."
},
"draftTitle": {
"message": "Odzyskiwanie wersji roboczej, utworzonej $date$",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "Upuść twój plik kopii zapasowej gdziekolwiek na tej stronie, aby zaimportować." "message": "Upuść twój plik kopii zapasowej gdziekolwiek na tej stronie, aby zaimportować."
}, },
@ -307,9 +293,6 @@
} }
} }
}, },
"editorSettings": {
"message": "Ustawienia edytora"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "Włącz" "message": "Włącz"
}, },
@ -319,9 +302,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "Wyklucz bieżący adres URL" "message": "Wyklucz bieżący adres URL"
}, },
"exportCompatible": {
"message": "Eksportuj (tryb kompatybilny)"
},
"exportLabel": { "exportLabel": {
"message": "Eksportuj" "message": "Eksportuj"
}, },
@ -360,6 +340,15 @@
"findStyles": { "findStyles": {
"message": "Znajdź style" "message": "Znajdź style"
}, },
"findStylesForSite": {
"message": "Znajdź więcej stylów dla tej strony"
},
"findStylesInline": {
"message": "Wstawka"
},
"findStylesInlineTooltip": {
"message": "Wyświetlaj wyniki wyszukiwania w tym oknie."
},
"genericAdd": { "genericAdd": {
"message": "Dodaj" "message": "Dodaj"
}, },
@ -393,12 +382,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "Zapisano" "message": "Zapisano"
}, },
"genericSize": {
"message": "Rozmiar"
},
"genericTest": {
"message": "Testuj"
},
"genericTitle": { "genericTitle": {
"message": "Tytuł" "message": "Tytuł"
}, },
@ -408,9 +391,6 @@
"gettingStyles": { "gettingStyles": {
"message": "Uzyskiwanie wszystkich stylów..." "message": "Uzyskiwanie wszystkich stylów..."
}, },
"headerResizerHint": {
"message": "Przytrzymaj Shift, aby zmienić rozmiar tylko w tego typu interfejsie, tj. edytorze, menedżerze, instalatorze"
},
"helpAlt": { "helpAlt": {
"message": "Pomoc" "message": "Pomoc"
}, },
@ -585,7 +565,7 @@
"message": "Wyszarzone" "message": "Wyszarzone"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus korzysta z usługi zewnętrznej https://icons.duckduckgo.com" "message": "Stylus korzysta z usługi zewnętrznej https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtry" "message": "Filtry"
@ -596,9 +576,6 @@
"manageMaxTargets": { "manageMaxTargets": {
"message": "Liczba dotyczących elementów " "message": "Liczba dotyczących elementów "
}, },
"manageMinColumnWidth": {
"message": "Minimalna szerokość kolumny (w pikselach; 9999 wyłącza tryb wielokolumnowy)"
},
"manageNewStyleAsUsercss": { "manageNewStyleAsUsercss": {
"message": "jako Usercss" "message": "jako Usercss"
}, },
@ -848,15 +825,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "Zaawansowane" "message": "Zaawansowane"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "Według preferencji systemowych"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "W nocy:"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "Wyłączone. Ustawienie ciemny/jasny w stylach jest ignorowane."
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "Dodaj 'Usuń' do menu kontekstowego edytora" "message": "Dodaj 'Usuń' do menu kontekstowego edytora"
}, },
@ -866,12 +834,6 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "Odsłania domenę najwyższego poziomu w każdym elemencie iframe.\nWłącza pisanie CSS specyficznych dla iframe w taki sposób:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }" "message": "Odsłania domenę najwyższego poziomu w każdym elemencie iframe.\nWłącza pisanie CSS specyficznych dla iframe w taki sposób:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }"
}, },
"optionsAdvancedExposeStyleName": {
"message": "Eksponuj nazwę stylu"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "Eksponuje nazwę stylu na stronie, aby ułatwić debugowanie stylów w narzędziach dla twórców witryn. Załaduj ponownie kartę(-y), aby zastosować nowe ustawienie."
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "Napisz nowy styl jako usercss" "message": "Napisz nowy styl jako usercss"
}, },
@ -917,9 +879,6 @@
"optionsHeading": { "optionsHeading": {
"message": "Opcje" "message": "Opcje"
}, },
"optionsIconAuto": {
"message": "Dopasuj tryb ciemny/jasny"
},
"optionsIconDark": { "optionsIconDark": {
"message": "Ciemne motywy przeglądarki" "message": "Ciemne motywy przeglądarki"
}, },
@ -942,7 +901,7 @@
"message": "Resetuj opcje" "message": "Resetuj opcje"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "Kliknij ikonę Stylusa na pasku narzędzi przeglądarki na dowolnej stronie Stylusa, w tym tej, a następnie kliknij 'Znajdź style'" "message": "Znajdź motyw interfejsu Stylusa"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "Więcej opcji" "message": "Więcej opcji"
@ -959,9 +918,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "Brak" "message": "Brak"
}, },
"optionsSyncPassword": {
"message": "Hasło"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "Połączono" "message": "Połączono"
}, },
@ -1005,12 +961,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Synchronizuj teraz" "message": "Synchronizuj teraz"
}, },
"optionsSyncUrl": {
"message": "Adres URL"
},
"optionsSyncUsername": {
"message": "Nazwa użytkownika"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "Podczas importowania kopii zapasowych stylu ze starej wersji lub ze Stylish jednorazowo sprawdź aktualizacje ręcznie w menedżerze stylów, aby upewnić się, że wszystkie style są zaktualizowane." "message": "Podczas importowania kopii zapasowych stylu ze starej wersji lub ze Stylish jednorazowo sprawdź aktualizacje ręcznie w menedżerze stylów, aby upewnić się, że wszystkie style są zaktualizowane."
}, },
@ -1053,9 +1003,6 @@
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "Kliknij, aby zobaczyć dostępne skróty klawiszowe" "message": "Kliknij, aby zobaczyć dostępne skróty klawiszowe"
}, },
"popupManageSiteStyles": {
"message": "Zarządzaj stylami witryny"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "Shift + kliknięcie lub kliknięcie prawym przyciskiem otwiera menedżera ze stylami obowiązującymi dla bieżącej witryny" "message": "Shift + kliknięcie lub kliknięcie prawym przyciskiem otwiera menedżera ze stylami obowiązującymi dla bieżącej witryny"
}, },
@ -1077,21 +1024,6 @@
"prefShowBadge": { "prefShowBadge": {
"message": "Liczba aktywnych stylów dla bieżącej witryny" "message": "Liczba aktywnych stylów dla bieżącej witryny"
}, },
"preferScheme": {
"message": "Preferencje trybu ciemnego/jasnego"
},
"preferSchemeAlways": {
"message": "Obecnie ignorowane (styl zawsze obowiązuje), ponieważ globalny tryb ciemny/jasny jest wyłączony"
},
"preferSchemeDark": {
"message": "Ciemny"
},
"preferSchemeLight": {
"message": "Jasny"
},
"preferSchemeNone": {
"message": "Brak (zawsze stosowane)"
},
"previewLabel": { "previewLabel": {
"message": "Podgląd na żywo" "message": "Podgląd na żywo"
}, },
@ -1120,7 +1052,7 @@
"message": "Odczytywanie stylów..." "message": "Odczytywanie stylów..."
}, },
"reload": { "reload": {
"message": "Przeładuj" "message": "Przeładuj rozszerzenie Stylus"
}, },
"replace": { "replace": {
"message": "Zamień" "message": "Zamień"
@ -1131,18 +1063,12 @@
"replaceWith": { "replaceWith": {
"message": "Zamień na" "message": "Zamień na"
}, },
"restoreTemplate": {
"message": "Przywróć szablon domyślny.\n\n(Aktualnie otwarte strony edytora nie ulegną zmianie.)"
},
"retrieveBckp": { "retrieveBckp": {
"message": "Importuj style" "message": "Importuj style"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "Importuj z Dropboksa" "message": "Importuj z Dropboksa"
}, },
"saveAsTemplate": {
"message": "Zapisz jako szablon"
},
"search": { "search": {
"message": "Szukaj" "message": "Szukaj"
}, },
@ -1183,7 +1109,7 @@
"message": "Tygodniowa liczba instalacji" "message": "Tygodniowa liczba instalacji"
}, },
"searchStyleQueryHint": { "searchStyleQueryHint": {
"message": "Szukaj nazwy stylów (z uwzględnieniem wielkości liter, jeśli używana jest wielka litera):\njakieś słowa wszystkie te słowa w dowolnej kolejności\n\"jakaś fraza\" ta fraza bez cudzysłowów\n/foo.*bar/i wyrażenie regularne bez spacji (zamiast tego użyj \\s)" "message": "Szukaj nazw stylów bez rozróżniania wielkości liter:\njakieś słowa - wszystkie słowa w dowolnej kolejności\n\"jakieś zdanie\" - dokładnie to zdanie bez cudzysłowów\n2020 - rok jak ten pokazuje również style zaktualizowane w 2020"
}, },
"searchStylesAll": { "searchStylesAll": {
"message": "Wszystkie" "message": "Wszystkie"
@ -1218,18 +1144,12 @@
"sections": { "sections": {
"message": "Sekcje" "message": "Sekcje"
}, },
"settings": {
"message": "Ustawienia"
},
"shortcuts": { "shortcuts": {
"message": "Skróty" "message": "Skróty"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "Zdefiniuj skróty klawiaturowe" "message": "Zdefiniuj skróty klawiaturowe"
}, },
"shortcutsNoteFF": {
"message": "W przeglądarce Firefox 66+ możesz ręcznie otworzyć interfejs wbudowanych skrótów:\n1) kliknij prawym przyciskiem myszy ikonę Stylusa na pasku narzędzi i wybierz 'Zarządzaj'\n(alternatywnie otwórz about:addon z menu głównego lub Ctrl-Shift-A),\n2) na stronie, która się otworzy, kliknij ikonę koła zębatego w prawym górnym rogu,\n3) wybierz 'Zarządzaj skrótami rozszerzeń'.\n\nMożesz także dostosować skróty tutaj."
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "najpierw najnowsze" "message": "najpierw najnowsze"
}, },
@ -1275,30 +1195,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "Włączony" "message": "Włączony"
}, },
"styleExcludeLabel": {
"message": "Niestandardowe wykluczone witryny"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "Nie udało się zaimportować z formatu Mozilla" "message": "Nie udało się zaimportować z formatu Mozilla"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "Wprowadź kod w formacie Mozilla" "message": "Wprowadź kod w formacie Mozilla"
}, },
"styleIncludeLabel": {
"message": "Niestandardowe uwzględnione witryny"
},
"styleInjectionImportance": {
"message": "Przełącz ważność stylu"
},
"styleInjectionOrder": {
"message": "Kolejność wstrzykiwania stylów"
},
"styleInjectionOrderHint": {
"message": "Przeciągnij i upuść styl, aby zmienić jego pozycję. Style są wstrzykiwane sekwencyjnie w kolejności pokazanej poniżej, dzięki czemu styl znajdujący się dalej na liście może zastąpić wcześniejsze style."
},
"styleInjectionOrderHint_prio": {
"message": "Ważne style wymienione poniżej są zawsze wstrzykiwane jako ostatnie, dzięki czemu mogą zastąpić nowo zainstalowane style. Kliknij oznaczenie stylu, aby zmienić jego ważność."
},
"styleInstall": { "styleInstall": {
"message": "Zainstalować '$stylename$' do Stylusa?", "message": "Zainstalować '$stylename$' do Stylusa?",
"placeholders": { "placeholders": {
@ -1341,15 +1243,6 @@
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "Styl nie został zastosowany z powodu nieprawidłowego użycia 'regexp()'" "message": "Styl nie został zastosowany z powodu nieprawidłowego użycia 'regexp()'"
}, },
"styleNotAppliedSchemeDark": {
"message": "Ten styl jest stosowany tylko w trybie ciemnym"
},
"styleNotAppliedSchemeLight": {
"message": "Ten styl jest stosowany tylko w trybie jasnym"
},
"stylePreferSchemeLabel": {
"message": "Tryb ciemny/jasny"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "Niektóre reguły 'regexp()', których nie można było w ogóle skompilować." "message": "Niektóre reguły 'regexp()', których nie można było w ogóle skompilować."
}, },
@ -1359,6 +1252,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Liczba sekcji, które nie zostały zastosowane z powodu nieprawidłowego użycia 'regexp()'" "message": "Liczba sekcji, które nie zostały zastosowane z powodu nieprawidłowego użycia 'regexp()'"
}, },
"styleRegexpTestButton": {
"message": "Test RegExp"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Dopasowane karty" "message": "Dopasowane karty"
}, },
@ -1380,9 +1276,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "Zapisz" "message": "Zapisz"
}, },
"styleSettings": {
"message": "Ustawienia stylu"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "Kod w formacie Mozilla może być przesłany do userstyles.org i użyty z klasycznym dodatkiem Stylish dla Firefoksa" "message": "Kod w formacie Mozilla może być przesłany do userstyles.org i użyty z klasycznym dodatkiem Stylish dla Firefoksa"
}, },
@ -1400,9 +1293,6 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "Styl zmieniono poza edytorem. Czy chcesz przeładować styl?" "message": "Styl zmieniono poza edytorem. Czy chcesz przeładować styl?"
}, },
"styleUpdateUrlLabel": {
"message": "Adres URL aktualizacji"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "Stylus nie działa na takich stronach." "message": "Stylus nie działa na takich stronach."
}, },
@ -1418,16 +1308,8 @@
"syncError": { "syncError": {
"message": "Synchronizacja nie powiodła się" "message": "Synchronizacja nie powiodła się"
}, },
"syncErrorLock": {
"message": "Baza danych jest już w użyciu. Blokada wygaśnie o $TIME$",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "Synchronizacja nie powiodła się. Wylogowano.\nSpróbuj ponownie zalogować się w opcjach Stylusa." "message": "Synchronizacja nie powiodła się.\nSpróbuj ponownie zalogować się w opcjach Stylusa:\nkliknij najpierw 'rozłącz' , następnie 'połącz'."
}, },
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "Nie można zapisać wartości. Spróbuj zmniejszyć ilość tekstu." "message": "Nie można zapisać wartości. Spróbuj zmniejszyć ilość tekstu."
@ -1518,6 +1400,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Zastąpić domyślny szablon dla nowych stylów Usercss aktualnym kodem?" "message": "Zastąpić domyślny szablon dla nowych stylów Usercss aktualnym kodem?"
}, },
"usercssReplaceTemplateName": {
"message": "Puste @name zastępuje szablon domyślny"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Wstaw kod tutaj..." "message": "Wstaw kod tutaj..."
}, },

View File

@ -64,6 +64,9 @@
"author": { "author": {
"message": "Autor" "message": "Autor"
}, },
"backupMessage": {
"message": "Selecione um arquivo ou arraste e solte nessa página."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Exportar estilos" "message": "Exportar estilos"
}, },
@ -121,9 +124,6 @@
"cm_selectByTokens": { "cm_selectByTokens": {
"message": "Clicar duas vezes seleciona tokens" "message": "Clicar duas vezes seleciona tokens"
}, },
"cm_selectByTokensTooltip": {
"message": "Exemplos de tokens: .foo-bar-2 #aabbcc 0.32 !Important\nQuando desativado: as palavras delimitadas por pontuação são selecionadas."
},
"cm_smartIndent": { "cm_smartIndent": {
"message": "Usar indentação inteligente" "message": "Usar indentação inteligente"
}, },
@ -231,9 +231,6 @@
"disableAllStyles": { "disableAllStyles": {
"message": "Desativar todos os estilos" "message": "Desativar todos os estilos"
}, },
"disableAllStylesOff": {
"message": "Estilos estão desligados"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "Desativar" "message": "Desativar"
}, },
@ -310,15 +307,21 @@
"findStyles": { "findStyles": {
"message": "Encontrar estilos" "message": "Encontrar estilos"
}, },
"findStylesForSite": {
"message": "Procurar mais estilos para este site"
},
"findStylesInline": {
"message": "Em linha"
},
"findStylesInlineTooltip": {
"message": "Mostrar resultados dentro dessa janela"
},
"genericAdd": { "genericAdd": {
"message": "Adicionar" "message": "Adicionar"
}, },
"genericClone": { "genericClone": {
"message": "Clonar" "message": "Clonar"
}, },
"genericDescription": {
"message": "Descrição"
},
"genericDisabledLabel": { "genericDisabledLabel": {
"message": "Desativado" "message": "Desativado"
}, },
@ -507,63 +510,27 @@
"liveReloadInstallHintFF": { "liveReloadInstallHintFF": {
"message": "Mantenha juntamente esta guia e a guia original abertas para atualizar automaticamente o estilo sob mudanças externas." "message": "Mantenha juntamente esta guia e a guia original abertas para atualizar automaticamente o estilo sob mudanças externas."
}, },
"liveReloadLabel": {
"message": "Recarregamento dinâmico"
},
"manageFavicons": {
"message": "Favicons em colunas de aplica-se a"
},
"manageFaviconsGray": {
"message": "Acinzentado(s)"
},
"manageFaviconsHelp": {
"message": "O Stylus usa um serviço externo https://icons.duckduckgo.com"
},
"manageFilters": { "manageFilters": {
"message": "Filtros" "message": "Filtros"
}, },
"manageHeading": { "manageHeading": {
"message": "Estilos instalados" "message": "Estilos instalados"
}, },
"manageMaxTargets": {
"message": "Número de aplica-se a itens"
},
"manageNewStyleAsUsercss": { "manageNewStyleAsUsercss": {
"message": "como UserCSS" "message": "como UserCSS"
}, },
"manageNewUI": {
"message": "Nova interface do gestor"
},
"manageOnlyDisabled": { "manageOnlyDisabled": {
"message": "Somente estilos desativados" "message": "Somente estilos desativados"
}, },
"manageOnlyEnabled": { "manageOnlyEnabled": {
"message": "Apenas estilos habilitados" "message": "Apenas estilos habilitados"
}, },
"manageOnlyExternal": {
"message": "Apenas estilos externos"
},
"manageOnlyLocal": {
"message": "Apenas estilos criados localmente"
},
"manageOnlyLocalTooltip": {
"message": "(os estilos não instalados através de uma página userstyles.org)"
},
"manageOnlyNonUsercss": { "manageOnlyNonUsercss": {
"message": "Apenas estilos sem UserCSS" "message": "Apenas estilos sem UserCSS"
}, },
"manageOnlyUpdates": {
"message": "Apenas com atualizações ou problemas"
},
"manageOnlyUsercss": { "manageOnlyUsercss": {
"message": "Apenas estilos com UserCSS" "message": "Apenas estilos com UserCSS"
}, },
"menuShowBadge": {
"message": "Mostrar a contagem de estilos ativados"
},
"meta_invalidCheckboxDefault": {
"message": "Inválida @var checkbox: o valor deve ser 0 ou 1"
},
"meta_invalidNumber": { "meta_invalidNumber": {
"message": "Espera-se um número" "message": "Espera-se um número"
}, },
@ -634,15 +601,6 @@
"optionsActions": { "optionsActions": {
"message": "Ações" "message": "Ações"
}, },
"optionsAdvanced": {
"message": "Avançadas"
},
"optionsAdvancedContextDelete": {
"message": "Adicionar 'Eliminar' no menu de contexto do editor"
},
"optionsAdvancedExposeIframes": {
"message": "Expor iframes via HTML[stylus-iframe]"
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "Escrever novo estilo como UserCSS" "message": "Escrever novo estilo como UserCSS"
}, },
@ -658,12 +616,6 @@
"optionsCheckUpdate": { "optionsCheckUpdate": {
"message": "Verifique e instale todas as atualizações disponíveis" "message": "Verifique e instale todas as atualizações disponíveis"
}, },
"optionsCustomizeBadge": {
"message": "Distintivo no ícone da barra de ferramentas"
},
"optionsCustomizeIcon": {
"message": "Ícone da barra de ferramentas"
},
"optionsCustomizeSync": { "optionsCustomizeSync": {
"message": "Sincronizar para a nuvem" "message": "Sincronizar para a nuvem"
}, },
@ -673,12 +625,6 @@
"optionsHeading": { "optionsHeading": {
"message": "Opções" "message": "Opções"
}, },
"optionsIconDark": {
"message": "Temas escuros do browser"
},
"optionsIconLight": {
"message": "Temas claros do browser"
},
"optionsOpen": { "optionsOpen": {
"message": "Abrir" "message": "Abrir"
}, },
@ -688,12 +634,6 @@
"optionsPopupWidth": { "optionsPopupWidth": {
"message": "Largura do popup (em pixels)" "message": "Largura do popup (em pixels)"
}, },
"optionsReset": {
"message": "Restabelecer as opções aos valores predefinidos"
},
"optionsResetButton": {
"message": "Restabelecer opções"
},
"optionsSubheading": { "optionsSubheading": {
"message": "Mais Opções" "message": "Mais Opções"
}, },
@ -706,9 +646,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "Nenhum" "message": "Nenhum"
}, },
"optionsSyncPassword": {
"message": "Senha"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "Conectado" "message": "Conectado"
}, },
@ -727,86 +664,20 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Sincronizar agora" "message": "Sincronizar agora"
}, },
"optionsUpdateImportNote": {
"message": "Ao importar uma cópia de segurança de estilo da versão antiga ou do Stylish, faça uma verificação única de atualizações manualmente no gestor de estilos para garantir que todos os estilos sejam atualizados."
},
"optionsUpdateInterval": {
"message": "Intervalo de atualização automática do estilo de usuário em horas (especifique 0 para desativar)"
},
"overwriteFileExport": { "overwriteFileExport": {
"message": "Você gostaria de substituir um arquivo existente?" "message": "Você gostaria de substituir um arquivo existente?"
}, },
"paginationCurrent": {
"message": "Pagina atual"
},
"paginationEstimated": {
"message": "Número estimado de páginas"
},
"paginationNext": {
"message": "Próxima página"
},
"paginationPrevious": {
"message": "Pagina anterior"
},
"paginationTotal": {
"message": "Páginas totais"
},
"parseUsercssError": { "parseUsercssError": {
"message": "Não foi possível analisar o UserCSS:" "message": "Não foi possível analisar o UserCSS:"
}, },
"popupBorders": {
"message": "Adicionar margens laterais brancas "
},
"popupBordersTooltip": {
"message": "Útil para temas escuros no novo Chrome, já que não pinta mais as margens laterais"
},
"popupHotkeysInfo": {
"message": "<1>- <9>,<0>, também no teclado numérico - alterna o Nth estilo (0 é 10)\n<A>- <Z>alterna o primeiro estilo com um nome que começa com a letra\n<Shift>abre o editor em vez de alternar\n<Numpad +>ativa estilos listados\n<Numpad >desativa estilos listados\n<Numpad *>e <`> (backtick) - alterna os estilos inicialmente ativados; não se aplica a estilos habilitados subsequentemente enquanto o popup está aberto, para que você possa restaurar a seleção inicial depois de testar o material: basta desabilitar todos e, em seguida, alternar, i.e. <Numpad ><Numpad *>\nMais informações no wiki"
},
"popupHotkeysTooltip": {
"message": "Clique para ver as teclas de atalho disponíveis"
},
"popupManageTooltip": {
"message": "Shift-clique ou clique com o botão direito abre o gestor com estilos aplicáveis ao site atual"
},
"popupOpenEditInWindow": {
"message": "Abra o editor em uma nova janela"
},
"popupOpenEditInWindowTooltip": {
"message": "Também ativado ao desanexar o separador do editor de uma janela do browser, e desativado por anexar um separador único do editor a outra janela."
},
"popupStylesFirst": {
"message": "Estilos antes de comandos"
},
"prefShowBadge": { "prefShowBadge": {
"message": "Número de estilos ativos para o site atual" "message": "Número de estilos ativos para o site atual"
}, },
"preferSchemeDark": {
"message": "Escuro"
},
"preferSchemeLight": {
"message": "Claro"
},
"preferSchemeNone": {
"message": "Nenhum (sempre aplicado)"
},
"previewLabel": {
"message": "Pré-visualização dinâmica"
},
"previewTooltip": {
"message": "Temporariamente aplica as alterações sem guardar.\nGuarde o estilo para tornar as alterações permanentes."
},
"publish": {
"message": "Publicar"
},
"publishStyle": {
"message": "Publicar estilo"
},
"readingStyles": { "readingStyles": {
"message": "Lendo estilos..." "message": "Lendo estilos..."
}, },
"reload": { "reload": {
"message": "Recarregar" "message": "Recarregar a extensão do Stylus"
}, },
"replace": { "replace": {
"message": "Substituir" "message": "Substituir"
@ -826,39 +697,9 @@
"search": { "search": {
"message": "Buscar" "message": "Buscar"
}, },
"searchCaseSensitive": {
"message": "Sensível a maiúsculas e minúsculas"
},
"searchGlobalStyles": {
"message": "Também buscar estilos globais"
},
"searchNumberOfResults": {
"message": "Número de correspondências"
},
"searchNumberOfResults2": {
"message": "Número de correspondências no código e aplica-se a valores"
},
"searchRegexp": { "searchRegexp": {
"message": "Use a sintaxe /re/ para busca por regexp" "message": "Use a sintaxe /re/ para busca por regexp"
}, },
"searchResultInstallCount": {
"message": "Instalações totais"
},
"searchResultNoneFound": {
"message": "Nenhum estilo encontrado para este site."
},
"searchResultRating": {
"message": "Classificação"
},
"searchResultUpdated": {
"message": "Atualizado"
},
"searchResultWeeklyCount": {
"message": "Instalações semanais"
},
"searchStylesCode": {
"message": "Código CSS"
},
"sectionAdd": { "sectionAdd": {
"message": "Adicionar outra seção" "message": "Adicionar outra seção"
}, },
@ -868,39 +709,9 @@
"sectionRemove": { "sectionRemove": {
"message": "Remover seção" "message": "Remover seção"
}, },
"sectionRestore": {
"message": "Restaurar secção removida"
},
"sections": {
"message": "Seções"
},
"shortcuts": { "shortcuts": {
"message": "Atalhos" "message": "Atalhos"
}, },
"shortcutsNote": {
"message": "Definir atalhos de teclado"
},
"sortDateNewestFirst": {
"message": "mais recente primeiro"
},
"sortDateOldestFirst": {
"message": "mais antigos primeiro"
},
"sortLabel": {
"message": "Selecione uma ordenação para aplicar aos estilos instalados"
},
"sortLabelTitleAsc": {
"message": "Título Ascendente"
},
"sortLabelTitleDesc": {
"message": "Título Descendente"
},
"sortStylesHelp": {
"message": "Escolha o tipo de ordenação a ser aplicado às entradas instaladas na lista suspensa de ordenação. A configuração predefinida aplica uma ordem crescente (A a Z) aos títulos de entrada. As ordenações dentro do grupo \"Título Decrescente\" aplicarão uma classificação decrescente (Z a A) ao título. Existem outras predefinições que permitirão classificar as entradas por vários critérios. Pense sobre isso como ordenar uma tabela com várias colunas e cada categoria em cada seleção (entre os sinais de mais) representa uma coluna ou grupo. Por exemplo, se a configuração for \"Ativado (primeiro) + Título\", as entradas serão ordenadas de modo que todas as entradas ativadas sejam classificadas no topo da lista, então uma classificação crescente de título de entrada (A a Z) será aplicada a ambas as entradas ativadas e desativadas separadamente."
},
"sortStylesHelpTitle": {
"message": "Ordenar conteúdos"
},
"styleBadRegexp": { "styleBadRegexp": {
"message": "Expressão regular é inválida" "message": "Expressão regular é inválida"
}, },
@ -910,12 +721,6 @@
"styleBeautifyHint": { "styleBeautifyHint": {
"message": "Dica: clique com o botão direito no botão \"Embelezar\" ou use o atalho de teclado definido para embelezar sem mostrar esse painel" "message": "Dica: clique com o botão direito no botão \"Embelezar\" ou use o atalho de teclado definido para embelezar sem mostrar esse painel"
}, },
"styleBeautifyIndentConditional": {
"message": "Indentar @media, @supports"
},
"styleBeautifyPreserveNewlines": {
"message": "Preserve novas linhas"
},
"styleCancelEditLabel": { "styleCancelEditLabel": {
"message": "Voltar ao gerenciamento" "message": "Voltar ao gerenciamento"
}, },
@ -925,12 +730,6 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "Ativado" "message": "Ativado"
}, },
"styleFromMozillaFormatError": {
"message": "Falha ao importar do formato Mozilla"
},
"styleFromMozillaFormatPrompt": {
"message": "Colar o código formato-Mozilla"
},
"styleInstall": { "styleInstall": {
"message": "Instalar \"$stylename$\" no Stylus?", "message": "Instalar \"$stylename$\" no Stylus?",
"placeholders": { "placeholders": {
@ -939,79 +738,18 @@
} }
} }
}, },
"styleInstallFailed": {
"message": "Falha ao instalar o estilo de usuário!\n$error$",
"placeholders": {
"error": {
"content": "$1"
}
}
},
"styleInstallOverwrite": {
"message": "$stylename$já está instalado. Substituir?\nVersão:$oldVersion$-> $newVersion$",
"placeholders": {
"newVersion": {
"content": "$3"
},
"oldVersion": {
"content": "$2"
},
"stylename": {
"content": "$1"
}
}
},
"styleMissingName": { "styleMissingName": {
"message": "Insira um nome" "message": "Insira um nome"
}, },
"styleMozillaFormatHeading": { "styleMozillaFormatHeading": {
"message": "Formato Mozilla" "message": "Formato Mozilla"
}, },
"styleName": {
"message": "Nome do estilo"
},
"styleNotAppliedRegexpProblemTooltip": {
"message": "O estilo não foi aplicado devido ao uso incorreto de 'regexp ()'"
},
"stylePreferSchemeLabel": {
"message": "Modo Escuro/Claro"
},
"styleRegexpInvalidExplanation": {
"message": "Algumas regras de 'regexp()' que não puderam ser compiladas."
},
"styleRegexpPartialExplanation": {
"message": "Este estilo usa regexps parcialmente correspondentes em violação da <a href='https://developer.mozilla.org/docs/Web/CSS/@document'>CSS4 @document specification</a> que requer um URL correspondente inteiro. As secções de CSS afetadas não foram aplicadas nesta página. Este estilo foi provavelmente criado no Stylish-for-Chrome o qual verifica incorretamente as regras 'regexp()' desde a primeira versão (bug conhecido)."
},
"styleRegexpProblemTooltip": {
"message": "Número de secções não aplicadas devido ao uso incorreto de 'regexp ()'"
},
"styleRegexpTestFull": {
"message": "Separadores correspondentes"
},
"styleRegexpTestInvalid": {
"message": "Regexps inválidos ignorados"
},
"styleRegexpTestNone": {
"message": "Nenhum separador correspondente"
},
"styleRegexpTestNote": {
"message": "Nota: use um único \\ para escapar no campo de entrada regexp, que será automaticamente convertido para \\\\ no código de estilo conforme especificação para strings entre aspas em CSS."
},
"styleRegexpTestPartial": {
"message": "Não corresponde totalmente, portanto ignorado"
},
"styleRegexpTestTitle": {
"message": "Lista de separadores correspondentes abertos (clique no URL para focar no separador)"
},
"styleSaveLabel": { "styleSaveLabel": {
"message": "Salvar" "message": "Salvar"
}, },
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "O formato Mozilla do código pode ser usado com o Stylish para Firefox e pode ser enviado para userstyles.org." "message": "O formato Mozilla do código pode ser usado com o Stylish para Firefox e pode ser enviado para userstyles.org."
}, },
"styleToMozillaFormatTitle": {
"message": "Estilo no formato Mozilla"
},
"styleUpdate": { "styleUpdate": {
"message": "Você tem certeza que quer atualizar '$stylename$'?", "message": "Você tem certeza que quer atualizar '$stylename$'?",
"placeholders": { "placeholders": {
@ -1020,57 +758,21 @@
} }
} }
}, },
"styleUpdateDiscardChanges": {
"message": "O estilo é alterado fora do editor. Gostaria de recarregar o estilo?"
},
"styleUpdateUrlLabel": {
"message": "Atualizar URL"
},
"stylusUnavailableForURL": {
"message": "O Stylus não funciona em páginas como esta."
},
"stylusUnavailableForURLdetails": {
"message": "Como precaução de segurança, o browser proíbe extensões de afetar as páginas embutidas (como chrome://version, a página de novo separador predefinido no Chrome 61, about:addons, e assim sucessivamente) tal como páginas de outras extensões. Cada browser também restringe acesso à sua própria galeria de extensões (como Chrome Web Store ou AMO)"
},
"syncDropboxStyles": { "syncDropboxStyles": {
"message": "Exportar para Dropbox" "message": "Exportar para Dropbox"
}, },
"syncStorageErrorSaving": {
"message": "O valor não pode ser guardado. Tente reduzir a quantidade de texto."
},
"toggleStyle": {
"message": "Alternar estilo"
},
"undo": { "undo": {
"message": "Desfazer" "message": "Desfazer"
}, },
"undoGlobal": { "undoGlobal": {
"message": "Desfazer todas as seções" "message": "Desfazer todas as seções"
}, },
"unreachableAMO": {
"message": "O Firefox proíbe o acesso ao site."
},
"unreachableAMOHint": {
"message": "Para permitir o acesso abra<about:config>,clique com o botão direito na lista, clique em \"Novo\", depois em \"Booleano\", cole <privacy.resistFingerprinting.block_mozAddonManager> e clique em OK,<true>,OK, recarregue a página <addons.mozilla.org>."
},
"unreachableContentScript": {
"message": "Não foi possível comunicar com a página. Tente recarregar o separador."
},
"unreachableFileHint": {
"message": "O Stylus pode aceder URLs file:// apenas se ativar a checkbox correspondente para a extensão Stylus na página chrome://extensions"
},
"unreachableMozSiteHintOldFF": {
"message": "Somente o Firefox 59 e o mais recente podem ser configurados para permitir que WebExtensions incluam elementos de estilo em sites protegidos por CSP como este."
},
"unzipStyles": { "unzipStyles": {
"message": "Descompactando estilos..." "message": "Descompactando estilos..."
}, },
"updateAllCheckSucceededNoUpdate": { "updateAllCheckSucceededNoUpdate": {
"message": "Nenhuma atualização encontrada." "message": "Nenhuma atualização encontrada."
}, },
"updateAllCheckSucceededSomeEdited": {
"message": "Alguns estilos que podem ser atualizados não foram verificados para evitar perder possíveis edições locais. As atualizações podem ser forçadas ao verificar individualmente, ou fazendo outra verificação para todos os estilos (edições locais vão ser sobrescrevidas)"
},
"updateCheckFailBadResponseCode": { "updateCheckFailBadResponseCode": {
"message": "A atualização falhou: o servidor respondeu com código $code$.", "message": "A atualização falhou: o servidor respondeu com código $code$.",
"placeholders": { "placeholders": {
@ -1082,12 +784,6 @@
"updateCheckFailServerUnreachable": { "updateCheckFailServerUnreachable": {
"message": "A atualização falhou: servidor inacessível." "message": "A atualização falhou: servidor inacessível."
}, },
"updateCheckHistory": {
"message": "Histórico de verificação de atualizações"
},
"updateCheckManualUpdateForce": {
"message": "Instalar atualização (edições locais vão ser sobrescritas)"
},
"updateCheckManualUpdateHint": { "updateCheckManualUpdateHint": {
"message": "Forçar uma atualização irá sobrescrever qualquer alteração local." "message": "Forçar uma atualização irá sobrescrever qualquer alteração local."
}, },
@ -1109,18 +805,12 @@
"uploadingFile": { "uploadingFile": {
"message": "Enviando arquivo..." "message": "Enviando arquivo..."
}, },
"usercssAvoidOverwriting": {
"message": "Por favor modifique o valor de @name ou @namespace para evitar sobrescrever um estilo existente."
},
"usercssConfigIncomplete": {
"message": "O estilo foi atualizado ou eliminado após a exibição do diálogo de configuração. Essas variáveis não foram guardadas para evitar corromper os metadados do estilo:"
},
"usercssEditorNamePlaceholder": {
"message": "Especificar @name no código"
},
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Substituir o template padrão por novos estilos com UserCSS com o código atual?" "message": "Substituir o template padrão por novos estilos com UserCSS com o código atual?"
}, },
"usercssReplaceTemplateName": {
"message": "@nome vazio substitui o template padrão"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Insira o código aqui..." "message": "Insira o código aqui..."
}, },

View File

@ -1,7 +1,4 @@
{ {
"InaccessibleFileHint": {
"message": "Stylus não pode acessar alguns tipos de arquivos (ex: arquivos pdf e json)."
},
"addStyleLabel": { "addStyleLabel": {
"message": "Escrever novo estilo" "message": "Escrever novo estilo"
}, },
@ -67,6 +64,9 @@
"backupButtons": { "backupButtons": {
"message": "Cópia de segurança" "message": "Cópia de segurança"
}, },
"backupMessage": {
"message": "Selecione um ficheiro ou arraste e solte-o nesta página."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Exportar estilos" "message": "Exportar estilos"
}, },
@ -133,9 +133,6 @@
"cm_theme": { "cm_theme": {
"message": "Tema" "message": "Tema"
}, },
"colorpickerPaletteHint": {
"message": "Clique com o botão direito em uma amostra para percorrer suas linhas de código"
},
"colorpickerSwitchFormatTooltip": { "colorpickerSwitchFormatTooltip": {
"message": "Mudar formatos: HEX -> RGB -> HSL.\nShift-clique para inverter a direção.\nTambém através das teclas PgUp (PageUp), PgDn (PageDown)." "message": "Mudar formatos: HEX -> RGB -> HSL.\nShift-clique para inverter a direção.\nTambém através das teclas PgUp (PageUp), PgDn (PageDown)."
}, },
@ -181,32 +178,6 @@
"confirmYes": { "confirmYes": {
"message": "Sim" "message": "Sim"
}, },
"connectingDropbox": {
"message": "Conectando ao Dropbox..."
},
"connectingDropboxNotAllowed": {
"message": "Conectar ao Dropbox somente é disponível em apps instalados diretamente da loja web"
},
"copied": {
"message": "Copiado para a área de transferência"
},
"copy": {
"message": "Copiar para a área de transferência"
},
"customNameHint": {
"message": "Digite um nome personalizado para renomear o estilo na UI sem quebrar suas atualizações"
},
"customNameResetHint": {
"message": "Deixar de usar o nome personalizado, usar o próprio nome do estilo"
},
"dateAbbrYear": {
"message": "$value$a",
"placeholders": {
"value": {
"content": "$1"
}
}
},
"dateInstalled": { "dateInstalled": {
"message": "Data de instalação" "message": "Data de instalação"
}, },
@ -237,9 +208,6 @@
"dragDropMessage": { "dragDropMessage": {
"message": "Solte o ficheiro da sua cópia de segurança em qualquer sítio nesta página para importar." "message": "Solte o ficheiro da sua cópia de segurança em qualquer sítio nesta página para importar."
}, },
"dragDropUsercssTabstrip": {
"message": "Para instalar o arquivo, solte-o na linha das abas (a área onde os títulos das abas são mostrados)."
},
"editDeleteText": { "editDeleteText": {
"message": "Eliminar" "message": "Eliminar"
}, },
@ -263,18 +231,9 @@
"enableStyleLabel": { "enableStyleLabel": {
"message": "Ativar" "message": "Ativar"
}, },
"excludeStyleByDomainLabel": {
"message": "Excluir o domínio atual"
},
"excludeStyleByUrlLabel": {
"message": "Excluir a URL atual"
},
"exportLabel": { "exportLabel": {
"message": "Exportar" "message": "Exportar"
}, },
"exportSavedSuccess": {
"message": "Arquivo salvo com sucesso"
},
"externalLink": { "externalLink": {
"message": "Hiperligação externa" "message": "Hiperligação externa"
}, },
@ -301,6 +260,15 @@
"findStyles": { "findStyles": {
"message": "Localizar estilos" "message": "Localizar estilos"
}, },
"findStylesForSite": {
"message": "Encontrar mais estilos para este site"
},
"findStylesInline": {
"message": "inline"
},
"findStylesInlineTooltip": {
"message": "Exibir os resultados da pesquisa dentro desta janela."
},
"genericAdd": { "genericAdd": {
"message": "Adicionar" "message": "Adicionar"
}, },
@ -334,9 +302,6 @@
"genericUnknown": { "genericUnknown": {
"message": "Desconhecido" "message": "Desconhecido"
}, },
"gettingStyles": {
"message": "Obtendo todos os estilos..."
},
"helpAlt": { "helpAlt": {
"message": "Ajuda" "message": "Ajuda"
}, },
@ -346,9 +311,6 @@
"helpKeyMapHotkey": { "helpKeyMapHotkey": {
"message": "Prima uma tecla de atalho" "message": "Prima uma tecla de atalho"
}, },
"hostDisabled": {
"message": "Este hospedeiro foi desabilitado devido a um bug na versão atual que o navegador utilizado se encontra"
},
"importAppendLabel": { "importAppendLabel": {
"message": "Acrescentar ao estilo" "message": "Acrescentar ao estilo"
}, },
@ -358,12 +320,6 @@
"importLabel": { "importLabel": {
"message": "Importar" "message": "Importar"
}, },
"importPreprocessor": {
"message": "Estilos com <code>@preprocessor</code> não irão funcionar no modo clássico. Você pode trocar o editor para o modo UserCSS: 1) abre o gerenciador de estilos, 2) ative a caixa \"como UserCSS\", 3) clique \"Escrever novo estilo\"\n\nDeseja importar mesmo assim?"
},
"importPreprocessorTitle": {
"message": "Possível problema causado pelo @preprocessor"
},
"importReplaceLabel": { "importReplaceLabel": {
"message": "Sobrescrever estilo" "message": "Sobrescrever estilo"
}, },
@ -486,12 +442,6 @@
"liveReloadError": { "liveReloadError": {
"message": "Ocorreu um erro ao vigiar o arquivo" "message": "Ocorreu um erro ao vigiar o arquivo"
}, },
"liveReloadInstallHint": {
"message": "Mantenha esta guia aberta para atualizar automaticamente o estilo sob mudanças externas."
},
"liveReloadInstallHintFF": {
"message": "Mantenha juntamente esta guia e a guia original abertas para atualizar automaticamente o estilo sob mudanças externas."
},
"liveReloadLabel": { "liveReloadLabel": {
"message": "Recarregamento dinâmico" "message": "Recarregamento dinâmico"
}, },
@ -502,7 +452,7 @@
"message": "Acinzentado(s)" "message": "Acinzentado(s)"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "O Stylus usa um serviço externo https://icons.duckduckgo.com" "message": "O Stylus usa um serviço externo https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtros" "message": "Filtros"
@ -546,73 +496,12 @@
"menuShowBadge": { "menuShowBadge": {
"message": "Mostrar a contagem de estilos ativados" "message": "Mostrar a contagem de estilos ativados"
}, },
"meta_invalidCheckboxDefault": {
"message": "Inválida @var checkbox: o valor deve ser 0 ou 1"
},
"meta_invalidNumber": {
"message": "Espera-se um número"
},
"meta_invalidRange": {
"message": "@var inválido $type$: valor deve ser um número ou um vetor",
"placeholders": {
"type": {
"content": "$1"
}
}
},
"meta_invalidString": {
"message": "Espera-se um texto entre aspas"
},
"meta_invalidWord": {
"message": "Espera-se uma palavra"
},
"meta_missingChar": {
"message": "Caracteres esperados: $chars$",
"placeholders": {
"chars": {
"content": "$1"
}
}
},
"meta_missingMandatory": {
"message": "Metadata obrigatório não encontrado: $keys$",
"placeholders": {
"keys": {
"content": "$1"
}
}
},
"meta_unknownMeta": {
"message": "Metadata desconhecido: $key$",
"placeholders": {
"key": {
"content": "$1"
}
}
},
"meta_unknownVarType": {
"message": "Tipo desconhecido da variável @$varkey$: $vartype$",
"placeholders": {
"varkey": {
"content": "$1"
},
"vartype": {
"content": "$2"
}
}
},
"noFileToImport": {
"message": "Para importar seus estilos, você deve exportar primeiro."
},
"noStylesForSite": { "noStylesForSite": {
"message": "Nenhum estilo instalado para este site." "message": "Nenhum estilo instalado para este site."
}, },
"openManage": { "openManage": {
"message": "Gerir" "message": "Gerir"
}, },
"openOptions": {
"message": "Opções"
},
"openStylesManager": { "openStylesManager": {
"message": "Abrir gestor de estilos" "message": "Abrir gestor de estilos"
}, },
@ -649,9 +538,6 @@
"optionsCustomizeIcon": { "optionsCustomizeIcon": {
"message": "Ícone da barra de ferramentas" "message": "Ícone da barra de ferramentas"
}, },
"optionsCustomizeSync": {
"message": "Sincronizar para a nuvem"
},
"optionsCustomizeUpdate": { "optionsCustomizeUpdate": {
"message": "Atualizações" "message": "Atualizações"
}, },
@ -682,42 +568,12 @@
"optionsSubheading": { "optionsSubheading": {
"message": "Mais Opções" "message": "Mais Opções"
}, },
"optionsSyncConnect": {
"message": "Conectar"
},
"optionsSyncDisconnect": {
"message": "Desconectar"
},
"optionsSyncNone": {
"message": "Nenhum"
},
"optionsSyncStatusConnected": {
"message": "Conectado"
},
"optionsSyncStatusConnecting": {
"message": "Conectando..."
},
"optionsSyncStatusDisconnected": {
"message": "Desconectado"
},
"optionsSyncStatusDisconnecting": {
"message": "Desconectando..."
},
"optionsSyncStatusSyncing": {
"message": "Sincronizando..."
},
"optionsSyncSyncNow": {
"message": "Sincronizar agora"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "Ao importar uma cópia de segurança de estilo da versão antiga ou do Stylish, faça uma verificação única de atualizações manualmente no gestor de estilos para garantir que todos os estilos sejam atualizados." "message": "Ao importar uma cópia de segurança de estilo da versão antiga ou do Stylish, faça uma verificação única de atualizações manualmente no gestor de estilos para garantir que todos os estilos sejam atualizados."
}, },
"optionsUpdateInterval": { "optionsUpdateInterval": {
"message": "Intervalo de atualização automática do estilo de usuário em horas (especifique 0 para desativar)" "message": "Intervalo de atualização automática do estilo de usuário em horas (especifique 0 para desativar)"
}, },
"overwriteFileExport": {
"message": "Você gostaria de substituir um arquivo existente?"
},
"paginationCurrent": { "paginationCurrent": {
"message": "Pagina atual" "message": "Pagina atual"
}, },
@ -769,9 +625,6 @@
"previewTooltip": { "previewTooltip": {
"message": "Temporariamente aplica as alterações sem guardar.\nGuarde o estilo para tornar as alterações permanentes." "message": "Temporariamente aplica as alterações sem guardar.\nGuarde o estilo para tornar as alterações permanentes."
}, },
"readingStyles": {
"message": "Lendo estilos..."
},
"replace": { "replace": {
"message": "Substituir" "message": "Substituir"
}, },
@ -784,9 +637,6 @@
"retrieveBckp": { "retrieveBckp": {
"message": "Importar estilos" "message": "Importar estilos"
}, },
"retrieveDropboxSync": {
"message": "Importar do Dropbox"
},
"search": { "search": {
"message": "Pesquisar" "message": "Pesquisar"
}, },
@ -829,9 +679,6 @@
"sectionRestore": { "sectionRestore": {
"message": "Restaurar secção removida" "message": "Restaurar secção removida"
}, },
"sections": {
"message": "Secções"
},
"shortcuts": { "shortcuts": {
"message": "Atalhos" "message": "Atalhos"
}, },
@ -865,9 +712,6 @@
"styleBeautify": { "styleBeautify": {
"message": "Embelezar" "message": "Embelezar"
}, },
"styleBeautifyHint": {
"message": "Dica: clique com o botão direito no botão \"Embelezar\" ou use o atalho de teclado definido para embelezar sem mostrar esse painel"
},
"styleBeautifyIndentConditional": { "styleBeautifyIndentConditional": {
"message": "Indentar @media, @supports" "message": "Indentar @media, @supports"
}, },
@ -937,6 +781,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Número de secções não aplicadas devido ao uso incorreto de 'regexp ()'" "message": "Número de secções não aplicadas devido ao uso incorreto de 'regexp ()'"
}, },
"styleRegexpTestButton": {
"message": "Testar RegExp"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Separadores correspondentes" "message": "Separadores correspondentes"
}, },
@ -981,9 +828,6 @@
"stylusUnavailableForURLdetails": { "stylusUnavailableForURLdetails": {
"message": "Como precaução de segurança, o browser proíbe extensões de afetar as páginas embutidas (como chrome://version, a página de novo separador predefinido no Chrome 61, about:addons, e assim sucessivamente) tal como páginas de outras extensões. Cada browser também restringe acesso à sua própria galeria de extensões (como Chrome Web Store ou AMO)" "message": "Como precaução de segurança, o browser proíbe extensões de afetar as páginas embutidas (como chrome://version, a página de novo separador predefinido no Chrome 61, about:addons, e assim sucessivamente) tal como páginas de outras extensões. Cada browser também restringe acesso à sua própria galeria de extensões (como Chrome Web Store ou AMO)"
}, },
"syncDropboxStyles": {
"message": "Exportar para Dropbox"
},
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "O valor não pode ser guardado. Tente reduzir a quantidade de texto." "message": "O valor não pode ser guardado. Tente reduzir a quantidade de texto."
}, },
@ -1008,12 +852,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "O Stylus pode aceder URLs file:// apenas se ativar a checkbox correspondente para a extensão Stylus na página chrome://extensions" "message": "O Stylus pode aceder URLs file:// apenas se ativar a checkbox correspondente para a extensão Stylus na página chrome://extensions"
}, },
"unreachableMozSiteHintOldFF": {
"message": "Somente o Firefox 59 e o mais recente podem ser configurados para permitir que WebExtensions incluam elementos de estilo em sites protegidos por CSP como este."
},
"unzipStyles": {
"message": "Descompactando estilos..."
},
"updateAllCheckSucceededNoUpdate": { "updateAllCheckSucceededNoUpdate": {
"message": "Nenhuma atualização encontrada." "message": "Nenhuma atualização encontrada."
}, },
@ -1055,9 +893,6 @@
"updatesCurrentlyInstalled": { "updatesCurrentlyInstalled": {
"message": "Atualizações instaladas:" "message": "Atualizações instaladas:"
}, },
"uploadingFile": {
"message": "Enviando arquivo..."
},
"usercssAvoidOverwriting": { "usercssAvoidOverwriting": {
"message": "Por favor modifique o valor de @name ou @namespace para evitar sobrescrever um estilo existente." "message": "Por favor modifique o valor de @name ou @namespace para evitar sobrescrever um estilo existente."
}, },
@ -1070,6 +905,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Substituir o modelo predefinido para novos estilos de Usercss com o código atual?" "message": "Substituir o modelo predefinido para novos estilos de Usercss com o código atual?"
}, },
"usercssReplaceTemplateName": {
"message": "@name vazio substitui o modelo predefinido"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Insira o código aqui..." "message": "Insira o código aqui..."
}, },
@ -1081,8 +919,5 @@
}, },
"writeStyleForURL": { "writeStyleForURL": {
"message": "este URL" "message": "este URL"
},
"zipStyles": {
"message": "Compactando estilos..."
} }
} }

View File

@ -61,6 +61,9 @@
"author": { "author": {
"message": "Autor" "message": "Autor"
}, },
"backupMessage": {
"message": "Selectați un fișier sau drag-and-drop aici"
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Exportați teme" "message": "Exportați teme"
}, },
@ -230,6 +233,12 @@
"findStyles": { "findStyles": {
"message": "Găsiți teme" "message": "Găsiți teme"
}, },
"findStylesForSite": {
"message": "Gasiți mai multe teme pentru acest site."
},
"findStylesInlineTooltip": {
"message": "Arătați rezultatele căutării în această pagină."
},
"genericAdd": { "genericAdd": {
"message": "Adaugă" "message": "Adaugă"
}, },
@ -404,7 +413,7 @@
"message": "Hașurat" "message": "Hașurat"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus folosește un serviciu extern https://icons.duckduckgo.com" "message": "Stylus folosește un serviciu extern https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtre" "message": "Filtre"
@ -445,18 +454,12 @@
"menuShowBadge": { "menuShowBadge": {
"message": "Afișați numărul temelor active" "message": "Afișați numărul temelor active"
}, },
"meta_invalidCheckboxDefault": {
"message": "@var checkbox invalidă: valuarea trebuie să fie 0 sau 1"
},
"noStylesForSite": { "noStylesForSite": {
"message": "Nicio temă instalată pentru acest site." "message": "Nicio temă instalată pentru acest site."
}, },
"openManage": { "openManage": {
"message": "Managerul" "message": "Managerul"
}, },
"openOptions": {
"message": "Opțiuni"
},
"openStylesManager": { "openStylesManager": {
"message": "Deschideți managerul de teme" "message": "Deschideți managerul de teme"
}, },
@ -622,9 +625,6 @@
"sectionRestore": { "sectionRestore": {
"message": "Restaurează o secțiune ștearsă" "message": "Restaurează o secțiune ștearsă"
}, },
"sections": {
"message": "Secțiuni"
},
"shortcutsNote": { "shortcutsNote": {
"message": "Creeați keyboard shortcuts" "message": "Creeați keyboard shortcuts"
}, },
@ -789,9 +789,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "Stylus poate accesa file:// URLs doar când este activată opțiunea respectivă din pagina cu setări chrome://extensions" "message": "Stylus poate accesa file:// URLs doar când este activată opțiunea respectivă din pagina cu setări chrome://extensions"
}, },
"unreachableMozSiteHintOldFF": {
"message": "Doar Firefox 59 sau mai nou poate fi configurat să permită WebExtension-urilor să adauge elemente la site-uri CSP-protected precum acesta."
},
"updateAllCheckSucceededNoUpdate": { "updateAllCheckSucceededNoUpdate": {
"message": "Niciun update găsit." "message": "Niciun update găsit."
}, },
@ -845,6 +842,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Înlocuiți tema de bază a formatului Usercss cu acest cod?" "message": "Înlocuiți tema de bază a formatului Usercss cu acest cod?"
}, },
"usercssReplaceTemplateName": {
"message": "@name este gol și înlocuiețte valoarea de bază"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Introduce cod aici..." "message": "Introduce cod aici..."
}, },

View File

@ -68,7 +68,7 @@
"message": "Резервное копирование" "message": "Резервное копирование"
}, },
"backupMessage": { "backupMessage": {
"message": "Чтобы импортировать архив стилей, перетащите файл в эту страницу или нажмите кнопку Импорт.\n\nЧтобы экспортировать совместимый архив для старого Stylus ранее чем 1.5.18, кликните на кнопку Экспорт правой кнопкой мыши или с нажатой клавишей Shift." "message": "Нажмите «Импорт», чтобы выбрать файл или просто перетащите его на эту страницу"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "Экспорт стилей" "message": "Экспорт стилей"
@ -267,17 +267,6 @@
"disableStyleLabel": { "disableStyleLabel": {
"message": "Отключить" "message": "Отключить"
}, },
"draftAction": {
"message": "Выберите «Да» для загрузки черновика или «Нет», чтобы выкинуть его."
},
"draftTitle": {
"message": "Восстановление черновика, созданного $date$",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "Перетащите файл с резервной копией стилей в любое место этой страницы, чтобы импортировать его." "message": "Перетащите файл с резервной копией стилей в любое место этой страницы, чтобы импортировать его."
}, },
@ -304,9 +293,6 @@
} }
} }
}, },
"editorSettings": {
"message": "Настройки редактирования"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "Включить" "message": "Включить"
}, },
@ -316,9 +302,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "Исключить текущий URL" "message": "Исключить текущий URL"
}, },
"exportCompatible": {
"message": "Экспорт (режим совместимости)"
},
"exportLabel": { "exportLabel": {
"message": "Экспорт" "message": "Экспорт"
}, },
@ -357,6 +340,15 @@
"findStyles": { "findStyles": {
"message": "Найти стили" "message": "Найти стили"
}, },
"findStylesForSite": {
"message": "Найти ещё стили для этого веб-сайта"
},
"findStylesInline": {
"message": "и показать здесь"
},
"findStylesInlineTooltip": {
"message": "Показывать найденные стили в этом окне."
},
"genericAdd": { "genericAdd": {
"message": "Добавить" "message": "Добавить"
}, },
@ -390,9 +382,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "Сохранено" "message": "Сохранено"
}, },
"genericTest": {
"message": "Тест"
},
"genericTitle": { "genericTitle": {
"message": "Имя" "message": "Имя"
}, },
@ -402,9 +391,6 @@
"gettingStyles": { "gettingStyles": {
"message": "Получение всех стилей..." "message": "Получение всех стилей..."
}, },
"headerResizerHint": {
"message": "С клавишей Shift размер меняется только для такого же типа страниц, т.е. редактор, менеджер, установщик"
},
"helpAlt": { "helpAlt": {
"message": "Справка" "message": "Справка"
}, },
@ -500,18 +486,9 @@
"linkGetHelp": { "linkGetHelp": {
"message": "Помощь" "message": "Помощь"
}, },
"linkGetShareStyles": {
"message": "Поиск/публикация стилей"
},
"linkGetShareStylesInfo": {
"message": "Новый сайт userstyles.world, созданный и развиваемый сообществом энтузиастов и авторов пользовательских стилей с целью заменить userstyles.org, который стал таким медленным и зависающим за последний год, что многие авторы перестали обновлять свои стили."
},
"linkGetStyles": { "linkGetStyles": {
"message": "Скачать стили" "message": "Скачать стили"
}, },
"linkGetStylesInfo": {
"message": "Архивный сайт от одного энтузиаста и автора пользовательских стилей, зеркальная копия userstyles.org, ныне медленного и зависающего. Архив обновляется примерно раз в день."
},
"linkStylusWiki": { "linkStylusWiki": {
"message": "Вики" "message": "Вики"
}, },
@ -582,7 +559,7 @@
"message": "Обесцвечивать" "message": "Обесцвечивать"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Используется сторонний сервис https://icons.duckduckgo.com" "message": "Используется сторонний сервис https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Фильтры" "message": "Фильтры"
@ -842,15 +819,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "Другое" "message": "Другое"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "По настройке операционной системы"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "В ночное время:"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "Отключен. Настройка режима в стилях игнорируется."
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "Показывать команду \"Удалить\" в контекстном меню редактора" "message": "Показывать команду \"Удалить\" в контекстном меню редактора"
}, },
@ -860,12 +828,6 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "Выставляет верхний домен сайта в каждом iframe.\nПозволяет писать iframe-specific CSS следующим образом:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }" "message": "Выставляет верхний домен сайта в каждом iframe.\nПозволяет писать iframe-specific CSS следующим образом:\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }"
}, },
"optionsAdvancedExposeStyleName": {
"message": "Проставлять имя стиля"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "Проставляет имя стиля внутри страницы чтобы облегчить отладку стилей в инструментах разработчика (devtools). Чтобы применить новую настройку пожалуйста обновите вкладку со страницей."
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "Создавать стили в формате usercss" "message": "Создавать стили в формате usercss"
}, },
@ -911,9 +873,6 @@
"optionsHeading": { "optionsHeading": {
"message": "Настройки" "message": "Настройки"
}, },
"optionsIconAuto": {
"message": "В соответствии с ночным/дневным режимом"
},
"optionsIconDark": { "optionsIconDark": {
"message": "Тёмный интерфейс браузера" "message": "Тёмный интерфейс браузера"
}, },
@ -936,7 +895,7 @@
"message": "Сброс настроек" "message": "Сброс настроек"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "Нажмите иконку Stylus в верхней панели браузера на любой странице Stylus, в том числе и этой, затем нажмите «Найти стили»" "message": "Найти тему интерфейса Stylus"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "Дополнительно" "message": "Дополнительно"
@ -953,9 +912,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "Ничего" "message": "Ничего"
}, },
"optionsSyncPassword": {
"message": "Пароль"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "Подключено" "message": "Подключено"
}, },
@ -999,12 +955,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Синхронизировать" "message": "Синхронизировать"
}, },
"optionsSyncUrl": {
"message": "URL адрес"
},
"optionsSyncUsername": {
"message": "Имя"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "После импорта базы стилей из старой версии или из Stylish запустите проверку на обновления из менеджера стилей, чтобы удостовериться в обновлении всех стилей." "message": "После импорта базы стилей из старой версии или из Stylish запустите проверку на обновления из менеджера стилей, чтобы удостовериться в обновлении всех стилей."
}, },
@ -1047,9 +997,6 @@
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "Показать горячие клавиши" "message": "Показать горячие клавиши"
}, },
"popupManageSiteStyles": {
"message": "Управлять стилями сайта"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "Shift-ЛКМ или ПКМ откроет менеджер с стилями только для этого сайта." "message": "Shift-ЛКМ или ПКМ откроет менеджер с стилями только для этого сайта."
}, },
@ -1071,21 +1018,6 @@
"prefShowBadge": { "prefShowBadge": {
"message": "Показывать количество активных стилей для открытого сайта" "message": "Показывать количество активных стилей для открытого сайта"
}, },
"preferScheme": {
"message": "Ночной/дневной режим"
},
"preferSchemeAlways": {
"message": "Пока что игнорируется (стиль всегда включен), т.к. общий режим ночи/дня отключен"
},
"preferSchemeDark": {
"message": "Ночной"
},
"preferSchemeLight": {
"message": "Дневной"
},
"preferSchemeNone": {
"message": "Нет (всегда включен)"
},
"previewLabel": { "previewLabel": {
"message": "Предпросмотр" "message": "Предпросмотр"
}, },
@ -1113,6 +1045,9 @@
"readingStyles": { "readingStyles": {
"message": "Чтение стилей..." "message": "Чтение стилей..."
}, },
"reload": {
"message": "Перезагрузить расширение Stylus"
},
"replace": { "replace": {
"message": "Заменить" "message": "Заменить"
}, },
@ -1122,18 +1057,12 @@
"replaceWith": { "replaceWith": {
"message": "Заменить на" "message": "Заменить на"
}, },
"restoreTemplate": {
"message": "Восстановить шаблон по-умолчанию.\n\n(Не влияет на уже открытые страницы редактирования)"
},
"retrieveBckp": { "retrieveBckp": {
"message": "Импорт стилей" "message": "Импорт стилей"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "Импорт Dropbox" "message": "Импорт Dropbox"
}, },
"saveAsTemplate": {
"message": "Сохранить как шаблон"
},
"search": { "search": {
"message": "Поиск" "message": "Поиск"
}, },
@ -1173,6 +1102,9 @@
"searchResultWeeklyCount": { "searchResultWeeklyCount": {
"message": "Загрузок за неделю" "message": "Загрузок за неделю"
}, },
"searchStyleQueryHint": {
"message": "Поиск стилистических имён по падежам:\nнекоторые слова - все слова в любом порядке\n\"какая-то фраза\" - именно эта фраза без кавычек\n2020 - такой год также показывает стили, обновлённые в 2020 году"
},
"searchStylesAll": { "searchStylesAll": {
"message": "Все" "message": "Все"
}, },
@ -1206,18 +1138,12 @@
"sections": { "sections": {
"message": "Разделы" "message": "Разделы"
}, },
"settings": {
"message": "Настройки"
},
"shortcuts": { "shortcuts": {
"message": "Клавиши" "message": "Клавиши"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "Назначить клавишу" "message": "Назначить клавишу"
}, },
"shortcutsNoteFF": {
"message": "В Firefox 66+ вы можете открыть интерфейс клавиатурных сочетаний самостоятельно:\n1) правой кнопкой мыши на иконке Stylus в панели браузера, затем «Менеджер»\n(либо откройте about:addons из главного меню или нажмите Ctrl-Shift-A),\n2) на странице, которая откроется, нажмите иконку шестеренки в правом верхнем углу,\n3) выберите «Управление горячими клавишами расширений».\n\nТакже настроить клавиши можно и здесь."
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "новые сверху" "message": "новые сверху"
}, },
@ -1263,30 +1189,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "Включено" "message": "Включено"
}, },
"styleExcludeLabel": {
"message": "Личный список исключенных сайтов"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "Ошибка импорта формата Mozilla" "message": "Ошибка импорта формата Mozilla"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "Вставьте содержимое стиля в формате Mozilla" "message": "Вставьте содержимое стиля в формате Mozilla"
}, },
"styleIncludeLabel": {
"message": "Личный список включенных сайтов"
},
"styleInjectionImportance": {
"message": "Переключить важность стиля"
},
"styleInjectionOrder": {
"message": "Порядок применения стилей"
},
"styleInjectionOrderHint": {
"message": "Перетащите стиль для изменения его позиции. Стили применяются последовательно в нижеуказанном порядке, так что последующий стиль может изменить эффект предыдущих."
},
"styleInjectionOrderHint_prio": {
"message": "Важные стили всегда применяются после новоустановленных стилей, чтобы иметь «последнее слово». Для изменения важности стиля нажмите на его значок."
},
"styleInstall": { "styleInstall": {
"message": "Установить \"$stylename$\" в Stylus?", "message": "Установить \"$stylename$\" в Stylus?",
"placeholders": { "placeholders": {
@ -1329,15 +1237,6 @@
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "Стиль не был применён из-за некорректного regexp()" "message": "Стиль не был применён из-за некорректного regexp()"
}, },
"styleNotAppliedSchemeDark": {
"message": "Стиль активен только в ночном режиме"
},
"styleNotAppliedSchemeLight": {
"message": "Стиль активен только в дневном режиме"
},
"stylePreferSchemeLabel": {
"message": "Ночной/дневной режим"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "Некоторые 'regexp()' выражения не удалось скомпилировать." "message": "Некоторые 'regexp()' выражения не удалось скомпилировать."
}, },
@ -1347,6 +1246,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Кол-во пропущенных разделов из-за неправильного regexp()" "message": "Кол-во пропущенных разделов из-за неправильного regexp()"
}, },
"styleRegexpTestButton": {
"message": "Тест регулярки"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Соответствующие вкладки" "message": "Соответствующие вкладки"
}, },
@ -1368,9 +1270,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "Сохранить" "message": "Сохранить"
}, },
"styleSettings": {
"message": "Настройки стиля"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "Формат кода Mozilla можно отправлять на сайт userstyles.org и использовать в дополнении Stylish для Firefox." "message": "Формат кода Mozilla можно отправлять на сайт userstyles.org и использовать в дополнении Stylish для Firefox."
}, },
@ -1388,9 +1287,6 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "Стиль был изменён вне редактора. Перезагрузить?" "message": "Стиль был изменён вне редактора. Перезагрузить?"
}, },
"styleUpdateUrlLabel": {
"message": "URL адрес обновлений"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "Такие адреса не поддерживаются." "message": "Такие адреса не поддерживаются."
}, },
@ -1406,16 +1302,8 @@
"syncError": { "syncError": {
"message": "Сбой синхронизации" "message": "Сбой синхронизации"
}, },
"syncErrorLock": {
"message": "Хранилище данных уже занято. Доступ закрыт до $TIME$",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "Ошибка синхронизации. Произведен выход из учетной записи.\n\nПопробуйте подключиться еще раз в настройках Stylus." "message": "Синхронизация не удалась.\nПопробуйте повторно войти в систему в опциях Stylus:\nсначала нажмите \"отсоединить\", затем \"подключить\"."
}, },
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "Ошибка сохранения. Попробуйте уменьшить количество текста." "message": "Ошибка сохранения. Попробуйте уменьшить количество текста."
@ -1506,6 +1394,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Заменить шаблон по умолчанию для нового стиля в формате Usercss текущим кодом?" "message": "Заменить шаблон по умолчанию для нового стиля в формате Usercss текущим кодом?"
}, },
"usercssReplaceTemplateName": {
"message": "Пустой @name заменяет шаблон по умолчанию"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Место для CSS кода..." "message": "Место для CSS кода..."
}, },

View File

@ -76,15 +76,9 @@
"cm_theme": { "cm_theme": {
"message": "Тема" "message": "Тема"
}, },
"confirmDelete": {
"message": "Избриши"
},
"confirmNo": { "confirmNo": {
"message": "Не" "message": "Не"
}, },
"confirmSave": {
"message": "Сачувај"
},
"confirmStop": { "confirmStop": {
"message": "Заустави" "message": "Заустави"
}, },
@ -112,9 +106,6 @@
"disableStyleLabel": { "disableStyleLabel": {
"message": "Онемогући" "message": "Онемогући"
}, },
"editDeleteText": {
"message": "Избриши"
},
"editGotoLine": { "editGotoLine": {
"message": "Иди на ред (или line:col)" "message": "Иди на ред (или line:col)"
}, },
@ -138,11 +129,8 @@
"exportLabel": { "exportLabel": {
"message": "Извези" "message": "Извези"
}, },
"genericAdd": { "findStylesForSite": {
"message": "Додај" "message": "Пронађи још стилова за овај сајт"
},
"genericEnabledLabel": {
"message": "Омогућено"
}, },
"helpAlt": { "helpAlt": {
"message": "Помоћ" "message": "Помоћ"
@ -206,15 +194,9 @@
"openManage": { "openManage": {
"message": "Управљај инсталираним стиловима" "message": "Управљај инсталираним стиловима"
}, },
"openOptions": {
"message": "Опције"
},
"optionsHeading": { "optionsHeading": {
"message": "Опције" "message": "Опције"
}, },
"optionsSyncUrl": {
"message": "УРЛ"
},
"popupStylesFirst": { "popupStylesFirst": {
"message": "Излистај стилове пре команди у менију дугмета на алатној траци" "message": "Излистај стилове пре команди у менију дугмета на алатној траци"
}, },
@ -245,9 +227,6 @@
"sectionRemove": { "sectionRemove": {
"message": "Уклони одељак" "message": "Уклони одељак"
}, },
"sections": {
"message": "Одељци"
},
"styleBadRegexp": { "styleBadRegexp": {
"message": "Регуларни израз је неисправан." "message": "Регуларни израз је неисправан."
}, },

View File

@ -70,6 +70,9 @@
"backupButtons": { "backupButtons": {
"message": "Säkerhetskopiera" "message": "Säkerhetskopiera"
}, },
"backupMessage": {
"message": "Välj en fil eller dra och släpp till den här sidan."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Exportera stilar" "message": "Exportera stilar"
}, },
@ -287,6 +290,12 @@
"findStyles": { "findStyles": {
"message": "Hitta stilar" "message": "Hitta stilar"
}, },
"findStylesForSite": {
"message": "Hitta fler stilar för denna webbplats"
},
"findStylesInlineTooltip": {
"message": "Visa sökresultat i det här fönstret."
},
"genericAdd": { "genericAdd": {
"message": "Lägg till" "message": "Lägg till"
}, },
@ -473,7 +482,7 @@
"message": "Nedtonade" "message": "Nedtonade"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus använder en extern tjänst https://icons.duckduckgo.com" "message": "Stylus använder en extern tjänst https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filter" "message": "Filter"
@ -815,9 +824,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "Synkronisera nu" "message": "Synkronisera nu"
}, },
"optionsSyncUrl": {
"message": "Webbadress"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "När du importerar säkerhetskopior av stilar från gammal version eller från Stylish, gör en engångskontroll för uppdateringar manuellt i stilhanteraren för att säkerställa att alla stilar uppdateras." "message": "När du importerar säkerhetskopior av stilar från gammal version eller från Stylish, gör en engångskontroll för uppdateringar manuellt i stilhanteraren för att säkerställa att alla stilar uppdateras."
}, },
@ -956,9 +962,6 @@
"sectionRestore": { "sectionRestore": {
"message": "Återställ borttagen sektion" "message": "Återställ borttagen sektion"
}, },
"sections": {
"message": "Sektioner"
},
"shortcuts": { "shortcuts": {
"message": "Genvägar" "message": "Genvägar"
}, },
@ -1058,6 +1061,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "Antal avsnitt som inte tillämpas på grund av felaktig användning av \"regexp()\"" "message": "Antal avsnitt som inte tillämpas på grund av felaktig användning av \"regexp()\""
}, },
"styleRegexpTestButton": {
"message": "RegExp-test"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Matchande flikar" "message": "Matchande flikar"
}, },
@ -1129,9 +1135,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "Stylus kan endast komma åt fil:// webbadresser om du aktiverar motsvarande kryssruta för Stylus-tillägg på chrome://extensions-sidan." "message": "Stylus kan endast komma åt fil:// webbadresser om du aktiverar motsvarande kryssruta för Stylus-tillägg på chrome://extensions-sidan."
}, },
"unreachableMozSiteHintOldFF": {
"message": "Endast Firefox 59 och nyare kan konfigureras för att tillåta WebExtensions att lägga till stilelement på CSP-skyddade webbplatser som den här."
},
"unzipStyles": { "unzipStyles": {
"message": "Packar upp stilar..." "message": "Packar upp stilar..."
}, },
@ -1191,6 +1194,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Ersätt standardmallen för nya Usercss-stilar med den aktuella koden?" "message": "Ersätt standardmallen för nya Usercss-stilar med den aktuella koden?"
}, },
"usercssReplaceTemplateName": {
"message": "Tom @name ersätter standardmallen"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Lägg in kod här..." "message": "Lägg in kod här..."
}, },

View File

@ -22,12 +22,6 @@
"appliesToEverything": { "appliesToEverything": {
"message": "అన్నిటికీ" "message": "అన్నిటికీ"
}, },
"confirmDelete": {
"message": "తొలగించు"
},
"confirmSave": {
"message": "భద్రపరచు"
},
"deleteStyleConfirm": { "deleteStyleConfirm": {
"message": "మీరు నజంగానే ఈ శైలిని తొలగించాలనుకుంటున్నారా?" "message": "మీరు నజంగానే ఈ శైలిని తొలగించాలనుకుంటున్నారా?"
}, },
@ -37,18 +31,12 @@
"disableStyleLabel": { "disableStyleLabel": {
"message": "అచేతనించు" "message": "అచేతనించు"
}, },
"editDeleteText": {
"message": "తొలగించు"
},
"editStyleLabel": { "editStyleLabel": {
"message": "మార్చు" "message": "మార్చు"
}, },
"enableStyleLabel": { "enableStyleLabel": {
"message": "చేతనించు" "message": "చేతనించు"
}, },
"genericAdd": {
"message": "చేర్చు"
},
"helpAlt": { "helpAlt": {
"message": "సహాయం" "message": "సహాయం"
}, },
@ -58,9 +46,6 @@
"manageTitle": { "manageTitle": {
"message": "స్టైలిష్" "message": "స్టైలిష్"
}, },
"sections": {
"message": "విభాగాలు"
},
"styleSaveLabel": { "styleSaveLabel": {
"message": "భద్రపరచు" "message": "భద్రపరచు"
} }

View File

@ -58,6 +58,9 @@
"backupButtons": { "backupButtons": {
"message": "Yedek" "message": "Yedek"
}, },
"backupMessage": {
"message": "Bir dosya seçin veya bu sayfaya sürükleyip bırakın."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Dışa aktar" "message": "Dışa aktar"
}, },
@ -284,6 +287,15 @@
"findStyles": { "findStyles": {
"message": "Stil bul" "message": "Stil bul"
}, },
"findStylesForSite": {
"message": "Bu site için başka stiller bul"
},
"findStylesInline": {
"message": "Hizada"
},
"findStylesInlineTooltip": {
"message": "Arama sonuçlarını bu pencerede görüntüleyin."
},
"genericAdd": { "genericAdd": {
"message": "Ekle" "message": "Ekle"
}, },
@ -414,7 +426,7 @@
"message": "Gri renkte" "message": "Gri renkte"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus harici bir servis kullanır https://icons.duckduckgo.com" "message": "Stylus harici bir servis kullanır https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "Filtreler" "message": "Filtreler"
@ -452,9 +464,6 @@
"openManage": { "openManage": {
"message": "Yüklü stilleri yönet" "message": "Yüklü stilleri yönet"
}, },
"openOptions": {
"message": "Seçenekler"
},
"openStylesManager": { "openStylesManager": {
"message": "Stil yöneticisini aç" "message": "Stil yöneticisini aç"
}, },
@ -602,9 +611,6 @@
"sectionRemove": { "sectionRemove": {
"message": "Bölümü kaldır" "message": "Bölümü kaldır"
}, },
"sections": {
"message": "Bölümler"
},
"shortcuts": { "shortcuts": {
"message": "Kısayollar" "message": "Kısayollar"
}, },
@ -658,6 +664,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "'Regexp ()' yanlış kullanımı nedeniyle uygulanmayan bölüm sayısı" "message": "'Regexp ()' yanlış kullanımı nedeniyle uygulanmayan bölüm sayısı"
}, },
"styleRegexpTestButton": {
"message": "RegExp testi"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "Eşleşen sekmeler" "message": "Eşleşen sekmeler"
}, },
@ -773,6 +782,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "Yeni Usercss stilleri için varsayılan şablonu geçerli kodla değiştir?" "message": "Yeni Usercss stilleri için varsayılan şablonu geçerli kodla değiştir?"
}, },
"usercssReplaceTemplateName": {
"message": "Boş @name, varsayılan şablonun yerini alır"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "Kodu buraya ekle..." "message": "Kodu buraya ekle..."
}, },

View File

@ -61,6 +61,9 @@
"backupButtons": { "backupButtons": {
"message": "Резервне копіювання" "message": "Резервне копіювання"
}, },
"backupMessage": {
"message": "Виберіть файл або перетягніть на цю сторінку."
},
"bckpInstStyles": { "bckpInstStyles": {
"message": "Експорт стилів" "message": "Експорт стилів"
}, },
@ -97,12 +100,18 @@
"disableStyleLabel": { "disableStyleLabel": {
"message": "Вимкнути" "message": "Вимкнути"
}, },
"editDeleteText": {
"message": "Видалити"
},
"findStyles": { "findStyles": {
"message": "Знайти стилі" "message": "Знайти стилі"
}, },
"findStylesForSite": {
"message": "Знайти більше стилів для цього сайту"
},
"findStylesInline": {
"message": "і показати тут"
},
"findStylesInlineTooltip": {
"message": "Показувати знайдені стилі в цьому вікні."
},
"genericAdd": { "genericAdd": {
"message": "Додати" "message": "Додати"
}, },
@ -169,20 +178,11 @@
"importReportTitle": { "importReportTitle": {
"message": "Імпорт стилів закінчено" "message": "Імпорт стилів закінчено"
}, },
"installUpdate": {
"message": "Встановити оновлення"
},
"manageFavicons": { "manageFavicons": {
"message": "Піктограми для цільових сайтів" "message": "Піктограми для цільових сайтів"
}, },
"manageFilters": {
"message": "Фільтри"
},
"manageHeading": {
"message": "Встановити Styles"
},
"manageNewUI": { "manageNewUI": {
"message": "Новий макет інтерфейсу управління користувача" "message": "Новий інтерфейс"
}, },
"meta_unknownJSONLiteral": { "meta_unknownJSONLiteral": {
"message": "Невірний JSON: $literal$не є дійсним літералом JSON", "message": "Невірний JSON: $literal$не є дійсним літералом JSON",
@ -193,13 +193,7 @@
} }
}, },
"openManage": { "openManage": {
"message": "Керування" "message": "Менеджер"
},
"openOptions": {
"message": "Налаштування"
},
"optionsHeading": {
"message": "Налаштування"
}, },
"optionsReset": { "optionsReset": {
"message": "Скидання налаштувань до значень за замовчуванням" "message": "Скидання налаштувань до значень за замовчуванням"
@ -213,39 +207,9 @@
"optionsSyncLogin": { "optionsSyncLogin": {
"message": "Логін" "message": "Логін"
}, },
"optionsSyncStatusRelogin": {
"message": "Сеанс закінчився, будь ласка, увійдіть ще раз."
},
"paginationNext": {
"message": "Наступна сторінка"
},
"paginationPrevious": {
"message": "Попередня сторінка"
},
"replace": {
"message": "Замінити"
},
"replaceAll": {
"message": "Замінити все"
},
"retrieveBckp": { "retrieveBckp": {
"message": "Імпорт стилів" "message": "Імпорт стилів"
}, },
"search": {
"message": "Пошук"
},
"searchStylesAll": {
"message": "Усі"
},
"searchStylesCode": {
"message": "CSS код"
},
"searchStylesHelp": {
"message": "</> або <Ctrl-F>клавіша фокусує поле пошуку.\nРежим за замовчуванням — це пошук у звичайному тексті для всіх термінів, розділених пробілами, у будь-якому порядку.\nТочні слова: оберніть запит у подвійні лапки, напр. <.header ~ div\">\nРегулярні вирази: включають косі риски та прапорці, напр.</body.*?\\ba\\b/i>\n«By URL» в селекторі області: знаходить стилі, які застосовуються до повністю вказаної URL-адреси, напр. https://www.example.org/\n\"Metadata\" в селектрі області: шукає в іменах, \"applies to\" специфікаторів, URL-адреси встановлення, URL-адресі оновлення та всьому блоку метаданих для стилів CSS користувача."
},
"searchStylesName": {
"message": "Назва"
},
"sectionCode": { "sectionCode": {
"message": "Код" "message": "Код"
}, },
@ -256,29 +220,14 @@
"message": "Облагородити" "message": "Облагородити"
}, },
"styleCancelEditLabel": { "styleCancelEditLabel": {
"message": "Повернутися до керування" "message": "Всі стилі"
},
"styleEnabledLabel": {
"message": "Увімкнено"
},
"stylePreferSchemeLabel": {
"message": "Темна/Світла тема"
},
"styleSaveLabel": {
"message": "Зберегти"
}, },
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "Помилка синхронізації. Ви вийшли із системи. \nСпробуйте повторно увійти до системи в налаштуваннях Stylus." "message": "Помилка синхронізації.\nСпробуйте повторно ввійти в налаштування Stylus:\nнатисніть спочатку «від’єднати», а потім «підключити». "
}, },
"toggleStyle": { "toggleStyle": {
"message": "Включити/виключити стиль" "message": "Включити/виключити стиль"
}, },
"writeStyleFor": {
"message": "Створити стиль для:"
},
"writeStyleForURL": {
"message": "цей URL"
},
"zipStyles": { "zipStyles": {
"message": "Запаковування стилів ... " "message": "Запаковування стилів ... "
} }

View File

@ -71,7 +71,7 @@
"message": "备份" "message": "备份"
}, },
"backupMessage": { "backupMessage": {
"message": "选择/拖拽 JSON 备份文件到本页面来导入备份。\n\n若要导出 Stylus V1.5.18 之前的兼容备份请右击或Shift+左击「导出」按钮。" "message": "拖拽JSON备份文件到本页面也可导入。"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "导出所有样式" "message": "导出所有样式"
@ -270,17 +270,6 @@
"disableStyleLabel": { "disableStyleLabel": {
"message": "禁用" "message": "禁用"
}, },
"draftAction": {
"message": "选择“是”以加载此草稿,或“否”以丢弃它。"
},
"draftTitle": {
"message": "恢复 $date$ 个小时前的草稿",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "拖放JSON备份文件到管理器页面即可导入!" "message": "拖放JSON备份文件到管理器页面即可导入!"
}, },
@ -307,9 +296,6 @@
} }
} }
}, },
"editorSettings": {
"message": "编辑器设置"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "启用" "message": "启用"
}, },
@ -319,9 +305,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "排除当前链接" "message": "排除当前链接"
}, },
"exportCompatible": {
"message": "导出(兼容模式)"
},
"exportLabel": { "exportLabel": {
"message": "导出" "message": "导出"
}, },
@ -360,15 +343,21 @@
"findStyles": { "findStyles": {
"message": "查找样式" "message": "查找样式"
}, },
"findStylesForSite": {
"message": "查找该域名的在线样式"
},
"findStylesInline": {
"message": "嵌入此页"
},
"findStylesInlineTooltip": {
"message": "将搜索结果嵌入到此页面显示"
},
"genericAdd": { "genericAdd": {
"message": "添加" "message": "添加"
}, },
"genericClone": { "genericClone": {
"message": "创建副本" "message": "创建副本"
}, },
"genericDescription": {
"message": "描述"
},
"genericDisabledLabel": { "genericDisabledLabel": {
"message": "禁用" "message": "禁用"
}, },
@ -393,9 +382,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "已保存" "message": "已保存"
}, },
"genericTest": {
"message": "测试"
},
"genericTitle": { "genericTitle": {
"message": "标题" "message": "标题"
}, },
@ -405,9 +391,6 @@
"gettingStyles": { "gettingStyles": {
"message": "正在获取所有样式..." "message": "正在获取所有样式..."
}, },
"headerResizerHint": {
"message": "仅在此类 编辑器/管理器/安装器 UI , 按住 Shift 可调整大小"
},
"helpAlt": { "helpAlt": {
"message": "帮助" "message": "帮助"
}, },
@ -503,18 +486,9 @@
"linkGetHelp": { "linkGetHelp": {
"message": "帮助" "message": "帮助"
}, },
"linkGetShareStyles": {
"message": "获取|分享 样式"
},
"linkGetShareStylesInfo": {
"message": "由社区驱动的新站点 userstyles.world 是由 userstyle 作者创建的,目的是取代 userstyles.org该站点在过去一年中缓慢且反应迟钝以至于许多作者停止更新他们的样式。"
},
"linkGetStyles": { "linkGetStyles": {
"message": "获取样式" "message": "获取样式"
}, },
"linkGetStylesInfo": {
"message": "该存档站点由 userstyle 社区成员创建,约每天更新一次其内容,用于备份缓慢且无响应的 userstyles.org。 "
},
"linkTranslate": { "linkTranslate": {
"message": "翻译" "message": "翻译"
}, },
@ -582,7 +556,7 @@
"message": "显示为灰色图标" "message": "显示为灰色图标"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus 使用外部服务 https://icons.duckduckgo.com 来获取图标" "message": "Stylus 使用外部服务 https://www.google.com/s2/favicons 来获取图标"
}, },
"manageFilters": { "manageFilters": {
"message": "过滤器" "message": "过滤器"
@ -623,6 +597,9 @@
"manageOnlyUsercss": { "manageOnlyUsercss": {
"message": "只显示 UserCSS" "message": "只显示 UserCSS"
}, },
"manageTitle": {
"message": "Stylus管理器"
},
"menuShowBadge": { "menuShowBadge": {
"message": "计数器角标" "message": "计数器角标"
}, },
@ -788,17 +765,6 @@
} }
} }
}, },
"meta_unknownMetaTypo": {
"message": "可能的 @$keyOk$ ? 未知元数据 @$keyErr$",
"placeholders": {
"keyErr": {
"content": "$1"
},
"keyOk": {
"content": "$2"
}
}
},
"meta_unknownPreprocessor": { "meta_unknownPreprocessor": {
"message": "未知的预处理器 @preprocessor$preprocessor$", "message": "未知的预处理器 @preprocessor$preprocessor$",
"placeholders": { "placeholders": {
@ -842,15 +808,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "高级设置" "message": "高级设置"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "按系统偏好设置"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "按夜间时间"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "已停用。樣式中的深色/淺色設定會被忽略。"
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "向编辑器右键菜单添加“删除”" "message": "向编辑器右键菜单添加“删除”"
}, },
@ -860,12 +817,6 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "给每个iframe内嵌框架的html注入 stylus-iframe=\"父域名hostname\" 属性(值)\n\n用途: 用作iframe选择器来一键确定 iframe 父窗(window.top) 页面 的唯一选择器.\n\n示例 html[stylus-iframe] 或 html[stylus-iframe$=\"twitter.com\"] h1 {...}\n\n详见源码github.com/openstyles/stylus/blob/master/content/apply.js#L270" "message": "给每个iframe内嵌框架的html注入 stylus-iframe=\"父域名hostname\" 属性(值)\n\n用途: 用作iframe选择器来一键确定 iframe 父窗(window.top) 页面 的唯一选择器.\n\n示例 html[stylus-iframe] 或 html[stylus-iframe$=\"twitter.com\"] h1 {...}\n\n详见源码github.com/openstyles/stylus/blob/master/content/apply.js#L270"
}, },
"optionsAdvancedExposeStyleName": {
"message": "暴露样式名称"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "在页面中暴露样式名称以方便在 DevTools 中调试样式。请重新加载标签页以应用新设置。"
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "将新样式的格式设为UserCSS" "message": "将新样式的格式设为UserCSS"
}, },
@ -911,9 +862,6 @@
"optionsHeading": { "optionsHeading": {
"message": "选项" "message": "选项"
}, },
"optionsIconAuto": {
"message": "符合深色/淺色模式"
},
"optionsIconDark": { "optionsIconDark": {
"message": "高对比度暗色图标" "message": "高对比度暗色图标"
}, },
@ -936,7 +884,7 @@
"message": "恢复默认值" "message": "恢复默认值"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "點擊任何 Stylus 頁面(包含這個)的瀏覽器工具列中的 Stylus 圖示,然後點擊「尋找樣式」" "message": "查找 Stylus UI 主题ᐝ"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "附加选项" "message": "附加选项"
@ -953,9 +901,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "无" "message": "无"
}, },
"optionsSyncPassword": {
"message": "密码"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "已连接" "message": "已连接"
}, },
@ -990,21 +935,12 @@
} }
} }
}, },
"optionsSyncStatusRelogin": {
"message": "工作階段已過期,請再次登入。"
},
"optionsSyncStatusSyncing": { "optionsSyncStatusSyncing": {
"message": "同步中..." "message": "同步中..."
}, },
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "现在同步" "message": "现在同步"
}, },
"optionsSyncUrl": {
"message": "URL "
},
"optionsSyncUsername": {
"message": "用户名"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "从旧版的 Stylus 或 Stylish 中导入备份样式时,\n请手动检测升级一次以确保所有样式都可更新到最新版本。" "message": "从旧版的 Stylus 或 Stylish 中导入备份样式时,\n请手动检测升级一次以确保所有样式都可更新到最新版本。"
}, },
@ -1047,9 +983,6 @@
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "查看Popup列表快捷键" "message": "查看Popup列表快捷键"
}, },
"popupManageSiteStyles": {
"message": "管理网站样式"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "右击 || Shift+左击: 打开管理器且筛选当前URL(不含iframe)" "message": "右击 || Shift+左击: 打开管理器且筛选当前URL(不含iframe)"
}, },
@ -1071,50 +1004,17 @@
"prefShowBadge": { "prefShowBadge": {
"message": "计数器角标 (或图标右键)" "message": "计数器角标 (或图标右键)"
}, },
"preferScheme": {
"message": "深色/淺色模式偏好設定"
},
"preferSchemeAlways": {
"message": "目前被忽略(樣式一律套用),因為全域深色/淺色模式已被停用"
},
"preferSchemeDark": {
"message": "深色"
},
"preferSchemeLight": {
"message": "淺色"
},
"preferSchemeNone": {
"message": "無(一律套用)"
},
"previewLabel": { "previewLabel": {
"message": "实时预览" "message": "实时预览"
}, },
"previewTooltip": { "previewTooltip": {
"message": "无需保存即可临时预览样式效果" "message": "无需保存即可临时预览样式效果"
}, },
"publish": {
"message": "发布"
},
"publishPush": {
"message": "发布更新"
},
"publishReconnect": {
"message": "尝试断开连接然后再次发布"
},
"publishRetry": {
"message": "Stylus 仍在尝试发布此样式,但如果您没有看到身份验证活动或弹出窗口,您可以重试, 现在重试 "
},
"publishStyle": {
"message": "发布样式"
},
"publishUsw": {
"message": "使用 <userstyles.world>"
},
"readingStyles": { "readingStyles": {
"message": "正在读取样式..." "message": "正在读取样式..."
}, },
"reload": { "reload": {
"message": "重启" "message": "重启 Stylus"
}, },
"replace": { "replace": {
"message": "替换" "message": "替换"
@ -1125,18 +1025,12 @@
"replaceWith": { "replaceWith": {
"message": "替换为" "message": "替换为"
}, },
"restoreTemplate": {
"message": "還原預設範本。\n\n不會變更目前開啟的編輯器頁面。"
},
"retrieveBckp": { "retrieveBckp": {
"message": "导入所有样式" "message": "导入所有样式"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "从 Dropbox 导入" "message": "从 Dropbox 导入"
}, },
"saveAsTemplate": {
"message": "另存为模板"
},
"search": { "search": {
"message": "搜索" "message": "搜索"
}, },
@ -1161,12 +1055,6 @@
"searchResultNoneFound": { "searchResultNoneFound": {
"message": "没有找到与此页面相关的样式。" "message": "没有找到与此页面相关的样式。"
}, },
"searchResultNotMatching": {
"message": "样式已安装但不适于当前 URL"
},
"searchResultNotMatchingNote": {
"message": "尝试请求该用户样式的作者添加 URL。\n\n您也可以在管理器中打开样式自己编辑\n但请注意这会禁用此样式的自动更新。"
},
"searchResultRating": { "searchResultRating": {
"message": "评价" "message": "评价"
}, },
@ -1212,18 +1100,12 @@
"sections": { "sections": {
"message": "章节" "message": "章节"
}, },
"settings": {
"message": "设置"
},
"shortcuts": { "shortcuts": {
"message": "快捷键" "message": "快捷键"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "设置快捷键" "message": "设置快捷键"
}, },
"shortcutsNoteFF": {
"message": "在 Firefox 66 或更新的版本中,您可以手動開啟內建的快捷鍵使用者介面:\n1) 右鍵點擊工具列中的 Stylus 圖示並選擇「管理」\n或是透過主選單或 Ctrl-Shift-A 開啟 about:addons\n2) 在開啟的頁面中點擊右上角的齒輪圖示,\n3) 選擇「管理擴充套件快捷鍵」。\n\n您也可以在此自訂快捷鍵。"
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "最新的优先" "message": "最新的优先"
}, },
@ -1269,30 +1151,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "启用" "message": "启用"
}, },
"styleExcludeLabel": {
"message": "自定义排除网站"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "导入 Mozilla 格式失败" "message": "导入 Mozilla 格式失败"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "粘贴 Mozilla 格式代码" "message": "粘贴 Mozilla 格式代码"
}, },
"styleIncludeLabel": {
"message": "自定义包括网站"
},
"styleInjectionImportance": {
"message": "切换样式的优先级"
},
"styleInjectionOrder": {
"message": "樣式注入順序"
},
"styleInjectionOrderHint": {
"message": "拖曳樣式可變更其位置。樣式按下列順序注入,所以清單下方的樣式可覆寫較早樣式。"
},
"styleInjectionOrderHint_prio": {
"message": "下面列出的重要样式始终是最后注入的,因此它们可以覆盖任何新安装的样式。单击样式的标记以切换其重要性。"
},
"styleInstall": { "styleInstall": {
"message": "要将“$stylename$”安装到 Stylus 中吗?", "message": "要将“$stylename$”安装到 Stylus 中吗?",
"placeholders": { "placeholders": {
@ -1329,21 +1193,9 @@
"styleMozillaFormatHeading": { "styleMozillaFormatHeading": {
"message": "Mozilla 格式" "message": "Mozilla 格式"
}, },
"styleName": {
"message": "樣式名稱"
},
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "正则表达式错误,样式无法正常运行" "message": "正则表达式错误,样式无法正常运行"
}, },
"styleNotAppliedSchemeDark": {
"message": "此样式仅在深色模式下应用"
},
"styleNotAppliedSchemeLight": {
"message": "此样式仅在浅色模式下应用"
},
"stylePreferSchemeLabel": {
"message": "深色/浅色模式"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "部分正则表达式无法生效。" "message": "部分正则表达式无法生效。"
}, },
@ -1353,6 +1205,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "多个部分代码无法正常处理正则表达式" "message": "多个部分代码无法正常处理正则表达式"
}, },
"styleRegexpTestButton": {
"message": "正则测试"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "匹配到标签页的正则" "message": "匹配到标签页的正则"
}, },
@ -1374,9 +1229,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "保存" "message": "保存"
}, },
"styleSettings": {
"message": "样式设置"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "导入 FireFox 格式即 @-moz-document ...的 CSS 代码后,会自动转换成 Stylus 使用的分段式 CSS 代码。\n反过来分段式的 CSS 也可以导出为 FireFox 格式。\n需要注意的是如果你要向 userstyles.org 提交你的代码,则必须使用 FireFox 格式。\n\n导入快捷键: 剪贴板含有 @-moz-document ....代码,则可以直接在编辑器里 Ctrl+V 会自动弹出导入对话框" "message": "导入 FireFox 格式即 @-moz-document ...的 CSS 代码后,会自动转换成 Stylus 使用的分段式 CSS 代码。\n反过来分段式的 CSS 也可以导出为 FireFox 格式。\n需要注意的是如果你要向 userstyles.org 提交你的代码,则必须使用 FireFox 格式。\n\n导入快捷键: 剪贴板含有 @-moz-document ....代码,则可以直接在编辑器里 Ctrl+V 会自动弹出导入对话框"
}, },
@ -1394,9 +1246,6 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "样式已经在编辑器外被修改。需要重新加载样式吗?" "message": "样式已经在编辑器外被修改。需要重新加载样式吗?"
}, },
"styleUpdateUrlLabel": {
"message": "更新 URL"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "Stylus 无法介入到此类页面" "message": "Stylus 无法介入到此类页面"
}, },
@ -1409,20 +1258,6 @@
"syncDropboxStyles": { "syncDropboxStyles": {
"message": "导出至 Dropbox" "message": "导出至 Dropbox"
}, },
"syncError": {
"message": "同步失敗"
},
"syncErrorLock": {
"message": "数据库已在使用中。锁定将在 $TIME$ 过期",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": {
"message": "同步失敗。您已被登出。\n嘗試在 Stylus 選項中重新登入。"
},
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "不能保存该值,请尝试减少文本数量。" "message": "不能保存该值,请尝试减少文本数量。"
}, },
@ -1447,12 +1282,6 @@
"unreachableFileHint": { "unreachableFileHint": {
"message": "! Stylus 无法介入到官方商店页面 !\n若是文件页面 或 小号无痕页面 请检查:\n1) 右键 - Stylus图标 并点击-\"管理扩展程序\"\n2) 勾选 - \"允许访问文件网址\" 以在 file:// 工作\n3) 勾选 - \"在无痕模式下启用\" 以在无痕页面和小号标签页工作 (有的浏览器带有小号功能、如CentBrowser)" "message": "! Stylus 无法介入到官方商店页面 !\n若是文件页面 或 小号无痕页面 请检查:\n1) 右键 - Stylus图标 并点击-\"管理扩展程序\"\n2) 勾选 - \"允许访问文件网址\" 以在 file:// 工作\n3) 勾选 - \"在无痕模式下启用\" 以在无痕页面和小号标签页工作 (有的浏览器带有小号功能、如CentBrowser)"
}, },
"unreachableMozSiteHint": {
"message": "在 Firefox ⩾60 版本里, 你需要在 <about:config> 里移除 extensions.webextensions.restrictedDomains"
},
"unreachableMozSiteHintOldFF": {
"message": "仅 Firefox 59 或更新的版本才能让你设置 WebExtensions 在例如这样一个有 CSP 保护的页面上新增样式。"
},
"unzipStyles": { "unzipStyles": {
"message": "正在解压样式..." "message": "正在解压样式..."
}, },
@ -1512,6 +1341,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "使用当前的 UserStyle 替换为新的UserCSS默认模板 ?" "message": "使用当前的 UserStyle 替换为新的UserCSS默认模板 ?"
}, },
"usercssReplaceTemplateName": {
"message": "该赋值为空的保存可设置默认模板"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "在此插入代码..." "message": "在此插入代码..."
}, },

View File

@ -12,7 +12,7 @@
"message": "透明度" "message": "透明度"
}, },
"appliesAdd": { "appliesAdd": {
"message": "添加" "message": "添加部分"
}, },
"appliesDisplay": { "appliesDisplay": {
"message": "应用于:$applies$", "message": "应用于:$applies$",
@ -35,7 +35,7 @@
"message": "应用于:" "message": "应用于:"
}, },
"appliesLineWidgetLabel": { "appliesLineWidgetLabel": {
"message": "显示「应用于」信息" "message": "显示应用于部件"
}, },
"appliesLineWidgetWarning": { "appliesLineWidgetWarning": {
"message": "对经过压缩的 CSS 无效" "message": "对经过压缩的 CSS 无效"
@ -71,7 +71,7 @@
"message": "备份" "message": "备份"
}, },
"backupMessage": { "backupMessage": {
"message": "选择/拖拽 JSON 备份文件到本页面来导入备份。\n\n若要导出 Stylus V1.5.18 之前的兼容备份请右击或Shift+左击「导出」按钮。" "message": "选择备份文件,或将 JSON 备份文件拖放到本页面,即可导入备份。"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "导出所有样式" "message": "导出所有样式"
@ -259,33 +259,19 @@
"message": "删除" "message": "删除"
}, },
"description": { "description": {
"message": "Stylus 是一个调整网页外观的用户样式管理器。它可让您轻松为许多热门网站安装主题和皮肤。" "message": "Stylus 是一个调整网页外观的用户样式管理器。它可让您轻松为许多热门网站网站安装主题和皮肤。"
}, },
"disableAllStyles": { "disableAllStyles": {
"message": "禁用所有样式" "message": "禁用所有样式"
}, },
"disableAllStylesOff": {
"message": "已禁用样式"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "禁用" "message": "禁用"
}, },
"draftAction": {
"message": "选择“是”以加载此草稿,或“否”以丢弃它。"
},
"draftTitle": {
"message": "恢复 $date$ 个小时前的草稿",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "把 JSON 备份文件拖放到该页面任意位置即可导入" "message": "把 JSON 备份文件拖放到该页面任意位置即可导入"
}, },
"dragDropUsercssTabstrip": { "dragDropUsercssTabstrip": {
"message": "要安装文件,就将其放在标签页条(显示标签页标题的区域)上。" "message": "要安装文件,就将其放在选项卡条(显示选项卡标题的区域)上。"
}, },
"editDeleteText": { "editDeleteText": {
"message": "删除" "message": "删除"
@ -307,9 +293,6 @@
} }
} }
}, },
"editorSettings": {
"message": "编辑器设置"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "启用" "message": "启用"
}, },
@ -319,9 +302,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "排除当前链接" "message": "排除当前链接"
}, },
"exportCompatible": {
"message": "导出(兼容模式)"
},
"exportLabel": { "exportLabel": {
"message": "导出" "message": "导出"
}, },
@ -344,7 +324,7 @@
"message": "UserCSS 文档" "message": "UserCSS 文档"
}, },
"filteredStyles": { "filteredStyles": {
"message": "已显示 $numShown$ 个,共 $numTotal$ 个", "message": "共 $numTotal$ 个,已显示 $numShown$ 个",
"placeholders": { "placeholders": {
"numShown": { "numShown": {
"content": "$1" "content": "$1"
@ -360,6 +340,15 @@
"findStyles": { "findStyles": {
"message": "查找更多样式" "message": "查找更多样式"
}, },
"findStylesForSite": {
"message": "查找适合此网站的更多样式"
},
"findStylesInline": {
"message": "嵌入到此页面"
},
"findStylesInlineTooltip": {
"message": "在此弹窗内显示搜索结果。"
},
"genericAdd": { "genericAdd": {
"message": "添加" "message": "添加"
}, },
@ -393,12 +382,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "已保存" "message": "已保存"
}, },
"genericSize": {
"message": "大小"
},
"genericTest": {
"message": "测试"
},
"genericTitle": { "genericTitle": {
"message": "标题" "message": "标题"
}, },
@ -408,9 +391,6 @@
"gettingStyles": { "gettingStyles": {
"message": "正在获取所有样式..." "message": "正在获取所有样式..."
}, },
"headerResizerHint": {
"message": "仅在此类 编辑器/管理器/安装器 UI , 按住 Shift 可调整大小"
},
"helpAlt": { "helpAlt": {
"message": "帮助" "message": "帮助"
}, },
@ -510,13 +490,13 @@
"message": "获取|分享 样式" "message": "获取|分享 样式"
}, },
"linkGetShareStylesInfo": { "linkGetShareStylesInfo": {
"message": "由社区驱动的新站点 userstyles.world 是用户样式的作者们创建的,目的是取代 userstyles.org。userstyles.org 在过去一年中运行缓慢且反应迟钝,以至于许多作者都不再更新他们的样式。" "message": "由社区驱动的新站点 userstyles.world 是由 userstyle 作者创建的,目的是取代 userstyles.org该站点在过去一年中缓慢且反应迟钝以至于许多作者停止更新他们的样式。"
}, },
"linkGetStyles": { "linkGetStyles": {
"message": "获取样式" "message": "获取样式"
}, },
"linkGetStylesInfo": { "linkGetStylesInfo": {
"message": "该存档站点是一位用户样式社区的成员创建的,约每天更新一次其内容,用于备份运行缓慢且反应迟钝的 userstyles.org。" "message": "该存档站点由 userstyle 社区成员创建,约每天更新一次其内容,用于备份缓慢且无响应的 userstyles.org。 "
}, },
"linkTranslate": { "linkTranslate": {
"message": "翻译" "message": "翻译"
@ -570,22 +550,22 @@
"message": "查看文件时发生错误" "message": "查看文件时发生错误"
}, },
"liveReloadInstallHint": { "liveReloadInstallHint": {
"message": "保持此标签页为打开状态,以便在外部更改时自动更新样式。" "message": "保持此选项卡为打开状态,将在外部更改后自动更新样式。"
}, },
"liveReloadInstallHintFF": { "liveReloadInstallHintFF": {
"message": "保持此标签页和原始标签页处于打开状态,以便在外部更改时自动更新样式。" "message": "保持此选项卡和原始选项卡处于打开状态,将在外部更改时自动重新更新样式。"
}, },
"liveReloadLabel": { "liveReloadLabel": {
"message": "动态刷新" "message": "动态刷新"
}, },
"manageFavicons": { "manageFavicons": {
"message": "显示「应用于」列的网站图标favicon" "message": "显示 \"应用于\" 的favicon图标"
}, },
"manageFaviconsGray": { "manageFaviconsGray": {
"message": "显示为灰色图标" "message": "显示为灰色图标"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus 使用外部服务 https://icons.duckduckgo.com 来获取图标" "message": "Stylus 使用外部服务 https://www.google.com/s2/favicons 来获取图标"
}, },
"manageFilters": { "manageFilters": {
"message": "过滤器" "message": "过滤器"
@ -845,15 +825,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "高级设置" "message": "高级设置"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "按系统偏好设置"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "按夜间时间"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "停用。忽略样式的深色/浅色模式设置。"
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "向编辑器右键菜单添加“删除”" "message": "向编辑器右键菜单添加“删除”"
}, },
@ -863,12 +834,6 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "给每个iframe框架的html注入window.top父域名的内联属性(值)\n即 html[stylus-iframe=\"hostname\"] \n\n用途: 确定或排除 iframe选择器以避免伤及无辜.\nhtml:not([stylus-iframe]) {...}\nhtml[stylus-iframe] 或 html[stylus-iframe$=\"twitter.com\"] h1 {...}\n\ngithub.com/openstyles/stylus/blob/master/content/apply.js#L270" "message": "给每个iframe框架的html注入window.top父域名的内联属性(值)\n即 html[stylus-iframe=\"hostname\"] \n\n用途: 确定或排除 iframe选择器以避免伤及无辜.\nhtml:not([stylus-iframe]) {...}\nhtml[stylus-iframe] 或 html[stylus-iframe$=\"twitter.com\"] h1 {...}\n\ngithub.com/openstyles/stylus/blob/master/content/apply.js#L270"
}, },
"optionsAdvancedExposeStyleName": {
"message": "暴露样式名称"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "在页面中暴露样式名称以方便在 DevTools 中调试样式。请重新加载标签页以应用新设置。"
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "将新样式的格式设为 UserCSS" "message": "将新样式的格式设为 UserCSS"
}, },
@ -902,6 +867,9 @@
"optionsCustomizeIcon": { "optionsCustomizeIcon": {
"message": "工具栏图标风格" "message": "工具栏图标风格"
}, },
"optionsCustomizePopup": {
"message": "Popup "
},
"optionsCustomizeSync": { "optionsCustomizeSync": {
"message": "同步到云端" "message": "同步到云端"
}, },
@ -911,14 +879,11 @@
"optionsHeading": { "optionsHeading": {
"message": "选项" "message": "选项"
}, },
"optionsIconAuto": {
"message": "跟随深色/浅色模式"
},
"optionsIconDark": { "optionsIconDark": {
"message": "深色浏览器主题" "message": "高对比度暗色图标"
}, },
"optionsIconLight": { "optionsIconLight": {
"message": "浅色浏览器主题" "message": "低对比度亮色图标"
}, },
"optionsOpen": { "optionsOpen": {
"message": "打开" "message": "打开"
@ -936,7 +901,7 @@
"message": "恢复默认值" "message": "恢复默认值"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "在任意 Stylus 页面(包括这个)点击浏览器工具栏的 Stylus 图标,然后点击「寻找样式」" "message": "查找 Stylus UI 主题ᐝ"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "附加选项" "message": "附加选项"
@ -953,9 +918,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "无" "message": "无"
}, },
"optionsSyncPassword": {
"message": "密码"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "已连接" "message": "已连接"
}, },
@ -969,7 +931,7 @@
"message": "正在断开连接..." "message": "正在断开连接..."
}, },
"optionsSyncStatusPull": { "optionsSyncStatusPull": {
"message": "正在拉取样式 $loaded$ / $total$", "message": "正在获取样式 $loaded$ 中的 $total$",
"placeholders": { "placeholders": {
"loaded": { "loaded": {
"content": "$1" "content": "$1"
@ -980,7 +942,7 @@
} }
}, },
"optionsSyncStatusPush": { "optionsSyncStatusPush": {
"message": "正在推送样式 $loaded$ / $total$", "message": "正在上传样式 $loaded$ 中的 $total$",
"placeholders": { "placeholders": {
"loaded": { "loaded": {
"content": "$1" "content": "$1"
@ -999,9 +961,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "现在同步" "message": "现在同步"
}, },
"optionsSyncUsername": {
"message": "用户名"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "从旧版本的 Stylus 或 Stylish 中导入样式备份时,请在样式管理器中手动检测升级一次,以确保所有样式都可更新到最新版本。" "message": "从旧版本的 Stylus 或 Stylish 中导入样式备份时,请在样式管理器中手动检测升级一次,以确保所有样式都可更新到最新版本。"
}, },
@ -1039,14 +998,11 @@
"message": "对于新 Chrome 中的暗色主题很有用,因为它不再绘制边框" "message": "对于新 Chrome 中的暗色主题很有用,因为它不再绘制边框"
}, },
"popupHotkeysInfo": { "popupHotkeysInfo": {
"message": "<1>-<9>, <0> 开启/关闭第 N 个样式0 就是 10\n<A>-<Z> 开启/关闭相应英文首字母的首个样式\n附加 <Shift> 打开编辑器而不是开启/关闭\n<Numpad +> 启用列出的样式\n<Numpad > 禁用列出的样式\n<Numpad *> 或 <`>(反引号)- 开启/关闭已启用的首个样式,当弹出菜单开启时不应用后来启用的样式,因此你可以在测试完成后恢复初始状态:简单地禁用所有样式,然后切换如 <Numpad > <Numpad *>\n更多信息见 Wiki" "message": "<1>-<9>, <0> 按第1 - 10个列表 来ON/OFF\n<A>-<Z> 按英文首字母 来ON/OFF\n附加<Shift>: 打开对应列表的编辑器\n<Numpad +> ON 全部列表\n<Numpad > OFF 全部列表\n<Numpad *> 或 <`> 仅针对Popup弹出时的初始ON列表, 它只会ON/OFF初始ON列表, 故:\n <Numpad > <Numpad *> 能快速恢复初始状态.\n 更多信息见Wiki"
}, },
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "查看 Popup 快捷键" "message": "查看 Popup 快捷键"
}, },
"popupManageSiteStyles": {
"message": "管理网站样式"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "右键单击或 Shift + 左键单击可打开管理器并显示适用于当前网站的样式" "message": "右键单击或 Shift + 左键单击可打开管理器并显示适用于当前网站的样式"
}, },
@ -1068,21 +1024,6 @@
"prefShowBadge": { "prefShowBadge": {
"message": "在当前网站生效的样式数量" "message": "在当前网站生效的样式数量"
}, },
"preferScheme": {
"message": "深色/浅色模式偏好设置"
},
"preferSchemeAlways": {
"message": "目前已忽略(总是应用的样式),因为已停用全局深色/浅色模式。"
},
"preferSchemeDark": {
"message": "深色"
},
"preferSchemeLight": {
"message": "浅色"
},
"preferSchemeNone": {
"message": "无(总是应用)"
},
"previewLabel": { "previewLabel": {
"message": "实时预览" "message": "实时预览"
}, },
@ -1093,7 +1034,7 @@
"message": "发布" "message": "发布"
}, },
"publishPush": { "publishPush": {
"message": "推送更新" "message": "发布更新"
}, },
"publishReconnect": { "publishReconnect": {
"message": "尝试断开连接然后再次发布" "message": "尝试断开连接然后再次发布"
@ -1111,7 +1052,7 @@
"message": "正在读取样式..." "message": "正在读取样式..."
}, },
"reload": { "reload": {
"message": "重启" "message": "重启 Stylus"
}, },
"replace": { "replace": {
"message": "替换" "message": "替换"
@ -1122,18 +1063,12 @@
"replaceWith": { "replaceWith": {
"message": "替换为" "message": "替换为"
}, },
"restoreTemplate": {
"message": "恢复默认模板。\n\n不会改变当前打开的编辑器页面"
},
"retrieveBckp": { "retrieveBckp": {
"message": "导入所有样式" "message": "导入所有样式"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "从 Dropbox 导入" "message": "从 Dropbox 导入"
}, },
"saveAsTemplate": {
"message": "另存为模板"
},
"search": { "search": {
"message": "搜索" "message": "搜索"
}, },
@ -1153,7 +1088,7 @@
"message": "/regex/ 用 / 包裹语法来正则搜索" "message": "/regex/ 用 / 包裹语法来正则搜索"
}, },
"searchResultInstallCount": { "searchResultInstallCount": {
"message": "总安装次数" "message": "总安装次数"
}, },
"searchResultNoneFound": { "searchResultNoneFound": {
"message": "没有找到与此页面相关的样式。" "message": "没有找到与此页面相关的样式。"
@ -1173,6 +1108,9 @@
"searchResultWeeklyCount": { "searchResultWeeklyCount": {
"message": "本周安装次数" "message": "本周安装次数"
}, },
"searchStyleQueryHint": {
"message": "空格多词 -- 同时精确匹配(无需双引号)\n2020 -- 也含更新日期的匹配结果\n大小写不敏感"
},
"searchStylesAll": { "searchStylesAll": {
"message": "全部" "message": "全部"
}, },
@ -1206,18 +1144,12 @@
"sections": { "sections": {
"message": "章节" "message": "章节"
}, },
"settings": {
"message": "设置"
},
"shortcuts": { "shortcuts": {
"message": "快捷键" "message": "快捷键"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "设置快捷键" "message": "设置快捷键"
}, },
"shortcutsNoteFF": {
"message": "在 Firefox 66 及之后更新的版本中:\n1) 右键单击工具栏的的 Stylus 图标,选择「管理」\n也可以通过主菜单或 Ctrl-Shift-A 打开 about:addons\n2) 在打开的页面点击右上角的齿轮图标;\n3) 选择「管理扩展快捷键」。\n\n您也可以在这自定义快捷键。"
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "最新的优先" "message": "最新的优先"
}, },
@ -1263,30 +1195,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "启用" "message": "启用"
}, },
"styleExcludeLabel": {
"message": "自定义排除网站"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "导入 Mozilla 格式失败" "message": "导入 Mozilla 格式失败"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "粘贴 Mozilla 格式代码" "message": "粘贴 Mozilla 格式代码"
}, },
"styleIncludeLabel": {
"message": "自定义包括网站"
},
"styleInjectionImportance": {
"message": "切换样式的优先级"
},
"styleInjectionOrder": {
"message": "样式注入顺序"
},
"styleInjectionOrderHint": {
"message": "拖拽样式可更改排序。样式会按如下顺序注入,所以在列表更下方样式可覆盖较前面的样式。"
},
"styleInjectionOrderHint_prio": {
"message": "下面列出的重要样式始终是最后注入的,因此它们可以覆盖任何新安装的样式。单击样式的标记以切换其重要性。"
},
"styleInstall": { "styleInstall": {
"message": "要将“$stylename$”安装到 Stylus 中吗?", "message": "要将“$stylename$”安装到 Stylus 中吗?",
"placeholders": { "placeholders": {
@ -1329,15 +1243,6 @@
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "正则表达式错误,样式无法正常运行" "message": "正则表达式错误,样式无法正常运行"
}, },
"styleNotAppliedSchemeDark": {
"message": "此样式仅在深色模式下应用"
},
"styleNotAppliedSchemeLight": {
"message": "此样式仅在浅色模式下应用"
},
"stylePreferSchemeLabel": {
"message": "深色/浅色模式"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "部分正则表达式无法生效。" "message": "部分正则表达式无法生效。"
}, },
@ -1347,6 +1252,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "多个部分代码无法正常处理正则表达式" "message": "多个部分代码无法正常处理正则表达式"
}, },
"styleRegexpTestButton": {
"message": "正则测试"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "匹配到标签页的正则" "message": "匹配到标签页的正则"
}, },
@ -1368,9 +1276,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "保存" "message": "保存"
}, },
"styleSettings": {
"message": "样式设置"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "导入 FireFox 格式即 @-moz-document ...的 CSS 代码后,会自动转换成 Stylus 使用的分段式 CSS 代码。\n反过来分段式的 CSS 也可以导出为 FireFox 格式。\n需要注意的是如果你要向 userstyles.org 提交你的代码,则必须使用 FireFox 格式。\n\n导入快捷键: 剪贴板含有 @-moz-document ....代码,则可以直接在编辑器里 Ctrl+V 会自动弹出导入对话框" "message": "导入 FireFox 格式即 @-moz-document ...的 CSS 代码后,会自动转换成 Stylus 使用的分段式 CSS 代码。\n反过来分段式的 CSS 也可以导出为 FireFox 格式。\n需要注意的是如果你要向 userstyles.org 提交你的代码,则必须使用 FireFox 格式。\n\n导入快捷键: 剪贴板含有 @-moz-document ....代码,则可以直接在编辑器里 Ctrl+V 会自动弹出导入对话框"
}, },
@ -1388,11 +1293,8 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "样式已经在编辑器外被修改。需要重新加载样式吗?" "message": "样式已经在编辑器外被修改。需要重新加载样式吗?"
}, },
"styleUpdateUrlLabel": {
"message": "更新 URL"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "Stylus 不能在此类页面上工作" "message": "Stylus 无法介入到此类页面"
}, },
"stylusUnavailableForURLdetails": { "stylusUnavailableForURLdetails": {
"message": "出于安全考虑,浏览器禁止扩展程序影响其内置页面(例如 chrome://versionChrome 61 后的标准新标签页about:addons 等等)以及其他扩展程序的页面。每个浏览器也限制了对于自己扩展程序库的介入〔例如 Chrome 网上应用店、Firefox 附加组件addons.mozilla.org。" "message": "出于安全考虑,浏览器禁止扩展程序影响其内置页面(例如 chrome://versionChrome 61 后的标准新标签页about:addons 等等)以及其他扩展程序的页面。每个浏览器也限制了对于自己扩展程序库的介入〔例如 Chrome 网上应用店、Firefox 附加组件addons.mozilla.org。"
@ -1406,16 +1308,8 @@
"syncError": { "syncError": {
"message": "同步失败" "message": "同步失败"
}, },
"syncErrorLock": {
"message": "数据库已在使用中。锁定将在 $TIME$ 过期",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "同步失败。你已退出登录。\n请尝试在 Stylus 选项里重新登录。" "message": "同步失败\n请尝试在 Stylus 选项里重新登录\n先点击「断开连接」再点击「连接」。"
}, },
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "不能保存该值,请尝试减少文本数量。" "message": "不能保存该值,请尝试减少文本数量。"
@ -1506,6 +1400,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "使用当前 UserStyle 代码替换为新的默认模板吗 ?" "message": "使用当前 UserStyle 代码替换为新的默认模板吗 ?"
}, },
"usercssReplaceTemplateName": {
"message": "@name 为空值 可设置新的默认模板"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "在此插入代码..." "message": "在此插入代码..."
}, },

View File

@ -71,7 +71,7 @@
"message": "備份" "message": "備份"
}, },
"backupMessage": { "backupMessage": {
"message": "要匯入備份檔案,請將其拖曳至此頁面或點擊匯入按鈕。\n\n要匯出相容於比 1.5.18 還舊的 Stylus 版本的備份,請右鍵點擊或按住 Shift 然後點擊匯出按鈕。" "message": "選取檔案並拖曳到此頁面。"
}, },
"bckpInstStyles": { "bckpInstStyles": {
"message": "匯出樣式" "message": "匯出樣式"
@ -264,23 +264,9 @@
"disableAllStyles": { "disableAllStyles": {
"message": "停用所有樣式" "message": "停用所有樣式"
}, },
"disableAllStylesOff": {
"message": "樣式已關閉"
},
"disableStyleLabel": { "disableStyleLabel": {
"message": "停用" "message": "停用"
}, },
"draftAction": {
"message": "選擇「是」以載入此草稿,或「否」以放棄"
},
"draftTitle": {
"message": "還原草稿,已建立 $date$",
"placeholders": {
"date": {
"content": "$1"
}
}
},
"dragDropMessage": { "dragDropMessage": {
"message": "將您的備份檔拖曳到此頁面的任何地方以匯入。" "message": "將您的備份檔拖曳到此頁面的任何地方以匯入。"
}, },
@ -307,9 +293,6 @@
} }
} }
}, },
"editorSettings": {
"message": "編輯器設定"
},
"enableStyleLabel": { "enableStyleLabel": {
"message": "啟用" "message": "啟用"
}, },
@ -319,9 +302,6 @@
"excludeStyleByUrlLabel": { "excludeStyleByUrlLabel": {
"message": "排除目前的 URL" "message": "排除目前的 URL"
}, },
"exportCompatible": {
"message": "匯出(相容模式)"
},
"exportLabel": { "exportLabel": {
"message": "匯出" "message": "匯出"
}, },
@ -360,6 +340,15 @@
"findStyles": { "findStyles": {
"message": "尋找樣式" "message": "尋找樣式"
}, },
"findStylesForSite": {
"message": "查找更多適合此網站的樣式"
},
"findStylesInline": {
"message": "嵌入"
},
"findStylesInlineTooltip": {
"message": "在此視窗中顯示搜尋結果。"
},
"genericAdd": { "genericAdd": {
"message": "新增" "message": "新增"
}, },
@ -393,12 +382,6 @@
"genericSavedMessage": { "genericSavedMessage": {
"message": "已儲存" "message": "已儲存"
}, },
"genericSize": {
"message": "大小"
},
"genericTest": {
"message": "測試"
},
"genericTitle": { "genericTitle": {
"message": "標題" "message": "標題"
}, },
@ -408,9 +391,6 @@
"gettingStyles": { "gettingStyles": {
"message": "正在取得所有樣式……" "message": "正在取得所有樣式……"
}, },
"headerResizerHint": {
"message": "按住 Shift 以僅在此類型的 UI 中調整大小,例如編輯器、管理程式、安裝程式等"
},
"helpAlt": { "helpAlt": {
"message": "說明" "message": "說明"
}, },
@ -585,7 +565,7 @@
"message": "灰階淡出" "message": "灰階淡出"
}, },
"manageFaviconsHelp": { "manageFaviconsHelp": {
"message": "Stylus 使用外部服務 https://icons.duckduckgo.com" "message": "Stylus 使用外部服務 https://www.google.com/s2/favicons"
}, },
"manageFilters": { "manageFilters": {
"message": "過濾器" "message": "過濾器"
@ -596,9 +576,6 @@
"manageMaxTargets": { "manageMaxTargets": {
"message": "已套用項目的數量" "message": "已套用項目的數量"
}, },
"manageMinColumnWidth": {
"message": "最小欄位寬度以像素為單位9999 停用多欄模式)"
},
"manageNewStyleAsUsercss": { "manageNewStyleAsUsercss": {
"message": "做為 Usercss" "message": "做為 Usercss"
}, },
@ -848,15 +825,6 @@
"optionsAdvanced": { "optionsAdvanced": {
"message": "進階" "message": "進階"
}, },
"optionsAdvancedAutoSwitchSchemeBySystem": {
"message": "按系統偏好設定"
},
"optionsAdvancedAutoSwitchSchemeByTime": {
"message": "到夜間:"
},
"optionsAdvancedAutoSwitchSchemeNever": {
"message": "已停用。樣式中的深色/淺色設定會被忽略。"
},
"optionsAdvancedContextDelete": { "optionsAdvancedContextDelete": {
"message": "在編輯器的右鍵選單中加入「刪除」" "message": "在編輯器的右鍵選單中加入「刪除」"
}, },
@ -866,12 +834,6 @@
"optionsAdvancedExposeIframesNote": { "optionsAdvancedExposeIframesNote": {
"message": "公開每個 iframe 中的頂級頁面網域。\n啟用編寫特別用於 iframe 的 CSS如這個\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }" "message": "公開每個 iframe 中的頂級頁面網域。\n啟用編寫特別用於 iframe 的 CSS如這個\nhtml[stylus-iframe$$=\"twitter.com\"] h1 { display:none }"
}, },
"optionsAdvancedExposeStyleName": {
"message": "發佈樣式名稱"
},
"optionsAdvancedExposeStyleNameNote": {
"message": "在頁面中發佈樣式名稱以方便在 devtools 中對樣式進行除錯。請重新載入分頁以套用新設定。"
},
"optionsAdvancedNewStyleAsUsercss": { "optionsAdvancedNewStyleAsUsercss": {
"message": "以 usercss 編寫新樣式" "message": "以 usercss 編寫新樣式"
}, },
@ -917,9 +879,6 @@
"optionsHeading": { "optionsHeading": {
"message": "選項" "message": "選項"
}, },
"optionsIconAuto": {
"message": "符合深色/淺色模式"
},
"optionsIconDark": { "optionsIconDark": {
"message": "暗色瀏覽器主題" "message": "暗色瀏覽器主題"
}, },
@ -942,7 +901,7 @@
"message": "重設選項" "message": "重設選項"
}, },
"optionsStylusThemes": { "optionsStylusThemes": {
"message": "點擊任何 Stylus 頁面(包含這個)的瀏覽器工具列中的 Stylus 圖示,然後點擊「尋找樣式」" "message": "尋找 Sytlus UI 佈景主題"
}, },
"optionsSubheading": { "optionsSubheading": {
"message": "更多選項" "message": "更多選項"
@ -959,9 +918,6 @@
"optionsSyncNone": { "optionsSyncNone": {
"message": "無" "message": "無"
}, },
"optionsSyncPassword": {
"message": "密碼"
},
"optionsSyncStatusConnected": { "optionsSyncStatusConnected": {
"message": "已連線" "message": "已連線"
}, },
@ -1005,9 +961,6 @@
"optionsSyncSyncNow": { "optionsSyncSyncNow": {
"message": "立刻同步" "message": "立刻同步"
}, },
"optionsSyncUsername": {
"message": "使用者名稱"
},
"optionsUpdateImportNote": { "optionsUpdateImportNote": {
"message": "當從舊版本或是從 Stylish 匯入樣式備份時,在樣式管理員中手動進行一次更新檢查,以確保所有樣式都被更新。" "message": "當從舊版本或是從 Stylish 匯入樣式備份時,在樣式管理員中手動進行一次更新檢查,以確保所有樣式都被更新。"
}, },
@ -1050,9 +1003,6 @@
"popupHotkeysTooltip": { "popupHotkeysTooltip": {
"message": "點選以檢視可用的快速鍵" "message": "點選以檢視可用的快速鍵"
}, },
"popupManageSiteStyles": {
"message": "管理網站樣式"
},
"popupManageTooltip": { "popupManageTooltip": {
"message": "Shift + 點選或右鍵 + 點選以在管理員中開啟目前頁面可用的樣式" "message": "Shift + 點選或右鍵 + 點選以在管理員中開啟目前頁面可用的樣式"
}, },
@ -1074,21 +1024,6 @@
"prefShowBadge": { "prefShowBadge": {
"message": "在工具欄按鈕上顯示當前網站已生效的樣式表數目。" "message": "在工具欄按鈕上顯示當前網站已生效的樣式表數目。"
}, },
"preferScheme": {
"message": "深色/淺色模式偏好設定"
},
"preferSchemeAlways": {
"message": "目前被忽略(樣式一律套用),因為全域深色/淺色模式已被停用"
},
"preferSchemeDark": {
"message": "深色"
},
"preferSchemeLight": {
"message": "淺色"
},
"preferSchemeNone": {
"message": "無(一律套用)"
},
"previewLabel": { "previewLabel": {
"message": "即時預覽" "message": "即時預覽"
}, },
@ -1117,7 +1052,7 @@
"message": "正在讀取樣式……" "message": "正在讀取樣式……"
}, },
"reload": { "reload": {
"message": "重" "message": "重新載入 Stylus 附加元件"
}, },
"replace": { "replace": {
"message": "取代" "message": "取代"
@ -1128,18 +1063,12 @@
"replaceWith": { "replaceWith": {
"message": "取代為" "message": "取代為"
}, },
"restoreTemplate": {
"message": "還原預設範本。\n\n不會變更目前開啟的編輯器頁面。"
},
"retrieveBckp": { "retrieveBckp": {
"message": "匯入樣式" "message": "匯入樣式"
}, },
"retrieveDropboxSync": { "retrieveDropboxSync": {
"message": "Dropbox 匯入" "message": "Dropbox 匯入"
}, },
"saveAsTemplate": {
"message": "另存為範本"
},
"search": { "search": {
"message": "搜尋" "message": "搜尋"
}, },
@ -1180,7 +1109,7 @@
"message": "每週安裝" "message": "每週安裝"
}, },
"searchStyleQueryHint": { "searchStyleQueryHint": {
"message": "搜尋樣式名稱(若使用大寫字母,則區分大小寫):\n一些字 - 以任何順序符合所有字\n\"一些詞\" - 精確符合引號內的詞\n/foo.*bar/i - 沒有空格的正規表示式(使用 \\s 代替)" "message": "搜尋樣式名稱時是區分大小寫的:\nsome words - 以任意順序搜尋所有文字\n\"some phrase\" - 精確符合引號內的詞語\n2020 - 如此年份顯示在2020年更新的樣式"
}, },
"searchStylesAll": { "searchStylesAll": {
"message": "全部" "message": "全部"
@ -1215,18 +1144,12 @@
"sections": { "sections": {
"message": "樣式段" "message": "樣式段"
}, },
"settings": {
"message": "設定"
},
"shortcuts": { "shortcuts": {
"message": "快速鍵" "message": "快速鍵"
}, },
"shortcutsNote": { "shortcutsNote": {
"message": "自訂鍵盤快速鍵" "message": "自訂鍵盤快速鍵"
}, },
"shortcutsNoteFF": {
"message": "在 Firefox 66 或更新的版本中,您可以手動開啟內建的快捷鍵使用者介面:\n1) 右鍵點擊工具列中的 Stylus 圖示並選擇「管理」\n或是透過主選單或 Ctrl-Shift-A 開啟 about:addons\n2) 在開啟的頁面中點擊右上角的齒輪圖示,\n3) 選擇「管理擴充套件快捷鍵」。\n\n您也可以在此自訂快捷鍵。"
},
"sortDateNewestFirst": { "sortDateNewestFirst": {
"message": "最新的優先" "message": "最新的優先"
}, },
@ -1272,30 +1195,12 @@
"styleEnabledLabel": { "styleEnabledLabel": {
"message": "已啟用" "message": "已啟用"
}, },
"styleExcludeLabel": {
"message": "自訂排除網站"
},
"styleFromMozillaFormatError": { "styleFromMozillaFormatError": {
"message": "從 Mozilla 格式匯入失敗" "message": "從 Mozilla 格式匯入失敗"
}, },
"styleFromMozillaFormatPrompt": { "styleFromMozillaFormatPrompt": {
"message": "貼上 Mozilla 格式代碼" "message": "貼上 Mozilla 格式代碼"
}, },
"styleIncludeLabel": {
"message": "自訂包含網站"
},
"styleInjectionImportance": {
"message": "切換樣式重要程度"
},
"styleInjectionOrder": {
"message": "樣式注入順序"
},
"styleInjectionOrderHint": {
"message": "拖曳樣式可變更其位置。樣式按下列順序注入,所以清單下方的樣式可覆寫較早樣式。"
},
"styleInjectionOrderHint_prio": {
"message": "下方列出的重要樣式一律最後注入,因此它們會覆寫新安裝的樣式。點擊樣式的標記以切換其重要程度。"
},
"styleInstall": { "styleInstall": {
"message": "安裝 '$stylename$' 到 Stylus ", "message": "安裝 '$stylename$' 到 Stylus ",
"placeholders": { "placeholders": {
@ -1338,15 +1243,6 @@
"styleNotAppliedRegexpProblemTooltip": { "styleNotAppliedRegexpProblemTooltip": {
"message": "因為不正確的 'regexp()' 使用導致未套用" "message": "因為不正確的 'regexp()' 使用導致未套用"
}, },
"styleNotAppliedSchemeDark": {
"message": "此樣式僅於深色模式中套用"
},
"styleNotAppliedSchemeLight": {
"message": "此樣式僅於淺色模式中套用"
},
"stylePreferSchemeLabel": {
"message": "深色/淺色模式"
},
"styleRegexpInvalidExplanation": { "styleRegexpInvalidExplanation": {
"message": "部份 'regexp()' 規則可能不再能被編譯。" "message": "部份 'regexp()' 規則可能不再能被編譯。"
}, },
@ -1356,6 +1252,9 @@
"styleRegexpProblemTooltip": { "styleRegexpProblemTooltip": {
"message": "因為不正確的 'regexp()' 使用導致未套用的樣式段數量" "message": "因為不正確的 'regexp()' 使用導致未套用的樣式段數量"
}, },
"styleRegexpTestButton": {
"message": "正規表示式測試"
},
"styleRegexpTestFull": { "styleRegexpTestFull": {
"message": "符合的分頁" "message": "符合的分頁"
}, },
@ -1377,9 +1276,6 @@
"styleSaveLabel": { "styleSaveLabel": {
"message": "儲存" "message": "儲存"
}, },
"styleSettings": {
"message": "樣式設定"
},
"styleToMozillaFormatHelp": { "styleToMozillaFormatHelp": {
"message": "Mozilla格式的樣式代碼能在火狐版Stylus使用也可以提交至 userstyles.org 。" "message": "Mozilla格式的樣式代碼能在火狐版Stylus使用也可以提交至 userstyles.org 。"
}, },
@ -1397,9 +1293,6 @@
"styleUpdateDiscardChanges": { "styleUpdateDiscardChanges": {
"message": "樣式已在編輯器外變更。您想要重新載入樣式嗎?" "message": "樣式已在編輯器外變更。您想要重新載入樣式嗎?"
}, },
"styleUpdateUrlLabel": {
"message": "更新 URL"
},
"stylusUnavailableForURL": { "stylusUnavailableForURL": {
"message": "Stylus 不能在諸如此類的網頁上生效。" "message": "Stylus 不能在諸如此類的網頁上生效。"
}, },
@ -1415,16 +1308,8 @@
"syncError": { "syncError": {
"message": "同步失敗" "message": "同步失敗"
}, },
"syncErrorLock": {
"message": "資料庫已在使用中。鎖將於 $TIME$ 過期",
"placeholders": {
"time": {
"content": "$1"
}
}
},
"syncErrorRelogin": { "syncErrorRelogin": {
"message": "同步失敗。您已被登出。\n嘗試在 Stylus 選項中重新登入。" "message": "同步失敗。\n重是在 Stylus 選項重新登入:\n先點擊「斷線」然後「連線」。"
}, },
"syncStorageErrorSaving": { "syncStorageErrorSaving": {
"message": "無法儲存值。嘗試減少文字量。" "message": "無法儲存值。嘗試減少文字量。"
@ -1515,6 +1400,9 @@
"usercssReplaceTemplateConfirmation": { "usercssReplaceTemplateConfirmation": {
"message": "為新的 Usercss 樣式取代預設的範本為目前的程式碼?" "message": "為新的 Usercss 樣式取代預設的範本為目前的程式碼?"
}, },
"usercssReplaceTemplateName": {
"message": "清空 @name 取代目前範本"
},
"usercssReplaceTemplateSectionBody": { "usercssReplaceTemplateSectionBody": {
"message": "在此插入程式碼……" "message": "在此插入程式碼……"
}, },

View File

@ -6,9 +6,15 @@
/* global syncMan */ /* global syncMan */
/* global updateMan */ /* global updateMan */
/* global usercssMan */ /* global usercssMan */
/* global usoApi */
/* global uswApi */ /* global uswApi */
/* global FIREFOX UA activateTab openURL */ // toolbox.js /* global
FIREFOX
URLS
activateTab
download
findExistingTab
openURL
*/ // toolbox.js
/* global colorScheme */ // color-scheme.js /* global colorScheme */ // color-scheme.js
'use strict'; 'use strict';
@ -35,12 +41,17 @@ addAPI(/** @namespace API */ {
sync: syncMan, sync: syncMan,
updater: updateMan, updater: updateMan,
usercss: usercssMan, usercss: usercssMan,
uso: usoApi,
usw: uswApi, usw: uswApi,
colorScheme, colorScheme,
/** @type {BackgroundWorker} */ /** @type {BackgroundWorker} */
worker: createWorker({url: '/background/background-worker'}), worker: createWorker({url: '/background/background-worker'}),
download(url, opts) {
return typeof url === 'string' && url.startsWith(URLS.uso) &&
this.sender.url.startsWith(URLS.uso) &&
download(url, opts || {});
},
/** @returns {string} */ /** @returns {string} */
getTabUrlPrefix() { getTabUrlPrefix() {
return this.sender.tab.url.match(/^([\w-]+:\/+[^/#]+)/)[1]; return this.sender.tab.url.match(/^([\w-]+:\/+[^/#]+)/)[1];
@ -58,24 +69,10 @@ addAPI(/** @namespace API */ {
async openEditor(params) { async openEditor(params) {
const u = new URL(chrome.runtime.getURL('edit.html')); const u = new URL(chrome.runtime.getURL('edit.html'));
u.search = new URLSearchParams(params); u.search = new URLSearchParams(params);
const wnd = chrome.windows && prefs.get('openEditInWindow'); const wnd = prefs.get('openEditInWindow');
const wndPos = wnd && prefs.get('windowPosition'); const wndPos = wnd && prefs.get('windowPosition');
const wndBase = wnd && prefs.get('openEditInWindow.popup') ? {type: 'popup'} : {}; const wndBase = wnd && prefs.get('openEditInWindow.popup') ? {type: 'popup'} : {};
const ffBug = wnd && FIREFOX; // https://bugzil.la/1271047 const ffBug = wnd && FIREFOX; // https://bugzil.la/1271047
if (wndPos) {
const {left, top, width, height} = wndPos;
const r = left + width;
const b = top + height;
const peek = 32;
if (isNaN(r) || r < peek || left > screen.availWidth - peek || width < 100) {
delete wndPos.left;
delete wndPos.width;
}
if (isNaN(b) || b < peek || top > screen.availHeight - peek || height < 100) {
delete wndPos.top;
delete wndPos.height;
}
}
const tab = await openURL({ const tab = await openURL({
url: `${u}`, url: `${u}`,
currentWindow: null, currentWindow: null,
@ -87,25 +84,27 @@ addAPI(/** @namespace API */ {
/** @returns {Promise<chrome.tabs.Tab>} */ /** @returns {Promise<chrome.tabs.Tab>} */
async openManage({options = false, search, searchMode} = {}) { async openManage({options = false, search, searchMode} = {}) {
const setUrlParams = url => { let url = chrome.runtime.getURL('manage.html');
const u = new URL(url); if (search) {
if (search) u.searchParams.set('search', search); url += `?search=${encodeURIComponent(search)}&searchMode=${searchMode}`;
if (searchMode) u.searchParams.set('searchMode', searchMode);
if (options) u.hash = '#stylus-options';
return u.href;
};
const base = chrome.runtime.getURL('manage.html');
const url = setUrlParams(base);
const tabs = await browser.tabs.query({url: base + '*'});
const same = tabs.find(t => t.url === url);
let tab = same || tabs[0];
if (!tab) {
API.prefsDb.get('badFavs'); // prime the cache to avoid flicker/delay when opening the page
tab = await openURL({url, newTab: true});
} else if (!same) {
msg.sendTab(tab.id, {method: 'pushState', url: setUrlParams(tab.url)});
} }
return activateTab(tab); // activateTab unminimizes the window if (options) {
url += '#stylus-options';
}
const tab = await findExistingTab({
url,
currentWindow: null,
ignoreHash: true,
ignoreSearch: true,
});
if (tab) {
await activateTab(tab);
if (url !== (tab.pendingUrl || tab.url)) {
await msg.sendTab(tab.id, {method: 'pushState', url}).catch(console.error);
}
return tab;
}
return openURL({url, ignoreExisting: true}).then(activateTab); // activateTab unminimizes the window
}, },
/** /**
@ -158,25 +157,10 @@ if (chrome.commands) {
} }
chrome.runtime.onInstalled.addListener(({reason, previousVersion}) => { chrome.runtime.onInstalled.addListener(({reason, previousVersion}) => {
if (reason === 'install') { if (reason === 'update') {
if (UA.mobile) prefs.set('manage.newUI', false); const [a, b, c] = (previousVersion || '').split('.');
if (UA.windows) prefs.set('editor.keyMap', 'sublime'); if (a <= 1 && b <= 5 && c <= 13) { // 1.5.13
} require(['/background/remove-unused-storage']);
// TODO: remove this before 1.5.23 as it's only for a few users who installed git 26b75e77
if (reason === 'update' && previousVersion === '1.5.22') {
for (const dbName of ['drafts', prefs.STORAGE_KEY]) {
try {
indexedDB.open(dbName).onsuccess = async e => {
const idb = /** @type IDBDatabase */ e.target.result;
const ta = idb.objectStoreNames[0] === 'data' && idb.transaction(['data']);
if (ta && ta.objectStore('data').autoIncrement) {
ta.abort();
idb.close();
await new Promise(setTimeout);
indexedDB.deleteDatabase(dbName);
}
};
} catch (e) {}
} }
} }
}); });
@ -208,6 +192,6 @@ Promise.all([
require(['/background/context-menus']), require(['/background/context-menus']),
]).then(() => { ]).then(() => {
bgReady._resolveAll(); bgReady._resolveAll();
msg.ready = true; msg.isBgReady = true;
msg.broadcast({method: 'backgroundReady'}); msg.broadcast({method: 'backgroundReady'});
}); });

100
dist/background/color-scheme.js vendored Normal file
View File

@ -0,0 +1,100 @@
/* global prefs */
/* exported colorScheme */
'use strict';
const colorScheme = (() => {
let systemPreferDark = false;
let timePreferDark = false;
const changeListeners = new Set();
const checkTime = ['schemeSwitcher.nightStart', 'schemeSwitcher.nightEnd'];
prefs.subscribe(checkTime, (key, value) => {
updateTimePreferDark();
createAlarm(key, value);
});
checkTime.forEach(key => createAlarm(key, prefs.get(key)));
prefs.subscribe(['schemeSwitcher.enabled'], emitChange);
chrome.alarms.onAlarm.addListener(info => {
if (checkTime.includes(info.name)) {
updateTimePreferDark();
}
});
updateSystemPreferDark();
updateTimePreferDark();
return {shouldIncludeStyle, onChange, updateSystemPreferDark};
function createAlarm(key, value) {
const date = new Date();
applyDate(date, value);
if (date.getTime() < Date.now()) {
date.setDate(date.getDate() + 1);
}
chrome.alarms.create(key, {
when: date.getTime(),
periodInMinutes: 24 * 60,
});
}
function shouldIncludeStyle(style) {
const isDark = style.preferScheme === 'dark';
const isLight = style.preferScheme === 'light';
if (!isDark && !isLight) {
return true;
}
const switcherState = prefs.get('schemeSwitcher.enabled');
if (switcherState === 'never') {
return true;
}
if (switcherState === 'system') {
return systemPreferDark && isDark ||
!systemPreferDark && isLight;
}
return timePreferDark && isDark ||
!timePreferDark && isLight;
}
function updateSystemPreferDark() {
const oldValue = systemPreferDark;
systemPreferDark = window.matchMedia('(prefers-color-scheme: dark)').matches;
if (systemPreferDark !== oldValue) {
emitChange();
}
return true;
}
function updateTimePreferDark() {
const oldValue = timePreferDark;
const date = new Date();
const now = date.getTime();
applyDate(date, prefs.get('schemeSwitcher.nightStart'));
const start = date.getTime();
applyDate(date, prefs.get('schemeSwitcher.nightEnd'));
const end = date.getTime();
timePreferDark = start > end ?
now >= start || now < end :
now >= start && now < end;
if (timePreferDark !== oldValue) {
emitChange();
}
}
function applyDate(date, time) {
const [h, m] = time.split(':').map(Number);
date.setHours(h, m, 0, 0);
}
function onChange(listener) {
changeListeners.add(listener);
}
function emitChange() {
for (const listener of changeListeners) {
listener();
}
}
})();

31
dist/background/common.js vendored Normal file
View File

@ -0,0 +1,31 @@
/* global API */// msg.js
'use strict';
/**
* Common stuff that's loaded first so it's immediately available to all background scripts
*/
/* exported
addAPI
bgReady
compareRevision
*/
const bgReady = {};
bgReady.styles = new Promise(r => (bgReady._resolveStyles = r));
bgReady.all = new Promise(r => (bgReady._resolveAll = r));
function addAPI(methods) {
for (const [key, val] of Object.entries(methods)) {
const old = API[key];
if (old && Object.prototype.toString.call(old) === '[object Object]') {
Object.assign(old, val);
} else {
API[key] = val;
}
}
}
function compareRevision(rev1, rev2) {
return rev1 - rev2;
}

101
dist/background/context-menus.js vendored Normal file
View File

@ -0,0 +1,101 @@
/* global browserCommands */// background.js
/* global msg */
/* global prefs */
/* global CHROME FIREFOX URLS ignoreChromeError */// toolbox.js
'use strict';
(() => {
const contextMenus = {
'show-badge': {
title: 'menuShowBadge',
click: info => prefs.set(info.menuItemId, info.checked),
},
'disableAll': {
title: 'disableAllStyles',
click: browserCommands.styleDisableAll,
},
'open-manager': {
title: 'openStylesManager',
click: browserCommands.openManage,
},
'open-options': {
title: 'openOptions',
click: browserCommands.openOptions,
},
'reload': {
presentIf: async () => (await browser.management.getSelf()).installType === 'development',
title: 'reload',
click: browserCommands.reload,
},
'editor.contextDelete': {
presentIf: () => !FIREFOX && prefs.get('editor.contextDelete'),
title: 'editDeleteText',
type: 'normal',
contexts: ['editable'],
documentUrlPatterns: [URLS.ownOrigin + 'edit*'],
click: (info, tab) => {
msg.sendTab(tab.id, {method: 'editDeleteText'}, undefined, 'extension')
.catch(msg.ignoreError);
},
},
};
// "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;
}
const keys = Object.keys(contextMenus);
prefs.subscribe(keys.filter(id => typeof prefs.defaults[id] === 'boolean'),
CHROME >= 62 && CHROME <= 64 ? toggleCheckmarkBugged : toggleCheckmark);
prefs.subscribe(keys.filter(id => contextMenus[id].presentIf && prefs.knownKeys.includes(id)),
togglePresence);
createContextMenus(keys);
chrome.contextMenus.onClicked.addListener((info, tab) =>
contextMenus[info.menuItemId].click(info, tab));
async function createContextMenus(ids) {
for (const id of ids) {
let item = contextMenus[id];
if (item.presentIf && !await item.presentIf()) {
continue;
}
item = Object.assign({id}, item);
delete item.presentIf;
item.title = chrome.i18n.getMessage(item.title);
if (!item.type && typeof prefs.defaults[id] === 'boolean') {
item.type = 'checkbox';
item.checked = prefs.get(id);
}
if (!item.contexts) {
item.contexts = ['browser_action'];
}
delete item.click;
chrome.contextMenus.create(item, ignoreChromeError);
}
}
function toggleCheckmark(id, checked) {
chrome.contextMenus.update(id, {checked}, ignoreChromeError);
}
/** Circumvents the bug with disabling check marks in Chrome 62-64 */
async function toggleCheckmarkBugged(id) {
await browser.contextMenus.remove(id).catch(ignoreChromeError);
createContextMenus([id]);
}
function togglePresence(id, checked) {
if (checked) {
createContextMenus([id]);
} else {
chrome.contextMenus.remove(id, ignoreChromeError);
}
}
})();

View File

@ -2,17 +2,17 @@
'use strict'; 'use strict';
/* exported createChromeStorageDB */ /* exported createChromeStorageDB */
function createChromeStorageDB(PREFIX) { function createChromeStorageDB() {
let INC; let INC;
const isMain = !PREFIX;
if (!PREFIX) PREFIX = 'style-';
return { const PREFIX = 'style-';
const METHODS = {
delete(id) { delete(id) {
return chromeLocal.remove(PREFIX + id); return chromeLocal.remove(PREFIX + id);
}, },
// FIXME: we don't use this method at all. Should we remove this?
get(id) { get(id) {
return chromeLocal.getValue(PREFIX + id); return chromeLocal.getValue(PREFIX + id);
}, },
@ -21,9 +21,7 @@ function createChromeStorageDB(PREFIX) {
const all = await chromeLocal.get(); const all = await chromeLocal.get();
if (!INC) prepareInc(all); if (!INC) prepareInc(all);
return Object.entries(all) return Object.entries(all)
.map(([key, val]) => key.startsWith(PREFIX) && .map(([key, val]) => key.startsWith(PREFIX) && Number(key.slice(PREFIX.length)) && val)
(!isMain || Number(key.slice(PREFIX.length))) &&
val)
.filter(Boolean); .filter(Boolean);
}, },
@ -61,4 +59,8 @@ function createChromeStorageDB(PREFIX) {
} }
} }
} }
return function dbExecChromeStorage(method, ...args) {
return METHODS[method](...args);
};
} }

95
dist/background/db.js vendored Normal file
View File

@ -0,0 +1,95 @@
/* global chromeLocal */// storage-util.js
/* global cloneError */// worker-util.js
'use strict';
/*
Initialize a database. There are some problems using IndexedDB in Firefox:
https://www.reddit.com/r/firefox/comments/74wttb/note_to_firefox_webextension_developers_who_use/
Some of them are fixed in FF59:
https://www.reddit.com/r/firefox/comments/7ijuaq/firefox_59_webextensions_can_use_indexeddb_when/
*/
/* exported db */
const db = (() => {
const DATABASE = 'stylish';
const STORE = 'styles';
const FALLBACK = 'dbInChromeStorage';
const dbApi = {
async exec(...args) {
dbApi.exec = await tryUsingIndexedDB().catch(useChromeStorage);
return dbApi.exec(...args);
},
};
return dbApi;
async function tryUsingIndexedDB() {
// we use chrome.storage.local fallback if IndexedDB doesn't save data,
// which, once detected on the first run, is remembered in chrome.storage.local
// note that accessing indexedDB may throw, https://github.com/openstyles/stylus/issues/615
if (typeof indexedDB === 'undefined') {
throw new Error('indexedDB is undefined');
}
switch (await chromeLocal.getValue(FALLBACK)) {
case true: throw null;
case false: break;
default: await testDB();
}
chromeLocal.setValue(FALLBACK, false);
return dbExecIndexedDB;
}
async function testDB() {
const id = `${performance.now()}.${Math.random()}.${Date.now()}`;
await dbExecIndexedDB('put', {id});
const e = await dbExecIndexedDB('get', id);
await dbExecIndexedDB('delete', e.id); // throws if `e` or id is null
}
async function useChromeStorage(err) {
chromeLocal.setValue(FALLBACK, true);
if (err) {
chromeLocal.setValue(FALLBACK + 'Reason', cloneError(err));
console.warn('Failed to access indexedDB. Switched to storage API.', err);
}
await require(['/background/db-chrome-storage']); /* global createChromeStorageDB */
return createChromeStorageDB();
}
async function dbExecIndexedDB(method, ...args) {
const mode = method.startsWith('get') ? 'readonly' : 'readwrite';
const store = (await open()).transaction([STORE], mode).objectStore(STORE);
const fn = method === 'putMany' ? putMany : storeRequest;
return fn(store, method, ...args);
}
function storeRequest(store, method, ...args) {
return new Promise((resolve, reject) => {
/** @type {IDBRequest} */
const request = store[method](...args);
request.onsuccess = () => resolve(request.result);
request.onerror = reject;
});
}
function putMany(store, _method, items) {
return Promise.all(items.map(item => storeRequest(store, 'put', item)));
}
function open() {
return new Promise((resolve, reject) => {
const request = indexedDB.open(DATABASE, 2);
request.onsuccess = () => resolve(request.result);
request.onerror = reject;
request.onupgradeneeded = create;
});
}
function create(event) {
if (event.oldVersion === 0) {
event.target.result.createObjectStore(STORE, {
keyPath: 'id',
autoIncrement: true,
});
}
}
})();

View File

@ -1,20 +1,19 @@
/* global API */// msg.js /* global API */// msg.js
/* global addAPI bgReady */// common.js /* global addAPI bgReady */// common.js
/* global colorScheme */
/* global prefs */ /* global prefs */
/* global tabMan */ /* global tabMan */
/* global CHROME FIREFOX UA debounce ignoreChromeError */// toolbox.js /* global CHROME FIREFOX VIVALDI debounce ignoreChromeError */// toolbox.js
'use strict'; 'use strict';
/* exported iconMan */ /* exported iconMan */
const iconMan = (() => { const iconMan = (() => {
const ICON_SIZES = FIREFOX || CHROME && !UA.vivaldi ? [16, 32] : [19, 38]; const ICON_SIZES = FIREFOX || CHROME >= 55 && !VIVALDI ? [16, 32] : [19, 38];
const staleBadges = new Set(); const staleBadges = new Set();
const imageDataCache = new Map(); const imageDataCache = new Map();
const badgeOvr = {color: '', text: ''}; const badgeOvr = {color: '', text: ''};
// https://github.com/openstyles/stylus/issues/1287 Fenix can't use custom ImageData // https://github.com/openstyles/stylus/issues/1287 Fenix can't use custom ImageData
const FIREFOX_ANDROID = FIREFOX && UA.mobile; const FIREFOX_ANDROID = FIREFOX && navigator.userAgent.includes('Android');
let isDark;
// https://github.com/openstyles/stylus/issues/335 // https://github.com/openstyles/stylus/issues/335
let hasCanvas = FIREFOX_ANDROID ? false : loadImage(`/images/icon/${ICON_SIZES[0]}.png`) let hasCanvas = FIREFOX_ANDROID ? false : loadImage(`/images/icon/${ICON_SIZES[0]}.png`)
.then(({data}) => (hasCanvas = data.some(b => b !== 255))); .then(({data}) => (hasCanvas = data.some(b => b !== 255)));
@ -38,17 +37,13 @@ const iconMan = (() => {
chrome.webNavigation.onCommitted.addListener(({tabId, frameId}) => { chrome.webNavigation.onCommitted.addListener(({tabId, frameId}) => {
if (!frameId) tabMan.set(tabId, 'styleIds', undefined); if (!frameId) tabMan.set(tabId, 'styleIds', undefined);
}); });
chrome.runtime.onConnect.addListener(port => { chrome.runtime.onConnect.addListener(port => {
if (port.name === 'iframe') { if (port.name === 'iframe') {
port.onDisconnect.addListener(onPortDisconnected); port.onDisconnect.addListener(onPortDisconnected);
} }
}); });
colorScheme.onChange(val => {
isDark = val;
if (prefs.get('iconset') === -1) {
debounce(refreshAllIcons);
}
});
bgReady.all.then(() => { bgReady.all.then(() => {
prefs.subscribe([ prefs.subscribe([
'disableAll', 'disableAll',
@ -100,10 +95,9 @@ const iconMan = (() => {
} }
function getIconName(hasStyles = false) { function getIconName(hasStyles = false) {
const i = prefs.get('iconset'); const iconset = prefs.get('iconset') === 1 ? 'light/' : '';
const prefix = i === 0 || i === -1 && isDark ? '' : 'light/';
const postfix = prefs.get('disableAll') ? 'x' : !hasStyles ? 'w' : ''; const postfix = prefs.get('disableAll') ? 'x' : !hasStyles ? 'w' : '';
return `${prefix}$SIZE$${postfix}`; return `${iconset}$SIZE$${postfix}`;
} }
function refreshIcon(tabId, force = false) { function refreshIcon(tabId, force = false) {

View File

@ -0,0 +1,15 @@
/* global chromeLocal */// storage-util.js
'use strict';
// Removing unused stuff from storage on extension update
// TODO: delete this by the middle of 2021
try {
localStorage.clear();
} catch (e) {}
setTimeout(async () => {
const del = Object.keys(await chromeLocal.get())
.filter(key => key.startsWith('usoSearchCache'));
if (del.length) chromeLocal.remove(del);
}, 15e3);

View File

@ -1,7 +1,7 @@
/* global API msg */// msg.js /* global API msg */// msg.js
/* global CHROME URLS deepEqual isEmptyObj mapObj stringAsRegExp tryRegExp tryURL */// toolbox.js /* global CHROME URLS isEmptyObj stringAsRegExp tryRegExp tryURL */// toolbox.js
/* global bgReady createCache uuidIndex */// common.js /* global bgReady compareRevision */// common.js
/* global calcStyleDigest styleCodeEmpty */// sections-util.js /* global calcStyleDigest styleCodeEmpty styleSectionGlobal */// sections-util.js
/* global db */ /* global db */
/* global prefs */ /* global prefs */
/* global tabMan */ /* global tabMan */
@ -18,26 +18,18 @@ The live preview feature relies on `runtime.connect` and `port.onDisconnect`
to cleanup the temporary code. See livePreview in /edit. to cleanup the temporary code. See livePreview in /edit.
*/ */
const styleUtil = {};
/* exported styleMan */
const styleMan = (() => { const styleMan = (() => {
Object.assign(styleUtil, {
id2style,
handleSave,
uuid2style,
});
//#region Declarations //#region Declarations
/** @typedef {{ /** @typedef {{
style: StyleObj, style: StyleObj
preview?: StyleObj, preview?: StyleObj
appliesTo: Set<string>, appliesTo: Set<string>
}} StyleMapData */ }} StyleMapData */
/** @type {Map<number,StyleMapData>} */ /** @type {Map<number,StyleMapData>} */
const dataMap = new Map(); const dataMap = new Map();
const uuidIndex = new Map();
/** @typedef {Object<styleId,{id: number, code: string[]}>} StyleSectionsToApply */ /** @typedef {Object<styleId,{id: number, code: string[]}>} StyleSectionsToApply */
/** @type {Map<string,{maybeMatch: Set<styleId>, sections: StyleSectionsToApply}>} */ /** @type {Map<string,{maybeMatch: Set<styleId>, sections: StyleSectionsToApply}>} */
const cachedStyleForUrl = createCache({ const cachedStyleForUrl = createCache({
@ -63,55 +55,18 @@ const styleMan = (() => {
name: style => `ID: ${style.id}`, name: style => `ID: ${style.id}`,
_id: () => uuidv4(), _id: () => uuidv4(),
_rev: () => Date.now(), _rev: () => Date.now(),
_usw: () => ({}),
}; };
const DELETE_IF_NULL = ['id', 'customName', 'md5Url', 'originalMd5']; const DELETE_IF_NULL = ['id', 'customName', 'md5Url', 'originalMd5'];
const INJ_ORDER = 'injectionOrder';
const order = {main: {}, prio: {}};
const orderWrap = {
id: INJ_ORDER,
value: mapObj(order, () => []),
_id: `${chrome.runtime.id}-${INJ_ORDER}`,
_rev: 0,
};
uuidIndex.addCustomId(orderWrap, {set: setOrder});
class MatchQuery {
constructor(url) {
this.url = url;
}
get urlWithoutHash() {
return this._set('urlWithoutHash', this.url.split('#', 1)[0]);
}
get urlWithoutParams() {
return this._set('urlWithoutParams', this.url.split(/[?#]/, 1)[0]);
}
get domain() {
return this._set('domain', tryURL(this.url).hostname);
}
get isOwnPage() {
return this._set('isOwnPage', this.url.startsWith(URLS.ownOrigin));
}
_set(name, value) {
Object.defineProperty(this, name, {value});
return value;
}
}
/** @type {Promise|boolean} will be `true` to avoid wasting a microtask tick on each `await` */ /** @type {Promise|boolean} will be `true` to avoid wasting a microtask tick on each `await` */
let ready = Promise.all([init(), prefs.ready]); let ready = init();
chrome.runtime.onConnect.addListener(port => { chrome.runtime.onConnect.addListener(handleLivePreview);
if (port.name === 'livePreview') { // function handleColorScheme() {
handleLivePreview(port); colorScheme.onChange(() => {
} else if (port.name.startsWith('draft:')) { for (const {style: data} of dataMap.values()) {
handleDraft(port); if (data.preferScheme === 'dark' || data.preferScheme === 'light') {
} broadcastStyleUpdated(data, 'colorScheme', undefined, false);
});
colorScheme.onChange(value => {
msg.broadcastExtension({method: 'colorScheme', value});
for (const {style} of dataMap.values()) {
if (colorScheme.SCHEMES.includes(style.preferScheme)) {
broadcastStyleUpdated(style, 'colorScheme');
} }
} }
}); });
@ -124,28 +79,22 @@ const styleMan = (() => {
/** @returns {Promise<number>} style id */ /** @returns {Promise<number>} style id */
async delete(id, reason) { async delete(id, reason) {
if (ready.then) await ready; if (ready.then) await ready;
const {style, appliesTo} = dataMap.get(id); const data = id2data(id);
const sync = reason !== 'sync'; const {style, appliesTo} = data;
const uuid = style._id; await db.exec('delete', id);
db.styles.delete(id); if (reason !== 'sync') {
if (sync) API.sync.delete(uuid, Date.now()); API.sync.delete(style._id, Date.now());
}
for (const url of appliesTo) { for (const url of appliesTo) {
const cache = cachedStyleForUrl.get(url); const cache = cachedStyleForUrl.get(url);
if (cache) delete cache.sections[id]; if (cache) delete cache.sections[id];
} }
dataMap.delete(id); dataMap.delete(id);
uuidIndex.delete(uuid); uuidIndex.delete(style._id);
mapObj(orderWrap.value, (group, type) => {
delete order[type][id];
const i = group.indexOf(uuid);
if (i >= 0) group.splice(i, 1);
});
setOrder(orderWrap, {calc: false});
if (style._usw && style._usw.token) { if (style._usw && style._usw.token) {
// Must be called after the style is deleted from dataMap // Must be called after the style is deleted from dataMap
API.usw.revoke(id); API.usw.revoke(id);
} }
API.drafts.delete(id);
await msg.broadcast({ await msg.broadcast({
method: 'styleDeleted', method: 'styleDeleted',
style: {id}, style: {id},
@ -153,6 +102,17 @@ const styleMan = (() => {
return id; return id;
}, },
/** @returns {Promise<number>} style id */
async deleteByUUID(_id, rev) {
if (ready.then) await ready;
const id = uuidIndex.get(_id);
const oldDoc = id && id2style(id);
if (oldDoc && compareRevision(oldDoc._rev, rev) <= 0) {
// FIXME: does it make sense to set reason to 'sync' in deleteByUUID?
return styleMan.delete(id, 'sync');
}
},
/** @returns {Promise<StyleObj>} */ /** @returns {Promise<StyleObj>} */
async editSave(style) { async editSave(style) {
if (ready.then) await ready; if (ready.then) await ready;
@ -162,67 +122,35 @@ const styleMan = (() => {
}, },
/** @returns {Promise<?StyleObj>} */ /** @returns {Promise<?StyleObj>} */
async find(...filters) { async find(filter) {
if (ready.then) await ready; if (ready.then) await ready;
for (const filter of filters) {
const filterEntries = Object.entries(filter); const filterEntries = Object.entries(filter);
for (const {style} of dataMap.values()) { for (const {style} of dataMap.values()) {
if (filterEntries.every(([key, val]) => style[key] === val)) { if (filterEntries.every(([key, val]) => style[key] === val)) {
return style; return style;
} }
} }
}
return null; return null;
}, },
/** @returns {Promise<StyleObj[]>} */ /** @returns {Promise<StyleObj[]>} */
async getAll() { async getAll() {
if (ready.then) await ready; if (ready.then) await ready;
return getAllAsArray(); return Array.from(dataMap.values(), data2style);
}, },
/** @returns {Promise<Object<string,StyleObj[]>>}>} */ /** @returns {Promise<StyleObj>} */
async getAllOrdered(keys) { async getByUUID(uuid) {
if (ready.then) await ready; if (ready.then) await ready;
const res = mapObj(orderWrap.value, group => group.map(uuid2style).filter(Boolean)); return id2style(uuidIndex.get(uuid));
if (res.main.length + res.prio.length < dataMap.size) {
for (const {style} of dataMap.values()) {
if (!(style.id in order.main) && !(style.id in order.prio)) {
res.main.push(style);
}
}
}
return keys
? mapObj(res, group => group.map(style => mapObj(style, null, keys)))
: res;
},
getOrder: () => orderWrap.value,
/** @returns {Promise<string | {[remoteId:string]: styleId}>}>} */
async getRemoteInfo(id) {
if (ready.then) await ready;
if (id) return calcRemoteId(id2style(id));
const res = {};
for (const {style} of dataMap.values()) {
const [rid, vars] = calcRemoteId(style);
if (rid) res[rid] = [style.id, vars];
}
return res;
}, },
/** @returns {Promise<StyleSectionsToApply>} */ /** @returns {Promise<StyleSectionsToApply>} */
async getSectionsByUrl(url, id, isInitialApply) { async getSectionsByUrl(url, id, isInitialApply) {
if (ready.then) await ready; if (ready.then) await ready;
if (isInitialApply && prefs.get('disableAll')) { if (isInitialApply && prefs.get('disableAll')) {
return { return {disableAll: true};
cfg: {
disableAll: true,
},
};
} }
// TODO: enable in FF when it supports sourceURL comment in style elements (also options.html)
const {exposeStyleName} = CHROME && prefs.__values;
const sender = CHROME && this && this.sender || {}; const sender = CHROME && this && this.sender || {};
if (sender.frameId === 0) { if (sender.frameId === 0) {
/* Chrome hides text frament from location.href of the page e.g. #:~:text=foo /* Chrome hides text frament from location.href of the page e.g. #:~:text=foo
@ -241,9 +169,9 @@ const styleMan = (() => {
} else if (cache.maybeMatch.size) { } else if (cache.maybeMatch.size) {
buildCache(cache, url, Array.from(cache.maybeMatch, id2data).filter(Boolean)); buildCache(cache, url, Array.from(cache.maybeMatch, id2data).filter(Boolean));
} }
return Object.assign({cfg: {exposeStyleName, order}}, return id
id ? mapObj(cache.sections, null, [id]) ? cache.sections[id] ? {[id]: cache.sections[id]} : {}
: cache.sections); : cache.sections;
}, },
/** @returns {Promise<StyleObj>} */ /** @returns {Promise<StyleObj>} */
@ -260,8 +188,8 @@ const styleMan = (() => {
const result = []; const result = [];
const styles = id const styles = id
? [id2style(id)].filter(Boolean) ? [id2style(id)].filter(Boolean)
: getAllAsArray(); : Array.from(dataMap.values(), data2style);
const query = new MatchQuery(url); const query = createMatchQuery(url);
for (const style of styles) { for (const style of styles) {
let excluded = false; let excluded = false;
let excludedScheme = false; let excludedScheme = false;
@ -283,7 +211,10 @@ const styleMan = (() => {
excludedScheme = true; excludedScheme = true;
} }
for (const section of style.sections) { for (const section of style.sections) {
const match = urlMatchSection(query, section, true); if (styleSectionGlobal(section) && styleCodeEmpty(section.code)) {
continue;
}
const match = urlMatchSection(query, section);
if (match) { if (match) {
if (match === 'sloppy') { if (match === 'sloppy') {
sloppy = true; sloppy = true;
@ -309,10 +240,11 @@ const styleMan = (() => {
await usercssMan.buildCode(style); await usercssMan.buildCode(style);
} }
} }
const events = await db.styles.putMany(items); const events = await db.exec('putMany', items);
return Promise.all(items.map((item, i) => return Promise.all(items.map((item, i) => {
handleSave(item, {reason: 'import'}, events[i]) afterSave(item, events[i]);
)); return handleSave(item, {reason: 'import'});
}));
}, },
/** @returns {Promise<StyleObj>} */ /** @returns {Promise<StyleObj>} */
@ -325,13 +257,33 @@ const styleMan = (() => {
return saveStyle(style, {reason}); return saveStyle(style, {reason});
}, },
save: saveStyle, /** @returns {Promise<?StyleObj>} */
async putByUUID(doc) {
async setOrder(value) {
if (ready.then) await ready; if (ready.then) await ready;
return setOrder({value}, {broadcast: true, sync: true}); const id = uuidIndex.get(doc._id);
if (id) {
doc.id = id;
} else {
delete doc.id;
}
const oldDoc = id && id2style(id);
let diff = -1;
if (oldDoc) {
diff = compareRevision(oldDoc._rev, doc._rev);
if (diff > 0) {
API.sync.put(oldDoc._id, oldDoc._rev);
return;
}
}
if (diff < 0) {
doc.id = await db.exec('put', doc);
uuidIndex.set(doc._id, doc.id);
return handleSave(doc, {reason: 'sync'});
}
}, },
save: saveStyle,
/** @returns {Promise<number>} style id */ /** @returns {Promise<number>} style id */
async toggle(id, enabled) { async toggle(id, enabled) {
if (ready.then) await ready; if (ready.then) await ready;
@ -354,8 +306,7 @@ const styleMan = (() => {
async config(id, prop, value) { async config(id, prop, value) {
if (ready.then) await ready; if (ready.then) await ready;
const style = Object.assign({}, id2style(id)); const style = Object.assign({}, id2style(id));
const {preview = {}} = dataMap.get(id); style[prop] = value;
style[prop] = preview[prop] = value;
return saveStyle(style, {reason: 'config'}); return saveStyle(style, {reason: 'config'});
}, },
}; };
@ -370,23 +321,12 @@ const styleMan = (() => {
/** @returns {?StyleObj} */ /** @returns {?StyleObj} */
function id2style(id) { function id2style(id) {
return (dataMap.get(Number(id)) || {}).style; return (dataMap.get(id) || {}).style;
} }
/** @returns {?StyleObj} */ /** @returns {?StyleObj} */
function uuid2style(uuid) { function data2style(data) {
return id2style(uuidIndex.get(uuid)); return data && data.style;
}
function calcRemoteId({md5Url, updateUrl, usercssData: ucd} = {}) {
let id;
id = (id = /\d+/.test(md5Url) || URLS.extractUsoArchiveId(updateUrl)) && `uso-${id}`
|| (id = URLS.extractUSwId(updateUrl)) && `usw-${id}`
|| '';
return id && [
id,
ucd && !isEmptyObj(ucd.vars),
];
} }
/** @returns {StyleObj} */ /** @returns {StyleObj} */
@ -407,7 +347,6 @@ const styleMan = (() => {
style, style,
appliesTo: new Set(), appliesTo: new Set(),
}); });
uuidIndex.set(style._id, style.id);
} }
/** @returns {StyleObj} */ /** @returns {StyleObj} */
@ -417,12 +356,10 @@ const styleMan = (() => {
style); style);
} }
function handleDraft(port) {
const id = port.name.split(':').pop();
port.onDisconnect.addListener(() => API.drafts.delete(Number(id) || id));
}
function handleLivePreview(port) { function handleLivePreview(port) {
if (port.name !== 'livePreview') {
return;
}
let id; let id;
port.onMessage.addListener(style => { port.onMessage.addListener(style => {
if (!id) id = style.id; if (!id) id = style.id;
@ -474,10 +411,10 @@ const styleMan = (() => {
cache.maybeMatch.add(id); cache.maybeMatch.add(id);
continue; continue;
} }
const code = getAppliedCode(new MatchQuery(url), style); const code = getAppliedCode(createMatchQuery(url), style);
if (code) { if (code) {
updated.add(url); updated.add(url);
buildCacheEntry(cache, style, code); cache.sections[id] = {id, code};
} else { } else {
excluded.add(url); excluded.add(url);
delete cache.sections[id]; delete cache.sections[id];
@ -511,24 +448,29 @@ const styleMan = (() => {
fixKnownProblems(style); fixKnownProblems(style);
} }
async function saveStyle(style, handlingOptions) { function afterSave(style, newId) {
beforeSave(style); if (style.id == null) {
const newId = await db.styles.put(style); style.id = newId;
return handleSave(style, handlingOptions, newId); }
uuidIndex.set(style._id, style.id);
API.sync.put(style._id, style._rev);
} }
function handleSave(style, {reason, broadcast = true}, id = style.id) { async function saveStyle(style, handlingOptions) {
if (style.id == null) style.id = id; beforeSave(style);
const data = id2data(id); const newId = await db.exec('put', style);
afterSave(style, newId);
return handleSave(style, handlingOptions);
}
function handleSave(style, {reason, broadcast = true}) {
const data = id2data(style.id);
const method = data ? 'styleUpdated' : 'styleAdded'; const method = data ? 'styleUpdated' : 'styleAdded';
if (!data) { if (!data) {
storeInMap(style); storeInMap(style);
} else { } else {
data.style = style; data.style = style;
} }
if (reason !== 'sync') {
API.sync.putDoc(style);
}
if (broadcast) broadcastStyleUpdated(style, reason, method); if (broadcast) broadcastStyleUpdated(style, reason, method);
return style; return style;
} }
@ -553,14 +495,15 @@ const styleMan = (() => {
} }
async function init() { async function init() {
const orderPromise = API.prefsDb.get(orderWrap.id); const styles = await db.exec('getAll') || [];
const styles = await db.styles.getAll() || [];
const updated = await Promise.all(styles.map(fixKnownProblems).filter(Boolean)); const updated = await Promise.all(styles.map(fixKnownProblems).filter(Boolean));
if (updated.length) { if (updated.length) {
await db.styles.putMany(updated); await db.exec('putMany', updated);
}
for (const style of styles) {
storeInMap(style);
uuidIndex.set(style._id, style.id);
} }
setOrder(await orderPromise, {store: false});
styles.forEach(storeInMap);
ready = true; ready = true;
bgReady._resolveStyles(); bgReady._resolveStyles();
} }
@ -644,56 +587,40 @@ const styleMan = (() => {
return true; return true;
} }
function urlMatchSection(query, section, skipEmptyGlobal) { function urlMatchSection(query, section) {
let dd, ddL, pp, ppL, rr, rrL, uu, uuL;
if ( if (
(dd = section.domains) && (ddL = dd.length) && dd.some(urlMatchDomain, query) || section.domains &&
(pp = section.urlPrefixes) && (ppL = pp.length) && pp.some(urlMatchPrefix, query) || section.domains.some(d => d === query.domain || query.domain.endsWith(`.${d}`))
/* Per the specification the fragment portion is ignored in @-moz-document:
https://www.w3.org/TR/2012/WD-css3-conditional-20120911/#url-of-doc
but the spec is outdated and doesn't account for SPA sites,
so we only respect it for `url()` function */
(uu = section.urls) && (uuL = uu.length) && (
uu.includes(query.url) ||
uu.includes(query.urlWithoutHash)
) ||
(rr = section.regexps) && (rrL = rr.length) && rr.some(urlMatchRegexp, query)
) { ) {
return true; return true;
} }
if (section.urlPrefixes && section.urlPrefixes.some(p => p && query.url.startsWith(p))) {
return true;
}
// as per spec the fragment portion is ignored in @-moz-document:
// https://www.w3.org/TR/2012/WD-css3-conditional-20120911/#url-of-doc
// but the spec is outdated and doesn't account for SPA sites
// so we only respect it for `url()` function
if (section.urls && (
section.urls.includes(query.url) ||
section.urls.includes(query.urlWithoutHash)
)) {
return true;
}
if (section.regexps && section.regexps.some(r => compileRe(r).test(query.url))) {
return true;
}
/* /*
According to CSS4 @document specification the entire URL must match. According to CSS4 @document specification the entire URL must match.
Stylish-for-Chrome implemented it incorrectly since the very beginning. Stylish-for-Chrome implemented it incorrectly since the very beginning.
We'll detect styles that abuse the bug by finding the sections that We'll detect styles that abuse the bug by finding the sections that
would have been applied by Stylish but not by us as we follow the spec. would have been applied by Stylish but not by us as we follow the spec.
*/ */
if (rrL && rr.some(urlMatchRegexpSloppy, query)) { if (section.regexps && section.regexps.some(r => compileSloppyRe(r).test(query.url))) {
return 'sloppy'; return 'sloppy';
} }
// TODO: check for invalid regexps? // TODO: check for invalid regexps?
return !rrL && !ppL && !uuL && !ddL && return styleSectionGlobal(section);
!query.isOwnPage && // We allow only intentionally targeted sections for own pages
(!skipEmptyGlobal || !styleCodeEmpty(section.code));
}
/** @this {MatchQuery} */
function urlMatchDomain(d) {
const _d = this.domain;
return d === _d ||
_d[_d.length - d.length - 1] === '.' && _d.endsWith(d);
}
/** @this {MatchQuery} */
function urlMatchPrefix(p) {
return p && this.url.startsWith(p);
}
/** @this {MatchQuery} */
function urlMatchRegexp(r) {
return (!this.isOwnPage || /\bextension\b/.test(r)) &&
compileRe(r).test(this.url);
}
/** @this {MatchQuery} */
function urlMatchRegexpSloppy(r) {
return (!this.isOwnPage || /\bextension\b/.test(r)) &&
compileSloppyRe(r).test(this.url);
} }
function createCompiler(compile) { function createCompiler(compile) {
@ -733,28 +660,45 @@ const styleMan = (() => {
'$'; '$';
} }
function buildCache(cache, url, styleList) { function createMatchQuery(url) {
const query = new MatchQuery(url); let urlWithoutHash;
for (const {style, appliesTo, preview} of styleList) { let urlWithoutParams;
const code = getAppliedCode(query, preview || style); let domain;
if (code) { return {
buildCacheEntry(cache, style, code); url,
appliesTo.add(url); get urlWithoutHash() {
if (!urlWithoutHash) {
urlWithoutHash = url.split('#')[0];
} }
return urlWithoutHash;
},
get urlWithoutParams() {
if (!urlWithoutParams) {
const u = tryURL(url);
urlWithoutParams = u.origin + u.pathname;
} }
return urlWithoutParams;
},
get domain() {
if (!domain) {
const u = tryURL(url);
domain = u.hostname;
} }
return domain;
function buildCacheEntry(cache, style, code = style.code) { },
cache.sections[style.id] = {
code,
id: style.id,
name: style.customName || style.name,
}; };
} }
/** @returns {StyleObj[]} */ function buildCache(cache, url, styleList) {
function getAllAsArray() { const query = createMatchQuery(url);
return Array.from(dataMap.values(), v => v.style); for (const {style, appliesTo, preview} of styleList) {
const code = getAppliedCode(query, preview || style);
if (code) {
const id = style.id;
cache.sections[id] = {id, code};
appliesTo.add(url);
}
}
} }
/** uuidv4 helper: converts to a 4-digit hex string and adds "-" at required positions */ /** uuidv4 helper: converts to a 4-digit hex string and adds "-" at required positions */
@ -762,30 +706,66 @@ const styleMan = (() => {
return (num + 0x10000).toString(16).slice(-4) + (i >= 1 && i <= 4 ? '-' : ''); return (num + 0x10000).toString(16).slice(-4) + (i >= 1 && i <= 4 ? '-' : '');
} }
async function setOrder(data, {broadcast, calc = true, store = true, sync} = {}) {
if (!data || !data.value || deepEqual(data.value, orderWrap.value)) {
return;
}
Object.assign(orderWrap, data, sync && {_rev: Date.now()});
if (calc) {
for (const [type, group] of Object.entries(data.value)) {
const dst = order[type] = {};
group.forEach((uuid, i) => {
const id = uuidIndex.get(uuid);
if (id) dst[id] = i;
});
}
}
if (broadcast) {
msg.broadcast({method: 'styleSort', order});
}
if (store) {
await API.prefsDb.put(orderWrap, orderWrap.id);
}
if (sync) {
API.sync.putDoc(orderWrap);
}
}
//#endregion //#endregion
})(); })();
/** Creates a FIFO limit-size map. */
function createCache({size = 1000, onDeleted} = {}) {
const map = new Map();
const buffer = Array(size);
let index = 0;
let lastIndex = 0;
return {
get(id) {
const item = map.get(id);
return item && item.data;
},
set(id, data) {
if (map.size === size) {
// full
map.delete(buffer[lastIndex].id);
if (onDeleted) {
onDeleted(buffer[lastIndex].id, buffer[lastIndex].data);
}
lastIndex = (lastIndex + 1) % size;
}
const item = {id, data, index};
map.set(id, item);
buffer[index] = item;
index = (index + 1) % size;
},
delete(id) {
const item = map.get(id);
if (!item) {
return false;
}
map.delete(item.id);
const lastItem = buffer[lastIndex];
lastItem.index = item.index;
buffer[item.index] = lastItem;
lastIndex = (lastIndex + 1) % size;
if (onDeleted) {
onDeleted(item.id, item.data);
}
return true;
},
clear() {
map.clear();
index = lastIndex = 0;
},
has: id => map.has(id),
*entries() {
for (const [id, item] of map) {
yield [id, item.data];
}
},
*values() {
for (const item of map.values()) {
yield item.data;
}
},
get size() {
return map.size;
},
};
}

View File

@ -56,7 +56,6 @@
return NOP; return NOP;
} }
return API.styles.getSectionsByUrl(url, id).then(sections => { return API.styles.getSectionsByUrl(url, id).then(sections => {
delete sections.cfg;
const tasks = []; const tasks = [];
for (const section of Object.values(sections)) { for (const section of Object.values(sections)) {
const styleId = section.id; const styleId = section.id;

View File

@ -1,5 +1,5 @@
/* global API */// msg.js /* global API */// msg.js
/* global CHROME URLS ignoreChromeError */// toolbox.js /* global CHROME ignoreChromeError */// toolbox.js
/* global prefs */ /* global prefs */
'use strict'; 'use strict';
@ -49,12 +49,6 @@
if (CHROME && !off) { if (CHROME && !off) {
chrome.webNavigation.onCommitted.addListener(injectData, {url: [{urlPrefix: 'http'}]}); chrome.webNavigation.onCommitted.addListener(injectData, {url: [{urlPrefix: 'http'}]});
} }
if (CHROME) {
chrome.webRequest.onBeforeRequest.addListener(openNamedStyle, {
urls: [URLS.ownOrigin + '*.user.css'],
types: ['main_frame'],
}, ['blocking']);
}
state.csp = csp; state.csp = csp;
state.off = off; state.off = off;
state.xhr = xhr; state.xhr = xhr;
@ -152,14 +146,6 @@
} }
} }
/** @param {chrome.webRequest.WebRequestBodyDetails} req */
function openNamedStyle(req) {
if (!req.url.includes('?')) { // skipping our usercss installer
chrome.tabs.update(req.tabId, {url: 'edit.html?id=' + req.url.split('#')[1]});
return {cancel: true};
}
}
function req2key(req) { function req2key(req) {
return req.tabId + ':' + req.frameId; return req.tabId + ':' + req.frameId;
} }

View File

@ -1,10 +1,8 @@
/* global API msg */// msg.js /* global API msg */// msg.js
/* global bgReady uuidIndex */// common.js
/* global chromeLocal chromeSync */// storage-util.js /* global chromeLocal chromeSync */// storage-util.js
/* global db */ /* global compareRevision */// common.js
/* global iconMan */ /* global iconMan */
/* global prefs */ /* global prefs */
/* global styleUtil */
/* global tokenMan */ /* global tokenMan */
'use strict'; 'use strict';
@ -30,12 +28,11 @@ const syncMan = (() => {
errorMessage: null, errorMessage: null,
login: false, login: false,
}; };
const compareRevision = (rev1, rev2) => rev1 - rev2;
let lastError = null; let lastError = null;
let ctrl; let ctrl;
let currentDrive; let currentDrive;
/** @type {Promise|boolean} will be `true` to avoid wasting a microtask tick on each `await` */ /** @type {Promise|boolean} will be `true` to avoid wasting a microtask tick on each `await` */
let ready = bgReady.styles.then(() => { let ready = prefs.ready.then(() => {
ready = true; ready = true;
prefs.subscribe('sync.enabled', prefs.subscribe('sync.enabled',
(_, val) => val === 'none' (_, val) => val === 'none'
@ -82,11 +79,11 @@ const syncMan = (() => {
} }
}, },
async putDoc({_id, _rev}) { async put(...args) {
if (ready.then) await ready; if (ready.then) await ready;
if (!currentDrive) return; if (!currentDrive) return;
schedule(); schedule();
return ctrl.put(_id, _rev); return ctrl.put(...args);
}, },
async setDriveOptions(driveName, options) { async setDriveOptions(driveName, options) {
@ -179,35 +176,19 @@ const syncMan = (() => {
//#region Utils //#region Utils
async function initController() { async function initController() {
await require(['/vendor/db-to-cloud/db-to-cloud']); /* global dbToCloud */ await require(['/vendor/db-to-cloud/db-to-cloud.min']); /* global dbToCloud */
ctrl = dbToCloud.dbToCloud({ ctrl = dbToCloud.dbToCloud({
onGet: styleUtil.uuid2style, onGet(id) {
async onPut(doc) { return API.styles.getByUUID(id);
const id = uuidIndex.get(doc._id);
const oldCust = uuidIndex.custom[id];
const oldDoc = oldCust || styleUtil.id2style(id);
const diff = oldDoc ? compareRevision(oldDoc._rev, doc._rev) : -1;
if (!diff) return;
if (diff > 0) {
syncMan.putDoc(oldDoc);
} else if (oldCust) {
uuidIndex.custom[id] = doc;
} else {
delete doc.id;
if (id) doc.id = id;
doc.id = await db.styles.put(doc);
await styleUtil.handleSave(doc, {reason: 'sync'});
}
}, },
onDelete(_id, rev) { onPut(doc) {
const id = uuidIndex.get(_id); return API.styles.putByUUID(doc);
const oldDoc = styleUtil.id2style(id); },
return oldDoc && onDelete(id, rev) {
compareRevision(oldDoc._rev, rev) <= 0 && return API.styles.deleteByUUID(id, rev);
API.styles.delete(id, 'sync');
}, },
async onFirstSync() { async onFirstSync() {
for (const i of Object.values(uuidIndex.custom).concat(await API.styles.getAll())) { for (const i of await API.styles.getAll()) {
ctrl.put(i._id, i._rev); ctrl.put(i._id, i._rev);
} }
}, },
@ -274,21 +255,11 @@ const syncMan = (() => {
if (name === 'dropbox' || name === 'google' || name === 'onedrive' || name === 'webdav') { if (name === 'dropbox' || name === 'google' || name === 'onedrive' || name === 'webdav') {
const options = await syncMan.getDriveOptions(name); const options = await syncMan.getDriveOptions(name);
options.getAccessToken = () => tokenMan.getToken(name); options.getAccessToken = () => tokenMan.getToken(name);
options.fetch = name === 'webdav' ? fetchWebDAV.bind(options) : fetch;
return dbToCloud.drive[name](options); return dbToCloud.drive[name](options);
} }
throw new Error(`unknown cloud name: ${name}`); throw new Error(`unknown cloud name: ${name}`);
} }
/** @this {Object} DriveOptions */
function fetchWebDAV(url, init = {}) {
init.credentials = 'omit'; // circumventing nextcloud CSRF token error
init.headers = Object.assign({}, init.headers, {
Authorization: `Basic ${btoa(`${this.username || ''}:${this.password || ''}`)}`,
});
return fetch(url, init);
}
function schedule(delay = SYNC_DELAY) { function schedule(delay = SYNC_DELAY) {
chrome.alarms.create('syncNow', { chrome.alarms.create('syncNow', {
delayInMinutes: delay, // fractional values are supported delayInMinutes: delay, // fractional values are supported

View File

@ -44,6 +44,9 @@ const tokenMan = (() => {
clientSecret: '9Pj=TpsrStq8K@1BiwB9PIWLppM:@s=w', clientSecret: '9Pj=TpsrStq8K@1BiwB9PIWLppM:@s=w',
authURL: 'https://login.microsoftonline.com/common/oauth2/v2.0/authorize', authURL: 'https://login.microsoftonline.com/common/oauth2/v2.0/authorize',
tokenURL: 'https://login.microsoftonline.com/common/oauth2/v2.0/token', tokenURL: 'https://login.microsoftonline.com/common/oauth2/v2.0/token',
redirect_uri: FIREFOX ?
'https://clngdbkpkpeebahjckkjfobafhncgmne.chromiumapp.org/' :
'https://' + location.hostname + '.chromiumapp.org/',
scopes: ['Files.ReadWrite.AppFolder', 'offline_access'], scopes: ['Files.ReadWrite.AppFolder', 'offline_access'],
}, },
userstylesworld: { userstylesworld: {
@ -57,9 +60,8 @@ const tokenMan = (() => {
}, },
}; };
const NETWORK_LATENCY = 30; // seconds const NETWORK_LATENCY = 30; // seconds
const DEFAULT_REDIRECT_URI = 'https://clngdbkpkpeebahjckkjfobafhncgmne.chromiumapp.org/';
let alwaysUseTab = !chrome.windows || (FIREFOX ? false : null); let alwaysUseTab = FIREFOX ? false : null;
class TokenError extends Error { class TokenError extends Error {
constructor(provider, message) { constructor(provider, message) {
@ -144,14 +146,14 @@ const tokenMan = (() => {
} }
async function authUser(keys, name, interactive = false, hooks = null) { async function authUser(keys, name, interactive = false, hooks = null) {
await require(['/vendor/webext-launch-web-auth-flow/webext-launch-web-auth-flow']); await require(['/vendor/webext-launch-web-auth-flow/webext-launch-web-auth-flow.min']);
/* global webextLaunchWebAuthFlow */ /* global webextLaunchWebAuthFlow */
const provider = AUTH[name]; const provider = AUTH[name];
const state = Math.random().toFixed(8).slice(2); const state = Math.random().toFixed(8).slice(2);
const query = { const query = {
response_type: provider.flow, response_type: provider.flow,
client_id: provider.clientId, client_id: provider.clientId,
redirect_uri: provider.redirect_uri || DEFAULT_REDIRECT_URI, redirect_uri: provider.redirect_uri || chrome.identity.getRedirectURL(),
state, state,
}; };
if (provider.scopes) { if (provider.scopes) {
@ -167,13 +169,13 @@ const tokenMan = (() => {
const url = `${provider.authURL}?${new URLSearchParams(query)}`; const url = `${provider.authURL}?${new URLSearchParams(query)}`;
const width = Math.min(screen.availWidth - 100, 800); const width = Math.min(screen.availWidth - 100, 800);
const height = Math.min(screen.availHeight - 100, 800); const height = Math.min(screen.availHeight - 100, 800);
const wnd = !alwaysUseTab && await browser.windows.getLastFocused(); const wnd = await browser.windows.getLastFocused();
const finalUrl = await webextLaunchWebAuthFlow({ const finalUrl = await webextLaunchWebAuthFlow({
url, url,
alwaysUseTab, alwaysUseTab,
interactive, interactive,
redirect_uri: query.redirect_uri, redirect_uri: query.redirect_uri,
windowOptions: wnd && Object.assign({ windowOptions: Object.assign({
state: 'normal', state: 'normal',
width, width,
height, height,
@ -248,7 +250,7 @@ const tokenMan = (() => {
// Workaround for https://github.com/openstyles/stylus/issues/1182 // Workaround for https://github.com/openstyles/stylus/issues/1182
// Note that modern Vivaldi isn't exposed in `navigator.userAgent` but it adds `extData` to tabs // Note that modern Vivaldi isn't exposed in `navigator.userAgent` but it adds `extData` to tabs
const anyTab = await getActiveTab() || (await browser.tabs.query({}))[0]; const anyTab = await getActiveTab() || (await browser.tabs.query({}))[0];
if (anyTab && !(anyTab.extData || anyTab.vivExtData)) { if (anyTab && !anyTab.extData) {
return false; return false;
} }
let bugged = true; let bugged = true;

View File

@ -1,6 +1,6 @@
/* global API */// msg.js /* global API */// msg.js
/* global RX_META URLS debounce deepMerge download ignoreChromeError */// toolbox.js /* global RX_META URLS debounce download ignoreChromeError */// toolbox.js
/* global calcStyleDigest styleSectionsEqual */ // sections-util.js /* global calcStyleDigest styleJSONseemsValid styleSectionsEqual */ // sections-util.js
/* global chromeLocal */// storage-util.js /* global chromeLocal */// storage-util.js
/* global compareVersion */// cmpver.js /* global compareVersion */// cmpver.js
/* global db */ /* global db */
@ -23,7 +23,6 @@ const updateMan = (() => {
ERROR_JSON: 'error: JSON is invalid', ERROR_JSON: 'error: JSON is invalid',
ERROR_VERSION: 'error: version is older than installed style', ERROR_VERSION: 'error: version is older than installed style',
}; };
const USO_STYLES_API = `${URLS.uso}api/v1/styles/`;
const RH_ETAG = {responseHeaders: ['etag']}; // a hashsum of file contents const RH_ETAG = {responseHeaders: ['etag']}; // a hashsum of file contents
const RX_DATE2VER = new RegExp([ const RX_DATE2VER = new RegExp([
/^(\d{4})/, /^(\d{4})/,
@ -38,7 +37,6 @@ const updateMan = (() => {
503, // service unavailable 503, // service unavailable
429, // too many requests 429, // too many requests
]; ];
let usoReferers = 0;
let lastUpdateTime; let lastUpdateTime;
let checkingAll = false; let checkingAll = false;
let logQueue = []; let logQueue = [];
@ -65,7 +63,7 @@ const updateMan = (() => {
checkingAll = true; checkingAll = true;
const port = observe && chrome.runtime.connect({name: 'updater'}); const port = observe && chrome.runtime.connect({name: 'updater'});
const styles = (await API.styles.getAll()) const styles = (await API.styles.getAll())
.filter(style => style.updateUrl && style.updatable !== false); .filter(style => style.updateUrl);
if (port) port.postMessage({count: styles.length}); if (port) port.postMessage({count: styles.length});
log(''); log('');
log(`${save ? 'Scheduled' : 'Manual'} update check for ${styles.length} styles`); log(`${save ? 'Scheduled' : 'Manual'} update check for ${styles.length} styles`);
@ -80,17 +78,17 @@ const updateMan = (() => {
/** /**
* @param {{ * @param {{
id?: number, id?: number
style?: StyleObj, style?: StyleObj
port?: chrome.runtime.Port, port?: chrome.runtime.Port
save?: boolean, save?: boolean = true
ignoreDigest?: boolean, ignoreDigest?: boolean
}} opts }} opts
* @returns {{ * @returns {{
style: StyleObj, style: StyleObj
updated?: boolean, updated?: boolean
error?: any, error?: any
STATES: UpdaterStates, STATES: UpdaterStates
}} }}
Original style digests are calculated in these cases: Original style digests are calculated in these cases:
@ -115,13 +113,12 @@ const updateMan = (() => {
save, save,
} = opts; } = opts;
if (!id) id = style.id; if (!id) id = style.id;
const {md5Url} = style; const ucd = style.usercssData;
let {usercssData: ucd, updateUrl} = style;
let res, state; let res, state;
try { try {
await checkIfEdited(); await checkIfEdited();
res = { res = {
style: await (ucd && !md5Url ? updateUsercss : updateUSO)().then(maybeSave), style: await (ucd ? updateUsercss : updateUSO)().then(maybeSave),
updated: true, updated: true,
}; };
state = STATES.UPDATED; state = STATES.UPDATED;
@ -130,7 +127,7 @@ const updateMan = (() => {
err && err.message || err && err.message ||
err; err;
res = {error, style, STATES}; res = {error, style, STATES};
state = `${STATES.SKIPPED} (${Array.isArray(err) ? err[0].message : error})`; state = `${STATES.SKIPPED} (${error})`;
} }
log(`${state} #${id} ${style.customName || style.name}`); log(`${state} #${id} ${style.customName || style.name}`);
if (port) port.postMessage(res); if (port) port.postMessage(res);
@ -145,45 +142,76 @@ const updateMan = (() => {
} }
async function updateUSO() { async function updateUSO() {
const md5 = await tryDownload(md5Url); const url = URLS.makeUsoArchiveCodeUrl(style.md5Url.match(/\d+/)[0]);
const req = await tryDownload(url, RH_ETAG).catch(() => null);
if (req) {
return updateToUSOArchive(url, req);
}
const md5 = await tryDownload(style.md5Url);
if (!md5 || md5.length !== 32) { if (!md5 || md5.length !== 32) {
return Promise.reject(STATES.ERROR_MD5); return Promise.reject(STATES.ERROR_MD5);
} }
if (md5 === style.originalMd5 && style.originalDigest && !ignoreDigest) { if (md5 === style.originalMd5 && style.originalDigest && !ignoreDigest) {
return Promise.reject(STATES.SAME_MD5); return Promise.reject(STATES.SAME_MD5);
} }
let varsUrl = ''; const json = await tryDownload(style.updateUrl, {responseType: 'json'});
if (!ucd) { if (!styleJSONseemsValid(json)) {
ucd = {}; return Promise.reject(STATES.ERROR_JSON);
varsUrl = updateUrl;
updateUrl = style.updateUrl = `${USO_STYLES_API}${md5Url.match(/\/(\d+)/)[1]}`;
}
usoSpooferStart();
let json;
try {
json = await tryDownload(style.updateUrl, {responseType: 'json'});
json = await updateUsercss(json.css) ||
(await API.uso.toUsercss(json)).style;
if (varsUrl) await API.uso.useVarsUrl(json, varsUrl);
} finally {
usoSpooferStop();
} }
// USO may not provide a correctly updated originalMd5 (#555) // USO may not provide a correctly updated originalMd5 (#555)
json.originalMd5 = md5; json.originalMd5 = md5;
return json; return json;
} }
async function updateUsercss(css) { async function updateToUSOArchive(url, req) {
const m2 = getUsoEmbeddedMeta(req.response);
if (m2) {
url = (await m2).updateUrl;
req = await tryDownload(url, RH_ETAG);
}
const json = await API.usercss.buildMeta({
id,
etag: req.headers.etag,
md5Url: null,
originalMd5: null,
sourceCode: req.response,
updateUrl: url,
url: URLS.extractUsoArchiveInstallUrl(url),
});
const varUrlValues = style.updateUrl.split('?')[1];
const varData = json.usercssData.vars;
if (varUrlValues && varData) {
const IK = 'ik-';
const IK_LEN = IK.length;
for (let [key, val] of new URLSearchParams(varUrlValues)) {
if (!key.startsWith(IK)) continue;
key = key.slice(IK_LEN);
const varDef = varData[key];
if (!varDef) continue;
if (varDef.options) {
let sel = val.startsWith(IK) && getVarOptByName(varDef, val.slice(IK_LEN));
if (!sel) {
key += '-custom';
sel = getVarOptByName(varDef, key + '-dropdown');
if (sel) varData[key].value = val;
}
if (sel) varDef.value = sel.name;
} else {
varDef.value = val;
}
}
}
return API.usercss.buildCode(json);
}
async function updateUsercss() {
let oldVer = ucd.version; let oldVer = ucd.version;
let {etag: oldEtag, updateUrl} = style; let {etag: oldEtag, updateUrl} = style;
const m2 = (css || URLS.extractUsoArchiveId(updateUrl)) && let m2 = URLS.extractUsoArchiveId(updateUrl) && getUsoEmbeddedMeta();
await getUsoEmbeddedMeta(css); if (m2 && (m2 = await m2).updateUrl) {
if (m2 && m2.updateUrl) {
updateUrl = m2.updateUrl; updateUrl = m2.updateUrl;
oldVer = m2.usercssData.version || '0'; oldVer = m2.usercssData.version || '0';
oldEtag = ''; oldEtag = '';
} else if (css) {
return;
} }
if (oldEtag && oldEtag === await downloadEtag()) { if (oldEtag && oldEtag === await downloadEtag()) {
return Promise.reject(STATES.SAME_CODE); return Promise.reject(STATES.SAME_CODE);
@ -206,7 +234,7 @@ const updateMan = (() => {
if (err && etag && !style.etag) { if (err && etag && !style.etag) {
// first check of ETAG, gonna write it directly to DB as it's too trivial to sync or announce // first check of ETAG, gonna write it directly to DB as it's too trivial to sync or announce
style.etag = etag; style.etag = etag;
await db.styles.put(style); await db.exec('put', style);
} }
return err return err
? Promise.reject(err) ? Promise.reject(err)
@ -236,7 +264,6 @@ const updateMan = (() => {
let {retryDelay = 1000} = opts; let {retryDelay = 1000} = opts;
while (true) { while (true) {
try { try {
params = deepMerge(params || {}, {headers: {'Cache-Control': 'no-cache'}});
return await download(url, params); return await download(url, params);
} catch (code) { } catch (code) {
if (!RETRY_ERRORS.includes(code) || if (!RETRY_ERRORS.includes(code) ||
@ -256,7 +283,8 @@ const updateMan = (() => {
} }
function getDateFromVer(style) { function getDateFromVer(style) {
const m = RX_DATE2VER.exec((style.usercssData || {}).version); const m = URLS.extractUsoArchiveId(style.updateUrl) &&
style.usercssData.version.match(RX_DATE2VER);
if (m) { if (m) {
m[2]--; // month is 0-based in `Date` constructor m[2]--; // month is 0-based in `Date` constructor
return new Date(...m.slice(1)).getTime(); return new Date(...m.slice(1)).getTime();
@ -265,10 +293,13 @@ const updateMan = (() => {
/** UserCSS metadata may be embedded in the original USO style so let's use its updateURL */ /** UserCSS metadata may be embedded in the original USO style so let's use its updateURL */
function getUsoEmbeddedMeta(code = style.sourceCode) { function getUsoEmbeddedMeta(code = style.sourceCode) {
const isRaw = arguments[0]; const m = code.includes('@updateURL') && code.replace(RX_META, '').match(RX_META);
const m = code.includes('@updateURL') && (isRaw ? code : code.replace(RX_META, '')).match(RX_META);
return m && API.usercss.buildMeta({sourceCode: m[0]}).catch(() => null); return m && API.usercss.buildMeta({sourceCode: m[0]}).catch(() => null);
} }
function getVarOptByName(varDef, name) {
return varDef.options.find(o => o.name === name);
}
} }
function schedule() { function schedule() {
@ -317,32 +348,4 @@ const updateMan = (() => {
logLastWriteTime = Date.now(); logLastWriteTime = Date.now();
logQueue = []; logQueue = [];
} }
function usoSpooferStart() {
if (++usoReferers === 1) {
chrome.webRequest.onBeforeSendHeaders.addListener(
usoSpoofer,
{types: ['xmlhttprequest'], urls: [USO_STYLES_API + '*']},
['blocking', 'requestHeaders', chrome.webRequest.OnBeforeSendHeadersOptions.EXTRA_HEADERS]
.filter(Boolean));
}
}
function usoSpooferStop() {
if (--usoReferers <= 0) {
usoReferers = 0;
chrome.webRequest.onBeforeSendHeaders.removeListener(usoSpoofer);
}
}
/** @param {chrome.webRequest.WebResponseHeadersDetails | browser.webRequest._OnBeforeSendHeadersDetails} info */
function usoSpoofer(info) {
if (info.tabId < 0 && URLS.ownOrigin.startsWith(info.initiator || info.originUrl || '')) {
const {requestHeaders: hh} = info;
const i = (hh.findIndex(h => /^referer$/i.test(h.name)) + 1 || hh.push({})) - 1;
hh[i].name = 'referer';
hh[i].value = URLS.uso;
return {requestHeaders: hh};
}
}
})(); })();

View File

@ -74,10 +74,6 @@ bgReady.all.then(() => {
) && download(url); ) && download(url);
} }
function makeInstallerUrl(url) {
return `${URLS.installUsercss}?updateUrl=${encodeURIComponent(url)}`;
}
function makeUsercssGlobs(host, path) { function makeUsercssGlobs(host, path) {
return '%css,%css?*,%styl,%styl?*'.replace(/%/g, `*://${host}${path}.user.`).split(','); return '%css,%css?*,%styl,%styl?*'.replace(/%/g, `*://${host}${path}.user.`).split(',');
} }
@ -86,11 +82,11 @@ bgReady.all.then(() => {
if (url.includes('.user.') && if (url.includes('.user.') &&
/^(https?|file|ftps?):/.test(url) && /^(https?|file|ftps?):/.test(url) &&
/\.user\.(css|styl)$/.test(url.split(/[#?]/, 1)[0]) && /\.user\.(css|styl)$/.test(url.split(/[#?]/, 1)[0]) &&
!oldUrl.startsWith(makeInstallerUrl(url))) { !oldUrl.startsWith(URLS.installUsercss)) {
const inTab = url.startsWith('file:') && !chrome.app; const inTab = url.startsWith('file:') && !chrome.app;
const code = await (inTab ? loadFromFile : loadFromUrl)(tabId, url); const code = await (inTab ? loadFromFile : loadFromUrl)(tabId, url);
if (!/^\s*</.test(code) && RX_META.test(code)) { if (!/^\s*</.test(code) && RX_META.test(code)) {
await openInstallerPage(tabId, url, {code, inTab}); openInstallerPage(tabId, url, {code, inTab});
} }
} }
} }
@ -103,33 +99,25 @@ bgReady.all.then(() => {
openInstallerPage(tabId, url, {}); openInstallerPage(tabId, url, {});
// Silently suppress navigation. // Silently suppress navigation.
// Don't redirect to the install URL as it'll flash the text! // Don't redirect to the install URL as it'll flash the text!
return {cancel: true}; return {redirectUrl: 'javascript:void 0'}; // eslint-disable-line no-script-url
} }
} }
async function openInstallerPage(tabId, url, {code, inTab} = {}) { function openInstallerPage(tabId, url, {code, inTab} = {}) {
const newUrl = makeInstallerUrl(url); const newUrl = `${URLS.installUsercss}?updateUrl=${encodeURIComponent(url)}`;
if (inTab) { if (inTab) {
const tab = await browser.tabs.get(tabId); browser.tabs.get(tabId).then(tab =>
return openURL({ openURL({
url: `${newUrl}&tabId=${tabId}`, url: `${newUrl}&tabId=${tabId}`,
active: tab.active, active: tab.active,
index: tab.index + 1, index: tab.index + 1,
openerTabId: tabId, openerTabId: tabId,
currentWindow: null, currentWindow: null,
}); }));
} } else {
const timer = setTimeout(clearInstallCode, 10e3, url); const timer = setTimeout(clearInstallCode, 10e3, url);
installCodeCache[url] = {code, timer}; installCodeCache[url] = {code, timer};
try { chrome.tabs.update(tabId, {url: newUrl});
await browser.tabs.update(tabId, {url: newUrl});
} catch (err) {
// FIXME: remove this when kiwi supports tabs.update
// https://github.com/openstyles/stylus/issues/1367
if (/Tabs cannot be edited right now/i.test(err.message)) {
return browser.tabs.create({url: newUrl});
}
throw err;
} }
} }

View File

@ -12,12 +12,10 @@ const usercssMan = {
name: null, name: null,
}), }),
/** `src` is a style or vars */ async assignVars(style, oldStyle) {
async assignVars(style, src) {
const meta = style.usercssData; const meta = style.usercssData;
const meta2 = src.usercssData; const vars = meta.vars;
const {vars} = meta; const oldVars = (oldStyle.usercssData || {}).vars;
const oldVars = meta2 ? meta2.vars : src;
if (vars && oldVars) { if (vars && oldVars) {
// The type of var might be changed during the update. Set value to null if the value is invalid. // The type of var might be changed during the update. Set value to null if the value is invalid.
for (const [key, v] of Object.entries(vars)) { for (const [key, v] of Object.entries(vars)) {
@ -45,7 +43,7 @@ const usercssMan = {
let log; let log;
if (!metaOnly) { if (!metaOnly) {
if (vars || assignVars) { if (vars || assignVars) {
await usercssMan.assignVars(style, vars || dup); await usercssMan.assignVars(style, vars ? {usercssData: {vars}} : dup);
} }
await usercssMan.buildCode(style); await usercssMan.buildCode(style);
log = style.log; // extracting the non-enumerable prop, otherwise it won't survive messaging log = style.log; // extracting the non-enumerable prop, otherwise it won't survive messaging
@ -139,18 +137,17 @@ const usercssMan = {
} }
}, },
async install(style, opts) { async install(style) {
return API.styles.install(await usercssMan.parse(style, opts)); return API.styles.install(await usercssMan.parse(style));
}, },
async parse(style, {dup, vars} = {}) { async parse(style) {
style = await usercssMan.buildMeta(style); style = await usercssMan.buildMeta(style);
// preserve style.vars during update // preserve style.vars during update
if (dup || (dup = await usercssMan.find(style))) { const dup = await usercssMan.find(style);
if (dup) {
style.id = dup.id; style.id = dup.id;
} await usercssMan.assignVars(style, dup);
if (vars || (vars = dup)) {
await usercssMan.assignVars(style, vars);
} }
return usercssMan.buildCode(style); return usercssMan.buildCode(style);
}, },

View File

@ -35,7 +35,7 @@ const uswApi = (() => {
const maxKeyLen = meta.reduce((res, [k]) => Math.max(res, k.length), 0); const maxKeyLen = meta.reduce((res, [k]) => Math.max(res, k.length), 0);
return [ return [
'/* ==UserStyle==', '/* ==UserStyle==',
...meta.map(([k, v]) => v && `${k}${' '.repeat(maxKeyLen - k.length + 2)}${v}`).filter(Boolean), ...meta.map(([k, v]) => `${k}${' '.repeat(maxKeyLen - k.length + 2)}${v || ''}`),
'==/UserStyle== */', '==/UserStyle== */',
].join('\n') + '\n\n'; ].join('\n') + '\n\n';
} }
@ -77,15 +77,14 @@ const uswApi = (() => {
*/ */
async publish(id, sourceCode) { async publish(id, sourceCode) {
const style = await API.styles.get(id); const style = await API.styles.get(id);
const code = style.usercssData ? sourceCode
: fakeUsercssHeader(style) + sourceCode;
const data = (style._usw || {}).token const data = (style._usw || {}).token
? style._usw ? style._usw
: await linkStyle(style, code); : await linkStyle(style, sourceCode);
const header = style.usercssData ? '' : fakeUsercssHeader(style);
return uswFetch(`style/${data.id}`, data.token, { return uswFetch(`style/${data.id}`, data.token, {
method: 'POST', method: 'POST',
headers: {'Content-Type': 'application/json'}, headers: {'Content-Type': 'application/json'},
body: JSON.stringify({code}), body: JSON.stringify({code: header + sourceCode}),
}); });
}, },

1
dist/codemirror/base.js vendored Normal file

File diff suppressed because one or more lines are too long

1
dist/codemirror/edit.js vendored Normal file

File diff suppressed because one or more lines are too long

View File

@ -5,7 +5,6 @@
(() => { (() => {
if (window.INJECTED === 1) return; if (window.INJECTED === 1) return;
window.INJECTED = 1;
/** true -> when the page styles are received, /** true -> when the page styles are received,
* false -> when disableAll mode is on at start, the styles won't be sent * false -> when disableAll mode is on at start, the styles won't be sent
@ -14,16 +13,11 @@
let hasStyles = false; let hasStyles = false;
let isDisabled = false; let isDisabled = false;
let isTab = !chrome.tabs || location.pathname !== '/popup.html'; let isTab = !chrome.tabs || location.pathname !== '/popup.html';
const order = {main: [], prio: []};
const calcOrder = ({id}) =>
(order.prio[id] || 0) * 1e6 ||
order.main[id] ||
id + .5e6; // no order = at the end of `main`
const isFrame = window !== parent; const isFrame = window !== parent;
const isFrameAboutBlank = isFrame && location.href === 'about:blank'; const isFrameAboutBlank = isFrame && location.href === 'about:blank';
const isUnstylable = !chrome.app && document instanceof XMLDocument; const isUnstylable = !chrome.app && document instanceof XMLDocument;
const styleInjector = StyleInjector({ const styleInjector = StyleInjector({
compare: (a, b) => calcOrder(a) - calcOrder(b), compare: (a, b) => a.id - b.id,
onUpdate: onInjectorUpdate, onUpdate: onInjectorUpdate,
}); });
// dynamic iframes don't have a URL yet so we'll use their parent's URL (hash isn't inherited) // dynamic iframes don't have a URL yet so we'll use their parent's URL (hash isn't inherited)
@ -44,23 +38,20 @@
/* about:blank iframes are often used by sites for file upload or background tasks /* about:blank iframes are often used by sites for file upload or background tasks
* and they may break if unexpected DOM stuff is present at `load` event * and they may break if unexpected DOM stuff is present at `load` event
* so we'll add the styles only if the iframe becomes visible */ * so we'll add the styles only if the iframe becomes visible */
const {IntersectionObserver} = window;
const xoEventId = `${Math.random()}`; const xoEventId = `${Math.random()}`;
/** @type IntersectionObserver */ /** @type IntersectionObserver */
let xo; let xo;
if (IntersectionObserver) {
window[Symbol.for('xo')] = (el, cb) => { window[Symbol.for('xo')] = (el, cb) => {
if (!xo) xo = new IntersectionObserver(onIntersect, {rootMargin: '100%'}); if (!xo) xo = new IntersectionObserver(onIntersect, {rootMargin: '100%'});
el.addEventListener(xoEventId, cb, {once: true}); el.addEventListener(xoEventId, cb, {once: true});
xo.observe(el); xo.observe(el);
}; };
}
// FIXME: move this to background page when following bugs are fixed:
// https://bugzil.la/1587723, https://crbug.com/968651
const mqDark = matchMedia('(prefers-color-scheme: dark)');
mqDark.onchange = e => API.colorScheme.updateSystemPreferDark(e.matches);
mqDark.onchange(mqDark);
// Declare all vars before init() or it'll throw due to "temporal dead zone" of const/let // Declare all vars before init() or it'll throw due to "temporal dead zone" of const/let
init(); const ready = init();
// the popup needs a check as it's not a tab but can be opened in a tab manually for whatever reason // the popup needs a check as it's not a tab but can be opened in a tab manually for whatever reason
if (!isTab) { if (!isTab) {
@ -71,17 +62,19 @@
} }
msg.onTab(applyOnMessage); msg.onTab(applyOnMessage);
window.addEventListener('pageshow', e => {
if (e.isTrusted && e.persisted) { // bfcache
updateCount();
}
});
if (!chrome.tabs) { if (!chrome.tabs) {
window.dispatchEvent(new CustomEvent(orphanEventId)); window.dispatchEvent(new CustomEvent(orphanEventId));
window.addEventListener(orphanEventId, orphanCheck, true); window.addEventListener(orphanEventId, orphanCheck, true);
} }
// detect media change in content script
// FIXME: move this to background page when following bugs are fixed:
// https://bugzilla.mozilla.org/show_bug.cgi?id=1561546
// https://bugs.chromium.org/p/chromium/issues/detail?id=968651
const media = window.matchMedia('(prefers-color-scheme: dark)');
media.addListener(() => API.colorScheme.updateSystemPreferDark().catch(console.error));
function onInjectorUpdate() { function onInjectorUpdate() {
if (!isOrphaned) { if (!isOrphaned) {
updateCount(); updateCount();
@ -107,10 +100,7 @@
parentStyles && await new Promise(onFrameElementInView) && parentStyles || parentStyles && await new Promise(onFrameElementInView) && parentStyles ||
!isFrameAboutBlank && chrome.app && !chrome.tabs && tryCatch(getStylesViaXhr) || !isFrameAboutBlank && chrome.app && !chrome.tabs && tryCatch(getStylesViaXhr) ||
await API.styles.getSectionsByUrl(matchUrl, null, true); await API.styles.getSectionsByUrl(matchUrl, null, true);
if (styles.cfg) { isDisabled = styles.disableAll;
isDisabled = styles.cfg.disableAll;
Object.assign(order, styles.cfg.order);
}
hasStyles = !isDisabled; hasStyles = !isDisabled;
if (hasStyles) { if (hasStyles) {
window[SYM] = styles; window[SYM] = styles;
@ -176,11 +166,6 @@
} }
break; break;
case 'styleSort':
Object.assign(order, request.order);
styleInjector.sort();
break;
case 'urlChanged': case 'urlChanged':
if (!hasStyles && isDisabled || matchUrl === request.url) break; if (!hasStyles && isDisabled || matchUrl === request.url) break;
matchUrl = request.url; matchUrl = request.url;
@ -190,6 +175,13 @@
}); });
break; break;
case 'backgroundReady':
ready.catch(err =>
msg.isIgnorableError(err)
? init()
: console.error(err));
break;
case 'updateCount': case 'updateCount':
updateCount(); updateCount();
break; break;
@ -239,7 +231,11 @@
} }
function onFrameElementInView(cb) { function onFrameElementInView(cb) {
if (IntersectionObserver) {
parent[parent.Symbol.for('xo')](frameElement, cb); parent[parent.Symbol.for('xo')](frameElement, cb);
} else {
requestAnimationFrame(cb);
}
} }
/** @param {IntersectionObserverEntry[]} entries */ /** @param {IntersectionObserverEntry[]} entries */
@ -259,11 +255,10 @@
} }
function orphanCheck() { function orphanCheck() {
if (chrome.runtime.id) return; if (tryCatch(() => chrome.i18n.getUILanguage())) return;
// In Chrome content script is orphaned on an extension update/reload // In Chrome content script is orphaned on an extension update/reload
// so we need to detach event listeners // so we need to detach event listeners
window.removeEventListener(orphanEventId, orphanCheck, true); window.removeEventListener(orphanEventId, orphanCheck, true);
mqDark.onchange = null;
isOrphaned = true; isOrphaned = true;
setTimeout(styleInjector.clear, 1000); // avoiding FOUC setTimeout(styleInjector.clear, 1000); // avoiding FOUC
tryCatch(msg.off, applyOnMessage); tryCatch(msg.off, applyOnMessage);

378
dist/content/install-hook-userstyles.js vendored Normal file
View File

@ -0,0 +1,378 @@
/* global API msg */// msg.js
'use strict';
// eslint-disable-next-line no-unused-expressions
/^\/styles\/(\d+)(\/([^/]*))?([?#].*)?$/.test(location.pathname) && (() => {
const styleId = RegExp.$1;
const pageEventId = `${performance.now()}${Math.random()}`;
window.dispatchEvent(new CustomEvent(chrome.runtime.id + '-install'));
window.addEventListener(chrome.runtime.id + '-install', orphanCheck, true);
document.addEventListener('stylishInstallChrome', onClick);
document.addEventListener('stylishUpdateChrome', onClick);
msg.on(onMessage);
let currentMd5;
const md5Url = getMeta('stylish-md5-url') || `https://update.userstyles.org/${styleId}.md5`;
Promise.all([
API.styles.find({md5Url}),
getResource(md5Url),
onDOMready(),
]).then(checkUpdatability);
document.documentElement.appendChild(
Object.assign(document.createElement('script'), {
textContent: `(${inPageContext})('${pageEventId}')`,
}));
function onMessage(msg) {
switch (msg.method) {
case 'ping':
// orphaned content script check
return true;
case 'openSettings':
openSettings();
return true;
}
}
/* since we are using "stylish-code-chrome" meta key on all browsers and
US.o does not provide "advanced settings" on this url if browser is not Chrome,
we need to fix this URL using "stylish-update-url" meta key
*/
function getStyleURL() {
const textUrl = getMeta('stylish-update-url') || '';
const jsonUrl = getMeta('stylish-code-chrome') ||
textUrl.replace(/styles\/(\d+)\/[^?]*/, 'styles/chrome/$1.json');
const paramsMissing = !jsonUrl.includes('?') && textUrl.includes('?');
return jsonUrl + (paramsMissing ? textUrl.replace(/^[^?]+/, '') : '');
}
function checkUpdatability([installedStyle, md5]) {
// TODO: remove the following statement when USO is fixed
document.dispatchEvent(new CustomEvent(pageEventId, {
detail: installedStyle && installedStyle.updateUrl,
}));
currentMd5 = md5;
if (!installedStyle) {
sendEvent({type: 'styleCanBeInstalledChrome'});
return;
}
const isCustomizable = /\?/.test(installedStyle.updateUrl);
const md5Url = getMeta('stylish-md5-url');
if (md5Url && installedStyle.md5Url && installedStyle.originalMd5) {
reportUpdatable(isCustomizable || md5 !== installedStyle.originalMd5);
} else {
getStyleJson().then(json => {
reportUpdatable(
isCustomizable ||
!json ||
!styleSectionsEqual(json, installedStyle));
});
}
function prepareInstallButton() {
return new Promise(resolve => {
const observer = new MutationObserver(check);
observer.observe(document.documentElement, {
childList: true,
subtree: true,
});
check();
function check() {
if (document.querySelector('#install_style_button')) {
resolve();
observer.disconnect();
}
}
});
}
function reportUpdatable(isUpdatable) {
prepareInstallButton().then(() => {
sendEvent({
type: isUpdatable
? 'styleCanBeUpdatedChrome'
: 'styleAlreadyInstalledChrome',
detail: {
updateUrl: installedStyle.updateUrl,
},
});
});
}
}
function sendEvent(event) {
sendEvent.lastEvent = event;
let {type, detail = null} = event;
if (typeof cloneInto !== 'undefined') {
// Firefox requires explicit cloning, however USO can't process our messages anyway
// because USO tries to use a global "event" variable deprecated in Firefox
detail = cloneInto({detail}, document); /* global cloneInto */
} else {
detail = {detail};
}
document.dispatchEvent(new CustomEvent(type, detail));
}
function onClick(event) {
if (onClick.processing || !orphanCheck()) {
return;
}
onClick.processing = true;
doInstall()
.then(() => {
if (!event.type.includes('Update')) {
// FIXME: sometimes the button is broken i.e. the button sends
// 'install' instead of 'update' event while the style is already
// install.
// This triggers an incorrect install count but we don't really care.
return getResource(getMeta('stylish-install-ping-url-chrome'));
}
})
.catch(console.error)
.then(done);
function done() {
setTimeout(() => {
onClick.processing = false;
});
}
}
function doInstall() {
let oldStyle;
return API.styles.find({
md5Url: getMeta('stylish-md5-url') || location.href,
})
.then(_oldStyle => {
oldStyle = _oldStyle;
return oldStyle ?
oldStyle.name :
getResource(getMeta('stylish-description'));
})
.then(name => {
const props = {};
if (oldStyle) {
props.id = oldStyle.id;
}
return saveStyleCode(oldStyle ? 'styleUpdate' : 'styleInstall', name, props);
});
}
async function saveStyleCode(message, name, addProps = {}) {
const isNew = message === 'styleInstall';
const needsConfirmation = isNew || !saveStyleCode.confirmed;
if (needsConfirmation && !confirm(chrome.i18n.getMessage(message, [name]))) {
return Promise.reject();
}
saveStyleCode.confirmed = true;
enableUpdateButton(false);
const json = await getStyleJson();
if (!json) {
prompt(chrome.i18n.getMessage('styleInstallFailed', ''),
'https://github.com/openstyles/stylus/issues/195');
return;
}
// Update originalMd5 since USO changed it (2018-11-11) to NOT match the current md5
const style = await API.styles.install(Object.assign(json, addProps, {originalMd5: currentMd5}));
if (!isNew && style.updateUrl.includes('?')) {
enableUpdateButton(true);
} else {
sendEvent({type: 'styleInstalledChrome'});
}
function enableUpdateButton(state) {
const important = s => s.replace(/;/g, '!important;');
const button = document.getElementById('update_style_button');
if (button) {
button.style.cssText = state ? '' : important('pointer-events: none; opacity: .35;');
const icon = button.querySelector('img[src*=".svg"]');
if (icon) {
icon.style.cssText = state ? '' : important('transition: transform 5s; transform: rotate(0);');
if (state) {
setTimeout(() => (icon.style.cssText += important('transform: rotate(10turn);')));
}
}
}
}
}
function getMeta(name) {
const e = document.querySelector(`link[rel="${name}"]`);
return e ? e.getAttribute('href') : null;
}
async function getResource(url, opts) {
try {
return url.startsWith('#')
? document.getElementById(url.slice(1)).textContent
: await API.download(url, opts);
} catch (error) {
alert('Error\n' + error.message);
return Promise.reject(error);
}
}
// USO providing md5Url as "https://update.update.userstyles.org/#####.md5"
// instead of "https://update.userstyles.org/#####.md5"
async function getStyleJson() {
try {
const style = await getResource(getStyleURL(), {responseType: 'json'});
const codeElement = document.getElementById('stylish-code');
if (!style || !Array.isArray(style.sections) || style.sections.length ||
codeElement && !codeElement.textContent.trim()) {
return style;
}
const code = await getResource(getMeta('stylish-update-url'));
style.sections = (await API.worker.parseMozFormat({code})).sections;
if (style.md5Url) style.md5Url = style.md5Url.replace('update.update', 'update');
return style;
} catch (e) {}
}
/**
* The sections are checked in successive order because it matters when many sections
* match the same URL and they have rules with the same CSS specificity
* @param {Object} a - first style object
* @param {Object} b - second style object
* @returns {?boolean}
*/
function styleSectionsEqual({sections: a}, {sections: b}) {
const targets = ['urls', 'urlPrefixes', 'domains', 'regexps'];
return a && b && a.length === b.length && a.every(sameSection);
function sameSection(secA, i) {
return equalOrEmpty(secA.code, b[i].code, 'string', (a, b) => a === b) &&
targets.every(target => equalOrEmpty(secA[target], b[i][target], 'array', arrayMirrors));
}
function equalOrEmpty(a, b, type, comparator) {
const typeA = type === 'array' ? Array.isArray(a) : typeof a === type;
const typeB = type === 'array' ? Array.isArray(b) : typeof b === type;
return typeA && typeB && comparator(a, b) ||
(a == null || typeA && !a.length) &&
(b == null || typeB && !b.length);
}
function arrayMirrors(a, b) {
return a.length === b.length &&
a.every(el => b.includes(el)) &&
b.every(el => a.includes(el));
}
}
function onDOMready() {
return document.readyState !== 'loading'
? Promise.resolve()
: new Promise(resolve => document.addEventListener('DOMContentLoaded', resolve, {once: true}));
}
function openSettings(countdown = 10e3) {
const button = document.querySelector('.customize_button');
if (button) {
button.dispatchEvent(new MouseEvent('click', {bubbles: true}));
setTimeout(function pollArea(countdown = 2000) {
const area = document.getElementById('advancedsettings_area');
if (area || countdown < 0) {
(area || button).scrollIntoView({behavior: 'smooth', block: area ? 'end' : 'center'});
} else {
setTimeout(pollArea, 100, countdown - 100);
}
}, 500);
} else if (countdown > 0) {
setTimeout(openSettings, 100, countdown - 100);
}
}
function orphanCheck() {
try {
if (chrome.i18n.getUILanguage()) {
return true;
}
} catch (e) {}
// In Chrome content script is orphaned on an extension update/reload
// so we need to detach event listeners
window.removeEventListener(chrome.runtime.id + '-install', orphanCheck, true);
document.removeEventListener('stylishInstallChrome', onClick);
document.removeEventListener('stylishUpdateChrome', onClick);
try {
msg.off(onMessage);
} catch (e) {}
}
})();
function inPageContext(eventId) {
document.currentScript.remove();
window.isInstalled = true;
const origMethods = {
json: Response.prototype.json,
byId: document.getElementById,
};
let vars;
// USO bug workaround: prevent errors in console after install and busy cursor
document.getElementById = id =>
origMethods.byId.call(document, id) ||
(/^(stylish-code|stylish-installed-style-installed-\w+|post-install-ad|style-install-unknown)$/.test(id)
? Object.assign(document.createElement('p'), {className: 'afterdownload-ad'})
: null);
// USO bug workaround: use the actual image data in customized settings
document.addEventListener(eventId, ({detail}) => {
vars = /\?/.test(detail) && new URL(detail).searchParams;
if (!vars) Response.prototype.json = origMethods.json;
}, {once: true});
Response.prototype.json = async function () {
const json = await origMethods.json.apply(this, arguments);
if (vars && json && Array.isArray(json.style_settings)) {
Response.prototype.json = origMethods.json;
const images = new Map();
for (const ss of json.style_settings) {
let value = vars.get('ik-' + ss.install_key);
if (!value || !(ss.style_setting_options || [])[0]) {
continue;
}
if (value.startsWith('ik-')) {
value = value.replace(/^ik-/, '');
const def = ss.style_setting_options.find(item => item.default);
if (!def || def.install_key !== value) {
if (def) def.default = false;
for (const item of ss.style_setting_options) {
if (item.install_key === value) {
item.default = true;
break;
}
}
}
} else if (ss.setting_type === 'image') {
let isListed;
for (const opt of ss.style_setting_options) {
isListed |= opt.default = (opt.value === value);
}
images.set(ss.install_key, {url: value, isListed});
} else {
const item = ss.style_setting_options[0];
if (item.value !== value && item.install_key === 'placeholder') {
item.value = value;
}
}
}
if (images.size) {
new MutationObserver((_, observer) => {
if (document.getElementById('style-settings')) {
observer.disconnect();
for (const [name, {url, isListed}] of images) {
const elRadio = document.querySelector(`input[name="ik-${name}"][value="user-url"]`);
const elUrl = elRadio &&
document.getElementById(elRadio.id.replace('url-choice', 'user-url'));
if (elUrl) {
elRadio.checked = !isListed;
elUrl.value = url;
}
}
}
}).observe(document, {childList: true, subtree: true});
}
}
return json;
};
}

View File

@ -9,14 +9,12 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
const PATCH_ID = 'transition-patch'; const PATCH_ID = 'transition-patch';
// styles are out of order if any of these elements is injected between them // styles are out of order if any of these elements is injected between them
const ORDERED_TAGS = new Set(['head', 'body', 'frameset', 'style', 'link']); const ORDERED_TAGS = new Set(['head', 'body', 'frameset', 'style', 'link']);
const docRewriteObserver = RewriteObserver(sort); const docRewriteObserver = RewriteObserver(_sort);
const docRootObserver = RootObserver(sortIfNeeded); const docRootObserver = RootObserver(_sortIfNeeded);
const toSafeChar = c => String.fromCharCode(0xFF00 + c.charCodeAt(0) - 0x20);
const list = []; const list = [];
const table = new Map(); const table = new Map();
let isEnabled = true; let isEnabled = true;
let isTransitionPatched = chrome.app && CSS.supports('accent-color', 'red'); // Chrome 93 let isTransitionPatched;
let exposeStyleName;
// will store the original method refs because the page can override them // will store the original method refs because the page can override them
let creationDoc, createElement, createElementNS; let creationDoc, createElement, createElementNS;
@ -25,24 +23,24 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
list, list,
async apply(styleMap) { async apply(styleMap) {
const styles = styleMapToArray(styleMap); const styles = _styleMapToArray(styleMap);
const value = !styles.length const value = !styles.length
? [] ? []
: await docRootObserver.evade(() => { : await docRootObserver.evade(() => {
if (!isTransitionPatched && isEnabled) { if (!isTransitionPatched && isEnabled) {
applyTransitionPatch(styles); _applyTransitionPatch(styles);
} }
return styles.map(addUpdate); return styles.map(_addUpdate);
}); });
emitUpdate(); _emitUpdate();
return value; return value;
}, },
clear() { clear() {
addRemoveElements(false); _addRemoveElements(false);
list.length = 0; list.length = 0;
table.clear(); table.clear();
emitUpdate(); _emitUpdate();
}, },
clearOrphans() { clearOrphans() {
@ -55,12 +53,12 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
}, },
remove(id) { remove(id) {
remove(id); _remove(id);
emitUpdate(); _emitUpdate();
}, },
replace(styleMap) { replace(styleMap) {
const styles = styleMapToArray(styleMap); const styles = _styleMapToArray(styleMap);
const added = new Set(styles.map(s => s.id)); const added = new Set(styles.map(s => s.id));
const removed = []; const removed = [];
for (const style of list) { for (const style of list) {
@ -68,24 +66,22 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
removed.push(style.id); removed.push(style.id);
} }
} }
styles.forEach(addUpdate); styles.forEach(_addUpdate);
removed.forEach(remove); removed.forEach(_remove);
emitUpdate(); _emitUpdate();
}, },
toggle(enable) { toggle(enable) {
if (isEnabled === enable) return; if (isEnabled === enable) return;
isEnabled = enable; isEnabled = enable;
if (!enable) toggleObservers(false); if (!enable) _toggleObservers(false);
addRemoveElements(enable); _addRemoveElements(enable);
if (enable) toggleObservers(true); if (enable) _toggleObservers(true);
}, },
sort: sort,
}; };
function add(style) { function _add(style) {
const el = style.el = createStyle(style); const el = style.el = _createStyle(style.id, style.code);
const i = list.findIndex(item => compare(item, style) > 0); const i = list.findIndex(item => compare(item, style) > 0);
table.set(style.id, style); table.set(style.id, style);
if (isEnabled) { if (isEnabled) {
@ -95,7 +91,7 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
return el; return el;
} }
function addRemoveElements(add) { function _addRemoveElements(add) {
for (const {el} of list) { for (const {el} of list) {
if (add) { if (add) {
document.documentElement.appendChild(el); document.documentElement.appendChild(el);
@ -105,11 +101,11 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
} }
} }
function addUpdate(style) { function _addUpdate(style) {
return table.has(style.id) ? update(style) : add(style); return table.has(style.id) ? _update(style) : _add(style);
} }
function applyTransitionPatch(styles) { function _applyTransitionPatch(styles) {
isTransitionPatched = true; isTransitionPatched = true;
// CSS transition bug workaround: since we insert styles asynchronously, // CSS transition bug workaround: since we insert styles asynchronously,
// the browsers, especially Firefox, may apply all transitions on page load // the browsers, especially Firefox, may apply all transitions on page load
@ -118,20 +114,19 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
!styles.some(s => s.code.includes('transition'))) { !styles.some(s => s.code.includes('transition'))) {
return; return;
} }
const el = createStyle({id: PATCH_ID, code: ` const el = _createStyle(PATCH_ID, `
:root:not(#\\0):not(#\\0) * { :root:not(#\\0):not(#\\0) * {
transition: none !important; transition: none !important;
} }
`}); `);
document.documentElement.appendChild(el); document.documentElement.appendChild(el);
// wait for the next paint to complete // wait for the next paint to complete
// note: requestAnimationFrame won't fire in inactive tabs // note: requestAnimationFrame won't fire in inactive tabs
requestAnimationFrame(() => setTimeout(() => el.remove())); requestAnimationFrame(() => setTimeout(() => el.remove()));
} }
function createStyle(style = {}) { function _createStyle(id, code = '') {
const {id} = style; if (!creationDoc) _initCreationDoc();
if (!creationDoc) initCreationDoc();
let el; let el;
if (document.documentElement instanceof SVGSVGElement) { if (document.documentElement instanceof SVGSVGElement) {
// SVG document style // SVG document style
@ -151,27 +146,18 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
el.type = 'text/css'; el.type = 'text/css';
// SVG className is not a string, but an instance of SVGAnimatedString // SVG className is not a string, but an instance of SVGAnimatedString
el.classList.add('stylus'); el.classList.add('stylus');
setTextAndName(el, style); el.textContent = code;
return el; return el;
} }
function setTextAndName(el, {id, code = '', name}) { function _toggleObservers(shouldStart) {
if (exposeStyleName && name) {
el.dataset.name = name;
name = encodeURIComponent(name.replace(/[?#/']/g, toSafeChar));
code += `\n/*# sourceURL=${chrome.runtime.getURL(name)}.user.css#${id} */`;
}
el.textContent = code;
}
function toggleObservers(shouldStart) {
const onOff = shouldStart && isEnabled ? 'start' : 'stop'; const onOff = shouldStart && isEnabled ? 'start' : 'stop';
docRewriteObserver[onOff](); docRewriteObserver[onOff]();
docRootObserver[onOff](); docRootObserver[onOff]();
} }
function emitUpdate() { function _emitUpdate() {
toggleObservers(list.length); _toggleObservers(list.length);
onUpdate(); onUpdate();
} }
@ -182,11 +168,11 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
and since userAgent.navigator can be spoofed via about:config or devtools, and since userAgent.navigator can be spoofed via about:config or devtools,
we're checking for getPreventDefault that was removed in FF59 we're checking for getPreventDefault that was removed in FF59
*/ */
function initCreationDoc() { function _initCreationDoc() {
creationDoc = !Event.prototype.getPreventDefault && document.wrappedJSObject; creationDoc = !Event.prototype.getPreventDefault && document.wrappedJSObject;
if (creationDoc) { if (creationDoc) {
({createElement, createElementNS} = creationDoc); ({createElement, createElementNS} = creationDoc);
const el = document.documentElement.appendChild(createStyle()); const el = document.documentElement.appendChild(_createStyle());
const isApplied = el.sheet; const isApplied = el.sheet;
el.remove(); el.remove();
if (isApplied) return; if (isApplied) return;
@ -195,7 +181,7 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
({createElement, createElementNS} = document); ({createElement, createElementNS} = document);
} }
function remove(id) { function _remove(id) {
const style = table.get(id); const style = table.get(id);
if (!style) return; if (!style) return;
table.delete(id); table.delete(id);
@ -203,14 +189,14 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
style.el.remove(); style.el.remove();
} }
function sort() { function _sort() {
docRootObserver.evade(() => { docRootObserver.evade(() => {
list.sort(compare); list.sort(compare);
addRemoveElements(true); _addRemoveElements(true);
}); });
} }
function sortIfNeeded() { function _sortIfNeeded() {
let needsSort; let needsSort;
let el = list.length && list[0].el; let el = list.length && list[0].el;
if (!el) { if (!el) {
@ -231,29 +217,22 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
// some styles are not injected to the document // some styles are not injected to the document
if (i < list.length) needsSort = true; if (i < list.length) needsSort = true;
} }
if (needsSort) sort(); if (needsSort) _sort();
return needsSort; return needsSort;
} }
function styleMapToArray(styleMap) { function _styleMapToArray(styleMap) {
if (styleMap.cfg) { return Object.values(styleMap).map(s => ({
({exposeStyleName} = styleMap.cfg); id: s.id,
delete styleMap.cfg; code: s.code.join(''),
}
return Object.values(styleMap).map(({id, code, name}) => ({
id,
name,
code: code.join(''),
})); }));
} }
function update(newStyle) { function _update({id, code}) {
const {id, code} = newStyle;
const style = table.get(id); const style = table.get(id);
if (style.code !== code || if (style.code !== code) {
style.name !== newStyle.name && exposeStyleName) {
style.code = code; style.code = code;
setTextAndName(style.el, newStyle); style.el.textContent = code;
} }
} }
@ -262,14 +241,14 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
let root; let root;
let observing = false; let observing = false;
let timer; let timer;
const observer = new MutationObserver(check); const observer = new MutationObserver(_check);
return {start, stop}; return {start, stop};
function start() { function start() {
if (observing) return; if (observing) return;
// detect dynamic iframes rewritten after creation by the embedder i.e. externally // detect dynamic iframes rewritten after creation by the embedder i.e. externally
root = document.documentElement; root = document.documentElement;
timer = setTimeout(check); timer = setTimeout(_check);
observer.observe(document, {childList: true}); observer.observe(document, {childList: true});
observing = true; observing = true;
} }
@ -281,7 +260,7 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
observing = false; observing = false;
} }
function check() { function _check() {
if (root !== document.documentElement) { if (root !== document.documentElement) {
root = document.documentElement; root = document.documentElement;
onChange(); onChange();
@ -311,7 +290,7 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
function evade(fn) { function evade(fn) {
const restore = observing && start; const restore = observing && start;
stop(); stop();
return new Promise(resolve => run(fn, resolve, waitForRoot)) return new Promise(resolve => _run(fn, resolve, _waitForRoot))
.then(restore); .then(restore);
} }
@ -329,7 +308,7 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
observing = false; observing = false;
} }
function run(fn, resolve, wait) { function _run(fn, resolve, wait) {
if (document.documentElement) { if (document.documentElement) {
resolve(fn()); resolve(fn());
return true; return true;
@ -337,8 +316,8 @@ window.StyleInjector = window.INJECTED === 1 ? window.StyleInjector : ({
if (wait) wait(fn, resolve); if (wait) wait(fn, resolve);
} }
function waitForRoot(...args) { function _waitForRoot(...args) {
new MutationObserver((_, observer) => run(...args) && observer.disconnect()) new MutationObserver((_, observer) => _run(...args) && observer.disconnect())
.observe(document, {childList: true}); .observe(document, {childList: true});
} }
} }

View File

@ -5,8 +5,7 @@
<meta name="viewport" content="width=device-width, initial-scale=1.0"> <meta name="viewport" content="width=device-width, initial-scale=1.0">
<link href="global.css" rel="stylesheet"> <link href="global.css" rel="stylesheet">
<link href="global-dark.css" rel="stylesheet"> <link id="cm-theme" rel="stylesheet">
<style id="cm-theme"></style>
<script src="js/polyfill.js"></script> <script src="js/polyfill.js"></script>
<script src="js/toolbox.js"></script> <script src="js/toolbox.js"></script>
@ -18,38 +17,19 @@
<script src="content/apply.js"></script> <script src="content/apply.js"></script>
<script src="js/sections-util.js"></script> <script src="js/sections-util.js"></script>
<script src="js/storage-util.js"></script> <script src="js/event-emitter.js"></script>
<script src="edit/codemirror-themes.js"></script> <!-- must precede base.js --> <script src="edit/codemirror-themes.js"></script> <!-- must precede base.js -->
<script src="edit/base.js"></script> <script src="edit/base.js"></script>
<script src="vendor/codemirror/lib/codemirror.js"></script> <script src="codemirror/base.js"></script><script src="codemirror/edit.js"></script> <!-- codemirror-edit -->
<script src="vendor/codemirror/mode/css/css.js"></script>
<script src="vendor/codemirror/mode/stylus/stylus.js"></script>
<script src="vendor/codemirror/addon/dialog/dialog.js"></script>
<script src="vendor/codemirror/addon/edit/closebrackets.js"></script>
<script src="vendor/codemirror/addon/scroll/annotatescrollbar.js"></script>
<script src="vendor/codemirror/addon/search/searchcursor.js"></script>
<script src="vendor/codemirror/addon/search/matchesonscrollbar.js"></script>
<script src="vendor/codemirror/addon/comment/comment.js"></script>
<script src="vendor/codemirror/addon/selection/active-line.js"></script>
<script src="vendor/codemirror/addon/edit/matchbrackets.js"></script>
<script src="vendor/codemirror/addon/fold/foldcode.js"></script>
<script src="vendor/codemirror/addon/fold/foldgutter.js"></script>
<script src="vendor/codemirror/addon/fold/brace-fold.js"></script>
<script src="vendor/codemirror/addon/fold/indent-fold.js"></script>
<script src="vendor/codemirror/addon/fold/comment-fold.js"></script>
<script src="vendor/codemirror/addon/lint/lint.js"></script>
<script src="vendor/codemirror/addon/hint/show-hint.js"></script>
<script src="vendor/codemirror/addon/hint/css-hint.js"></script>
<script src="vendor/codemirror/keymap/emacs.js"></script>
<script src="vendor/codemirror/keymap/sublime.js"></script>
<script src="vendor/codemirror/keymap/vim.js"></script>
<script src="vendor-overwrites/codemirror-addon/match-highlighter.js"></script> <script src="vendor-overwrites/codemirror-addon/match-highlighter.js"></script>
<script src="vendor/lz-string-unsafe/lz-string-unsafe.min.js"></script>
<script src="js/color/color-converter.js"></script> <script src="js/color/color-converter.js"></script>
<script src="js/color/color-mimicry.js"></script> <script src="js/color/color-mimicry.js"></script>
<script src="js/color/color-picker.js"></script> <script src="js/color/color-picker.js"></script>
<script src="js/color/color-view.js"></script> <script src="js/color/color-view.js"></script>
<script src="js/storage-util.js"></script>
<script src="js/worker-util.js"></script> <script src="js/worker-util.js"></script>
<script src="edit/util.js"></script> <script src="edit/util.js"></script>
@ -63,24 +43,26 @@
<script src="edit/sections-editor-section.js"></script> <script src="edit/sections-editor-section.js"></script>
<script src="edit/sections-editor.js"></script> <script src="edit/sections-editor.js"></script>
<script src="edit/usw-integration.js"></script> <script src="edit/usw-integration.js"></script>
<script src="edit/settings.js"></script>
<script src="edit/edit.js"></script>
<template data-id="appliesTo"> <template data-id="appliesTo">
<li class="applies-to-item"> <li class="applies-to-item">
<div class="select-resizer"> <div class="select-resizer">
<select name="applies-type" class="applies-type style-contributor"> <select name="applies-type" class="applies-type style-contributor">
<option value="url" i18n="appliesUrlOption"></option> <option value="url" i18n-text="appliesUrlOption"></option>
<option value="url-prefix" i18n="appliesUrlPrefixOption"></option> <option value="url-prefix" i18n-text="appliesUrlPrefixOption"></option>
<option value="domain" i18n="appliesDomainOption"></option> <option value="domain" i18n-text="appliesDomainOption"></option>
<option value="regexp" i18n="appliesRegexpOption"></option> <option value="regexp" i18n-text="appliesRegexpOption"></option>
</select> </select>
<svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg> <svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg>
</div> </div>
<div class="applies-value-wrapper"> <div class="applies-value-wrapper">
<input name="applies-value" class="applies-value style-contributor" spellcheck="false"> <input name="applies-value" class="applies-value style-contributor" spellcheck="false">
<a class="remove-applies-to" i18n="appliesRemove, title:appliesRemove" tabindex="0"> <a class="remove-applies-to" i18n-text="appliesRemove" i18n-title="appliesRemove" tabindex="0">
<svg class="svg-icon remove"><use xlink:href="#svg-icon-minus"/></svg> <svg class="svg-icon remove"><use xlink:href="#svg-icon-minus"/></svg>
</a> </a>
<a class="add-applies-to" i18n="appliesAdd, title:appliesAdd" tabindex="0"> <a class="add-applies-to" i18n-text="appliesAdd" i18n-title="appliesAdd" tabindex="0">
<svg class="svg-icon add"><use xlink:href="#svg-icon-plus"/></svg> <svg class="svg-icon add"><use xlink:href="#svg-icon-plus"/></svg>
</a> </a>
</div> </div>
@ -88,8 +70,8 @@
</template> </template>
<template data-id="appliesToEverything"> <template data-id="appliesToEverything">
<li class="applies-to-everything" i18n="appliesToEverything"> <li class="applies-to-everything" i18n-text="appliesToEverything">
<a class="add-applies-to" i18n="appliesAdd, title:appliesAdd" tabindex="0"> <a class="add-applies-to" i18n-text="appliesAdd" i18n-title="appliesAdd" tabindex="0">
<svg class="svg-icon add"><use xlink:href="#svg-icon-plus"/></svg> <svg class="svg-icon add"><use xlink:href="#svg-icon-plus"/></svg>
</a> </a>
</li> </li>
@ -99,11 +81,11 @@
<div class="section"> <div class="section">
<!-- not using DIV to make our CSS work for #sections > div:only-of-type .remove-section --> <!-- not using DIV to make our CSS work for #sections > div:only-of-type .remove-section -->
<p class="deleted-section"> <p class="deleted-section">
<button class="restore-section" i18n="sectionRestore"></button> <button class="restore-section" i18n-text="sectionRestore"></button>
</p> </p>
<label i18n="sectionCode" class="code-label"></label> <label i18n-text="sectionCode" class="code-label"></label>
<div class="applies-to"> <div class="applies-to">
<label i18n="appliesLabel, title:appliesHelp" data-cmd="note"> <label i18n-text="appliesLabel">
<a class="svg-inline-wrapper applies-to-help" tabindex="0"> <a class="svg-inline-wrapper applies-to-help" tabindex="0">
<svg class="svg-icon info"><use xlink:href="#svg-icon-help"/></svg> <svg class="svg-icon info"><use xlink:href="#svg-icon-help"/></svg>
</a> </a>
@ -111,13 +93,13 @@
<ul class="applies-to-list"></ul> <ul class="applies-to-list"></ul>
</div> </div>
<div class="edit-actions"> <div class="edit-actions">
<button class="remove-section" i18n="sectionRemove"></button> <button class="remove-section" i18n-text="sectionRemove"></button>
<button class="add-section" i18n="long-text:sectionAdd, short-text:genericAdd"></button> <button class="add-section" i18n-long-text="sectionAdd" i18n-short-text="genericAdd"></button>
<button class="clone-section" i18n="genericClone"></button> <button class="clone-section" i18n-text="genericClone"></button>
<button class="move-section-up"></button> <button class="move-section-up"></button>
<button class="move-section-down"></button> <button class="move-section-down"></button>
<button class="beautify-section" i18n="styleBeautify"></button> <button class="beautify-section" i18n-text="styleBeautify"></button>
<button class="test-regexp" i18n="genericTest"></button> <button class="test-regexp" i18n-text="styleRegexpTestButton"></button>
</div> </div>
</div> </div>
</template> </template>
@ -127,27 +109,27 @@
<div data-type="main"> <div data-type="main">
<div data-type="content"></div> <div data-type="content"></div>
<div data-type="actions"> <div data-type="actions">
<a data-action="case" i18n="title:searchCaseSensitive" tabindex="0">Aa</a> <a data-action="case" i18n-title="searchCaseSensitive" tabindex="0">Aa</a>
<a data-action="prev" i18n="title:genericPrevious" data-hotkey-tooltip="findPrev" tabindex="0"> <a data-action="prev" i18n-title="genericPrevious" data-hotkey-tooltip="findPrev" tabindex="0">
<svg class="svg-icon" style="transform: rotate(180deg)"><use xlink:href="#svg-icon-v"/></svg> <svg class="svg-icon" style="transform: rotate(180deg)"><use xlink:href="#svg-icon-v"/></svg>
</a> </a>
<a data-action="next" i18n="title:genericNext" data-hotkey-tooltip="findNext" tabindex="0"> <a data-action="next" i18n-title="genericNext" data-hotkey-tooltip="findNext" tabindex="0">
<svg class="svg-icon"><use xlink:href="#svg-icon-v"/></svg> <svg class="svg-icon"><use xlink:href="#svg-icon-v"/></svg>
</a> </a>
<a data-action="close" i18n="title:confirmClose" data-hotkey-tooltip="=Esc" tabindex="0"> <a data-action="close" i18n-title="confirmClose" data-hotkey-tooltip="=Esc" tabindex="0">
<svg class="svg-icon dismiss"><use xlink:href="#svg-icon-close"/></svg> <svg class="svg-icon dismiss"><use xlink:href="#svg-icon-close"/></svg>
</a> </a>
</div> </div>
</div> </div>
<div data-type="status"> <div data-type="status">
<div class="CodeMirror-search-hint" i18n-text="searchRegexp"></div> <div class="CodeMirror-search-hint" i18n-text="searchRegexp"></div>
<div data-type="tally" i18n="title:searchNumberOfResults"></div> <div data-type="tally" i18n-title="searchNumberOfResults"></div>
</div> </div>
</div> </div>
</template> </template>
<template data-id="clearSearch"> <template data-id="clearSearch">
<div data-type="hover" i18n="title:confirmDelete"> <div data-type="hover" i18n-title="confirmDelete">
<svg data-action="clear" class="svg-icon"><use xlink:href="#svg-icon-close"></use></svg> <svg data-action="clear" class="svg-icon"><use xlink:href="#svg-icon-close"></use></svg>
</div> </div>
</template> </template>
@ -156,7 +138,7 @@
<div data-type="content"> <div data-type="content">
<div data-type="input-wrapper"> <div data-type="input-wrapper">
<textarea class="CodeMirror-search-field" rows="1" spellcheck="false" required <textarea class="CodeMirror-search-field" rows="1" spellcheck="false" required
i18n="placeholder:search"></textarea> i18n-placeholder="search"></textarea>
</div> </div>
</div> </div>
</template> </template>
@ -165,36 +147,36 @@
<div data-type="content"> <div data-type="content">
<div data-type="input-wrapper"> <div data-type="input-wrapper">
<textarea data-type="replace-from" <textarea data-type="replace-from"
i18n="placeholder:replace" i18n-placeholder="replace"
class="CodeMirror-search-field" rows="1" required class="CodeMirror-search-field" rows="1" required
spellcheck="false"></textarea> spellcheck="false"></textarea>
</div> </div>
<div data-type="input-wrapper"> <div data-type="input-wrapper">
<textarea data-type="replace-to" <textarea data-type="replace-to"
i18n="placeholder:replaceWith" i18n-placeholder="replaceWith"
class="CodeMirror-search-field" rows="1" required class="CodeMirror-search-field" rows="1" required
spellcheck="false"></textarea> spellcheck="false"></textarea>
</div> </div>
<button data-action="replace" i18n="replace" disabled></button> <button data-action="replace" i18n-text="replace" disabled></button>
<button data-action="replaceAll" i18n="replaceAll" disabled></button> <button data-action="replaceAll" i18n-text="replaceAll" disabled></button>
<button data-action="undo" i18n="undo" disabled></button> <button data-action="undo" i18n-text="undo" disabled></button>
<!-- <!--
Using a separate set of buttons because Using a separate set of buttons because
1. FF can display tooltips only when specified on the <button>, ignores the nested <title> in <svg> 1. FF can display tooltips only when specified on the <button>, ignores the nested <title> in <svg>
2. the icon doesn't fill the entire button area so tooltips aren't shown when the edges are hovered 2. the icon doesn't fill the entire button area so tooltips aren't shown when the edges are hovered
--> -->
<button class="hidden" data-action="replace" i18n="title:replace" disabled> <button class="hidden" data-action="replace" i18n-title="replace" disabled>
<svg class="svg-icon" viewBox="0 0 20 20"> <svg class="svg-icon" viewBox="0 0 20 20">
<polygon points="15.83 4.75 8.76 11.82 5.2 8.26 3.51 9.95 8.76 15.19 17.52 6.43 15.83 4.75"/> <polygon points="15.83 4.75 8.76 11.82 5.2 8.26 3.51 9.95 8.76 15.19 17.52 6.43 15.83 4.75"/>
</svg> </svg>
</button> </button>
<button class="hidden" data-action="replaceAll" i18n="title:replaceAll" disabled> <button class="hidden" data-action="replaceAll" i18n-title="replaceAll" disabled>
<svg class="svg-icon" viewBox="0 0 20 20"> <svg class="svg-icon" viewBox="0 0 20 20">
<polygon points="15.8,1.8 8.8,8.8 5.2,5.3 3.5,6.9 8.8,12.2 17.5,3.4 "/> <polygon points="15.8,1.8 8.8,8.8 5.2,5.3 3.5,6.9 8.8,12.2 17.5,3.4 "/>
<polygon points="15.8,7.8 8.8,14.8 5.2,11.3 3.5,12.9 8.8,18.2 17.5,9.4 "/> <polygon points="15.8,7.8 8.8,14.8 5.2,11.3 3.5,12.9 8.8,18.2 17.5,9.4 "/>
</svg> </svg>
</button> </button>
<button class="hidden" data-action="undo" i18n="title:undo" disabled> <button class="hidden" data-action="undo" i18n-title="undo" disabled>
<svg class="svg-icon" viewBox="0 0 20 20"> <svg class="svg-icon" viewBox="0 0 20 20">
<path d="M11.3,5.5H8.7V1.4L1.9,6.5l6.8,5.1V7.5h2.6c1.8,0,3.2,1.4,3.2,3.2s-1.4,3.2-3.2,3.2H7.8v2h3.5c2.9,0,5.2-2.3,5.2-5.2S14.2,5.5,11.3,5.5z"/> <path d="M11.3,5.5H8.7V1.4L1.9,6.5l6.8,5.1V7.5h2.6c1.8,0,3.2,1.4,3.2,3.2s-1.4,3.2-3.2,3.2H7.8v2h3.5c2.9,0,5.2-2.3,5.2-5.2S14.2,5.5,11.3,5.5z"/>
</svg> </svg>
@ -203,7 +185,7 @@
</template> </template>
<template data-id="jumpToLine"> <template data-id="jumpToLine">
<span i18n="editGotoLine">: <input class="CodeMirror-jump-field" type="text"></span> <span i18n-text="editGotoLine">: <input class="CodeMirror-jump-field" type="text"></span>
</template> </template>
<template data-id="regexpTestPartial"> <template data-id="regexpTestPartial">
@ -211,15 +193,15 @@
</template> </template>
<template data-id="resizeGrip"> <template data-id="resizeGrip">
<div class="resize-grip" i18n="title:cm_resizeGripHint"></div> <div class="resize-grip" i18n-title="cm_resizeGripHint"></div>
</template> </template>
<template data-id="keymapHelp"> <template data-id="keymapHelp">
<table class="keymap-list"> <table class="keymap-list">
<thead> <thead>
<tr> <tr>
<th><input i18n="placeholder:helpKeyMapHotkey" type="search"></th> <th><input i18n-placeholder="helpKeyMapHotkey" type="search" class="can-close-on-esc"></th>
<th><input i18n="placeholder:helpKeyMapCommand" type="search"></th> <th><input i18n-placeholder="helpKeyMapCommand" type="search" class="can-close-on-esc" spellcheck="false"></th>
</tr> </tr>
</thead> </thead>
<tbody> <tbody>
@ -239,20 +221,18 @@
<link href="vendor/codemirror/addon/search/matchesonscrollbar.css" rel="stylesheet"> <link href="vendor/codemirror/addon/search/matchesonscrollbar.css" rel="stylesheet">
<link href="js/color/color-picker.css" rel="stylesheet"> <link href="js/color/color-picker.css" rel="stylesheet">
<link href="edit/codemirror-default.css" rel="stylesheet"> <link href="edit/codemirror-default.css" rel="stylesheet">
<link href="edit/edit.css" rel="stylesheet">
<link rel="stylesheet" href="edit/tab.css">
<link rel="stylesheet" href="edit/settings.css">
</head>
<template data-id="body"> <!-- https://crbug.com/1288447 --> <body id="stylus-edit">
<div id="header"> <div id="header">
<h1 id="heading" i18n="data-edit:editStyleHeading, data-add:addStyleTitle"> <h1 id="heading" i18n-data-edit="editStyleHeading" i18n-data-add="addStyleTitle"></h1>
<a class="usercss-only"
href="https://github.com/openstyles/stylus/wiki/Usercss"
i18n="title:externalUsercssDocument" target="_blank">
<svg class="svg-icon info"><use xlink:href="#svg-icon-help"/></svg>
</a>
</h1>
<section id="basic-info"> <section id="basic-info">
<div id="basic-info-name"> <div id="basic-info-name">
<input id="name" class="style-contributor" spellcheck="false"> <input id="name" class="style-contributor" spellcheck="false">
<a id="reset-name" i18n="title:customNameResetHint" tabindex="0" hidden> <a id="reset-name" i18n-title="customNameResetHint" tabindex="0" hidden>
<svg class="svg-icon" viewBox="0 0 20 20"> <svg class="svg-icon" viewBox="0 0 20 20">
<polygon points="16.2,5.5 14.5,3.8 10,8.3 5.5,3.8 3.8,5.5 8.3,10 3.8,14.5 <polygon points="16.2,5.5 14.5,3.8 10,8.3 5.5,3.8 3.8,5.5 8.3,10 3.8,14.5
5.5,16.2 10,11.7 14.5,16.2 16.2,14.5 11.7,10 "/> 5.5,16.2 10,11.7 14.5,16.2 16.2,14.5 11.7,10 "/>
@ -262,99 +242,87 @@
</div> </div>
<div id="basic-info-enabled"> <div id="basic-info-enabled">
<label id="enabled-label" <label id="enabled-label"
i18n="styleEnabledLabel, title:toggleStyle" i18n-text="styleEnabledLabel"
i18n-title="toggleStyle"
data-hotkey-tooltip="toggleStyle"> data-hotkey-tooltip="toggleStyle">
<input type="checkbox" id="enabled" class="style-contributor"> <input type="checkbox" id="enabled" class="style-contributor">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
<label id="preview-label" i18n="previewLabel, title:previewTooltip"> <label id="preview-label" i18n-text="previewLabel" i18n-title="previewTooltip">
<input type="checkbox" id="editor.livePreview"> <input type="checkbox" id="editor.livePreview">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
<label id="disableAll-label" i18n="data-on:disableAllStyles, data-off:disableAllStylesOff"> <span id="preview-errors" class="hidden">!</span>
<input id="disableAll" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label>
<span id="preview-errors" hidden>!</span>
</div> </div>
</section> </section>
<section id="actions"> <section id="actions">
<div class="buttons"> <div>
<div class="split-btn"> <button id="save-button" i18n-text="styleSaveLabel" data-hotkey-tooltip="save" disabled></button>
<button id="save-button" i18n="styleSaveLabel" data-hotkey-tooltip="save" disabled></button <button id="beautify" i18n-text="styleBeautify"></button>
><button class="split-btn-pedal usercss-only" i18n="menu-tpl:saveAsTemplate"></button> <a href="manage.html" tabindex="-1"><button id="cancel-button" i18n-text="styleCancelEditLabel"></button></a>
</div> </div>
<button id="beautify" i18n="styleBeautify"></button> <div id="mozilla-format-buttons" class="sectioned-only">
<button id="style-settings-btn" i18n="settings"></button> <button id="from-mozilla" i18n-text="importLabel"></button>
<button id="cancel-button" i18n="title:styleCancelEditLabel"></button> <button id="to-mozilla" i18n-text="exportLabel"></button>
</div>
<div id="mozilla-format-buttons" class="buttons sectioned-only">
<button id="from-mozilla" i18n="importLabel"></button>
<button id="to-mozilla" i18n="exportLabel"></button>
<a id="to-mozilla-help" class="svg-inline-wrapper" tabindex="0" <a id="to-mozilla-help" class="svg-inline-wrapper" tabindex="0"
i18n="title:styleMozillaFormatHeading"> i18n-title="styleMozillaFormatHeading">
<svg class="svg-icon info"><use xlink:href="#svg-icon-help"/></svg> <svg class="svg-icon info"><use xlink:href="#svg-icon-help"/></svg>
</a> </a>
</div> </div>
</section> </section>
<div id="details-wrapper"> <div id="details-wrapper">
<details id="options" data-pref="editor.options.expanded" class="ignore-pref-if-compact"> <details id="options" data-pref="editor.options.expanded" class="ignore-pref-if-compact">
<summary><h2 id="options-heading" i18n="editorSettings"></h2></summary> <summary><h2 id="options-heading" i18n-text="optionsHeading"></h2></summary>
<div id="options-wrapper"> <div id="options-wrapper">
<div class="options-column"> <div class="options-column">
<div class="option"> <div class="option">
<label id="lineWrapping-label" i18n="cm_lineWrapping"> <label id="lineWrapping-label" i18n-text="cm_lineWrapping">
<input id="editor.lineWrapping" type="checkbox"> <input id="editor.lineWrapping" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
</div> </div>
<div class="option"> <div class="option">
<label id="smartIndent-label" i18n="cm_smartIndent"> <label id="smartIndent-label" i18n-text="cm_smartIndent">
<input id="editor.smartIndent" type="checkbox"> <input id="editor.smartIndent" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
</div> </div>
<div class="option"> <div class="option">
<label id="indentWithTabs-label" i18n="cm_indentWithTabs"> <label id="indentWithTabs-label" i18n-text="cm_indentWithTabs">
<input id="editor.indentWithTabs" type="checkbox"> <input id="editor.indentWithTabs" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
</div> </div>
<div class="option"> <div class="option">
<label i18n="cm_autoCloseBrackets, title:cm_autoCloseBracketsTooltip"> <label i18n-text="cm_autoCloseBrackets" i18n-title="cm_autoCloseBracketsTooltip">
<input id="editor.autoCloseBrackets" type="checkbox"> <input id="editor.autoCloseBrackets" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
</div> </div>
<div class="option"> <div class="option">
<label i18n="cm_autocompleteOnTyping"> <label i18n-text="cm_autocompleteOnTyping">
<input id="editor.autocompleteOnTyping" type="checkbox"> <input id="editor.autocompleteOnTyping" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
</div> </div>
<div class="option"> <div class="option">
<label i18n="cm_selectByTokens, title:cm_selectByTokensTooltip"> <label i18n-text="cm_selectByTokens"
i18n-title="cm_selectByTokensTooltip">
<input id="editor.selectByTokens" type="checkbox"> <input id="editor.selectByTokens" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
</div> </div>
<div class="option sectioned-only">
<label i18n="cm_arrowKeysTraverse">
<input id="editor.arrowKeysTraverse" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label>
</div>
<div class="option"> <div class="option">
<label i18n="cm_colorpicker"> <label i18n-text="cm_colorpicker">
<input id="editor.colorpicker" type="checkbox"> <input id="editor.colorpicker" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
<a id="colorpicker-settings" class="svg-inline-wrapper" i18n="title:shortcutsNote" tabindex="0"> <a id="colorpicker-settings" class="svg-inline-wrapper" i18n-title="shortcutsNote" tabindex="0">
<svg class="svg-icon settings"><use xlink:href="#svg-icon-config"/></svg> <svg class="svg-icon settings"><use xlink:href="#svg-icon-config"/></svg>
</a> </a>
</div> </div>
<div class="option usercss-only"> <div class="option usercss-only">
<label i18n="appliesLineWidgetLabel, title:appliesLineWidgetWarning"> <label i18n-text="appliesLineWidgetLabel" i18n-title="appliesLineWidgetWarning">
<input id="editor.appliesToLineWidget" type="checkbox"> <input id="editor.appliesToLineWidget" type="checkbox">
<svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg> <svg class="svg-icon checked"><use xlink:href="#svg-icon-checked"/></svg>
</label> </label>
@ -362,11 +330,11 @@
</div> </div>
<div class="options-column"> <div class="options-column">
<div class="option aligned"> <div class="option aligned">
<label id="tabSize-label" for="editor.tabSize" i18n="cm_tabSize"></label> <label id="tabSize-label" for="editor.tabSize" i18n-text="cm_tabSize"></label>
<input id="editor.tabSize" type="number" min="0"> <input id="editor.tabSize" type="number" min="0">
</div> </div>
<div class="option aligned"> <div class="option aligned">
<label id="keyMap-label" for="editor.keyMap" i18n="cm_keyMap"></label> <label id="keyMap-label" for="editor.keyMap" i18n-text="cm_keyMap"></label>
<div class="select-resizer"> <div class="select-resizer">
<select id="editor.keyMap"></select> <select id="editor.keyMap"></select>
<svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg> <svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg>
@ -376,34 +344,34 @@
</a> </a>
</div> </div>
<div class="option aligned"> <div class="option aligned">
<label id="theme-label" for="editor.theme" i18n="cm_theme"></label> <label id="theme-label" for="editor.theme" i18n-text="cm_theme"></label>
<div class="select-resizer"> <div class="select-resizer">
<select id="editor.theme"></select> <select id="editor.theme"></select>
<svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg> <svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg>
</div> </div>
</div> </div>
<div class="option aligned"> <div class="option aligned">
<label id="highlight-label" for="editor.matchHighlight" i18n="cm_matchHighlight"></label> <label id="highlight-label" for="editor.matchHighlight" i18n-text="cm_matchHighlight"></label>
<div class="select-resizer"> <div class="select-resizer">
<select id="editor.matchHighlight"> <select id="editor.matchHighlight">
<option i18n="cm_matchHighlightToken" value="token"> <option i18n-text="cm_matchHighlightToken" value="token">
<option i18n="cm_matchHighlightSelection" value="selection"> <option i18n-text="cm_matchHighlightSelection" value="selection">
<option i18n="genericDisabledLabel" value=""> <option i18n-text="genericDisabledLabel" value="">
</select> </select>
<svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg> <svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg>
</div> </div>
</div> </div>
<div class="option aligned"> <div class="option aligned">
<label id="linter-label" for="editor.linter" i18n="cm_linter"></label> <label id="linter-label" for="editor.linter" i18n-text="cm_linter"></label>
<div class="select-resizer"> <div class="select-resizer">
<select id="editor.linter"> <select id="editor.linter">
<option value="csslint" selected>CSSLint</option> <option value="csslint" selected>CSSLint</option>
<option value="stylelint">Stylelint</option> <option value="stylelint">Stylelint</option>
<option value="" i18n="genericDisabledLabel"></option> <option value="" i18n-text="genericDisabledLabel"></option>
</select> </select>
<svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg> <svg class="svg-icon select-arrow"><use xlink:href="#svg-icon-select-arrow"/></svg>
</div> </div>
<a id="linter-settings" class="svg-inline-wrapper" i18n="title:linterConfigTooltip" tabindex="0"> <a id="linter-settings" class="svg-inline-wrapper" i18n-title="linterConfigTooltip" tabindex="0">
<svg class="svg-icon settings"><use xlink:href="#svg-icon-config"/></svg> <svg class="svg-icon settings"><use xlink:href="#svg-icon-config"/></svg>
</a> </a>
</div> </div>
@ -411,44 +379,92 @@
</div> </div>
</details> </details>
<details id="publish" data-pref="editor.publish.expanded" class="ignore-pref-if-compact"> <details id="publish" data-pref="editor.publish.expanded" class="ignore-pref-if-compact">
<summary><h2 i18n="publish"></h2></summary> <summary><h2 i18n-text="publish"></h2></summary>
<div> <div>
<a id="usw-url" href="https://userstyles.world" target="_blank">&nbsp;</a> <a id="usw-url" href="https://userstyles.world" target="_blank">&nbsp;</a>
<div id="usw-link-info"> <div id="usw-link-info">
<dl><dt i18n="styleName"></dt><dd data-usw="name"></dd></dl> <dl><dt i18n-text="styleName"></dt><dd data-usw="name"></dd></dl>
<dl><dt i18n="genericDescription"></dt><dd data-usw="description"></dd></dl> <dl><dt i18n-text="genericDescription"></dt><dd data-usw="description"></dd></dl>
</div> </div>
<div> <div>
<button id="usw-publish-style" <button id="usw-publish-style"
i18n="data-publish:publishStyle, data-push:publishPush"></button> i18n-data-publish="publishStyle"
<button id="usw-disconnect" i18n="optionsSyncDisconnect"></button> i18n-data-push="publishPush"></button>
<button id="usw-disconnect" i18n-text="optionsSyncDisconnect"></button>
<span id="usw-progress"></span> <span id="usw-progress"></span>
</div> </div>
</div> </div>
</details> </details>
<details id="sections-list" data-pref="editor.toc.expanded" class="ignore-pref-if-compact"> <details id="sections-list" data-pref="editor.toc.expanded" class="ignore-pref-if-compact">
<summary><h2 i18n="sections"></h2></summary> <summary><h2 i18n-text="sections"></h2></summary>
<ol id="toc"></ol> <ol id="toc"></ol>
</details> </details>
<details id="lint" data-pref="editor.lint.expanded" class="ignore-pref-if-compact" hidden> <details id="lint" data-pref="editor.lint.expanded" class="hidden-unless-compact ignore-pref-if-compact">
<summary> <summary>
<h2><span i18n="linterIssues"></span><span id="issue-count"></span> <h2 i18n-text="linterIssues">: <span id="issue-count"></span>
<a id="lint-help" class="svg-inline-wrapper intercepts-click" tabindex="0"> <a id="lint-help" class="svg-inline-wrapper intercepts-click" tabindex="0">
<svg class="svg-icon info"><use xlink:href="#svg-icon-help"/></svg> <svg class="svg-icon info"><use xlink:href="#svg-icon-help"/></svg>
</a> </a>
</h2> </h2>
</summary> </summary>
<div class="lint-scroll-container">
<div class="lint-report-container"></div> <div class="lint-report-container"></div>
</div>
</details> </details>
</div> </div>
<div id="header-resizer" i18n="title:headerResizerHint"></div>
<div id="footer" class="hidden"> <div id="footer" class="hidden">
<a href="https://github.com/openstyles/stylus/wiki/Usercss" <a href="https://github.com/openstyles/stylus/wiki/Usercss"
i18n="externalUsercssDocument" i18n-text="externalUsercssDocument"
target="_blank"></a> target="_blank"></a>
</div> </div>
</div> </div>
<section id="sections"></section> <div class="main tab-container">
<div class="tab-bar">
<div class="tab-bar-item active" i18n-text="editorCodeLabel"></div>
<div class="tab-bar-item" i18n-text="editorSettingLabel"></div>
</div>
<div class="tab-panel">
<section id="sections" class="active"></section>
<fieldset class="style-settings" disabled>
<!-- <label class="style-origin">
<span class="form-label" i18n-text="styleOriginLabel"></span>
<input id="styleOrigin" type="text">
</label> -->
<label class="form-group style-update-url">
<span class="form-label" i18n-text="styleUpdateUrlLabel"></span>
<input type="text">
</label>
<div class="form-group style-prefer-scheme radio-group">
<!-- FIXME: should we use a different message from install page? -->
<span class="form-label" i18n-text="installPreferSchemeLabel"></span>
<label class="radio-item">
<input type="radio" name="preferScheme" value="none">
<span class="radio-label" i18n-text="installPreferSchemeNone"></span>
</label>
<label class="radio-item">
<input type="radio" name="preferScheme" value="dark">
<span class="radio-label" i18n-text="installPreferSchemeDark"></span>
</label>
<label class="radio-item">
<input type="radio" name="preferScheme" value="light">
<span class="radio-label" i18n-text="installPreferSchemeLight"></span>
</label>
</div>
<label class="form-group style-include">
<span class="form-label" i18n-text="styleIncludeLabel"></span>
<textarea spellcheck="false" placeholder="*://site1.com/*&#10;*://site2.com/*"></textarea>
</label>
<label class="form-group style-exclude">
<span class="form-label" i18n-text="styleExcludeLabel"></span>
<textarea spellcheck="false" placeholder="*://site1.com/*&#10;*://site2.com/*"></textarea>
</label>
<!-- <label class="style-always-important">
<input type="checkbox">
<span class="form-label" i18n-text="styleAlwaysImportantLabel"></span>
</label> -->
</fieldset>
</div>
</div>
<div id="help-popup"> <div id="help-popup">
<div class="title"></div><svg id="sections-help" class="svg-icon dismiss"><use xlink:href="#svg-icon-close"/></svg> <div class="title"></div><svg id="sections-help" class="svg-icon dismiss"><use xlink:href="#svg-icon-close"/></svg>
<div class="contents"></div> <div class="contents"></div>
@ -460,10 +476,8 @@
<path d="M0 0v8h8v-2h-1v1h-6v-6h1v-1h-2zm4 0l1.5 1.5-2.5 2.5 1 1 2.5-2.5 1.5 1.5v-4h-4z"></path> <path d="M0 0v8h8v-2h-1v1h-6v-6h1v-1h-2zm4 0l1.5 1.5-2.5 2.5 1 1 2.5-2.5 1.5 1.5v-4h-4z"></path>
</symbol> </symbol>
<symbol id="svg-icon-help" viewBox="0 0 14 16" i18n="alt:helpAlt"> <symbol id="svg-icon-help" viewBox="0 0 14 16" i18n-alt="helpAlt">
<circle cx="7" cy="5" r="1"/> <path fill-rule="evenodd" d="M6.3 5.69a.942.942 0 0 1-.28-.7c0-.28.09-.52.28-.7.19-.18.42-.28.7-.28.28 0 .52.09.7.28.18.19.28.42.28.7 0 .28-.09.52-.28.7a1 1 0 0 1-.7.3c-.28 0-.52-.11-.7-.3zM8 7.99c-.02-.25-.11-.48-.31-.69-.2-.19-.42-.3-.69-.31H6c-.27.02-.48.13-.69.31-.2.2-.3.44-.31.69h1v3c.02.27.11.5.31.69.2.2.42.31.69.31h1c.27 0 .48-.11.69-.31.2-.19.3-.42.31-.69H8V7.98v.01zM7 2.3c-3.14 0-5.7 2.54-5.7 5.68 0 3.14 2.56 5.7 5.7 5.7s5.7-2.55 5.7-5.7c0-3.15-2.56-5.69-5.7-5.69v.01zM7 .98c3.86 0 7 3.14 7 7s-3.14 7-7 7-7-3.12-7-7 3.14-7 7-7z"></path>
<path d="M8,8c0-0.5-0.5-1-1-1H6C5.5,7,5,7.4,5,8h1v3c0,0.5,0.5,1,1,1h1c0.5,0,1-0.4,1-1H8V8z"/>
<path d="M7,1c3.9,0,7,3.1,7,7s-3.1,7-7,7s-7-3.1-7-7S3.1,1,7,1z M7,2.3C3.9,2.3,1.3,4.9,1.3,8s2.6,5.7,5.7,5.7s5.7-2.6,5.7-5.7S10.1,2.3,7,2.3C7,2.3,7,2.3,7,2.3z"/>
</symbol> </symbol>
<symbol id="svg-icon-close" viewBox="0 0 12 16"> <symbol id="svg-icon-close" viewBox="0 0 12 16">
@ -487,21 +501,14 @@
</symbol> </symbol>
<symbol id="svg-icon-plus" viewBox="0 0 8 8"> <symbol id="svg-icon-plus" viewBox="0 0 8 8">
<path d="M3 0v3h-3v2h3v3h2v-3h3v-2h-3v-3h-2z"/> <path fill-rule="evenodd" d="M3 0v3h-3v2h3v3h2v-3h3v-2h-3v-3h-2z"/>
</symbol> </symbol>
<symbol id="svg-icon-minus" viewBox="0 0 8 8"> <symbol id="svg-icon-minus" viewBox="0 0 8 8">
<path d="M0 3v2h8v-2h-8z"/> <path fill-rule="evenodd" d="M0 3v2h8v-2h-8z"/>
</symbol> </symbol>
</svg> </svg>
</template> <script src="edit/tab.js"></script>
<link href="edit/edit.css" rel="stylesheet">
<script src="js/dark-themer.js"></script> <!-- must be last in HEAD to avoid FOUC -->
</head>
<body id="stylus-edit">
<script src="edit/edit.js"></script>
</body> </body>
</html> </html>

View File

@ -12,7 +12,7 @@
const USO_VAR = 'uso-variable'; const USO_VAR = 'uso-variable';
const USO_VALID_VAR = 'variable-3 ' + USO_VAR; const USO_VALID_VAR = 'variable-3 ' + USO_VAR;
const USO_INVALID_VAR = 'error ' + USO_VAR; const USO_INVALID_VAR = 'error ' + USO_VAR;
const rxPROP = /^(prop(erty)?|variable-2|string-2)\b/; const rxPROP = /^(prop(erty)?|variable-2)\b/;
const rxVAR = /(^|[^-.\w\u0080-\uFFFF])var\(/iyu; const rxVAR = /(^|[^-.\w\u0080-\uFFFF])var\(/iyu;
const rxCONSUME = /([-\w]*\s*:\s?)?/yu; const rxCONSUME = /([-\w]*\s*:\s?)?/yu;
const cssMime = CodeMirror.mimeModes['text/css']; const cssMime = CodeMirror.mimeModes['text/css'];
@ -41,7 +41,6 @@
const {line, ch} = pos; const {line, ch} = pos;
const {styles, text} = cm.getLineHandle(line); const {styles, text} = cm.getLineHandle(line);
const {style, index} = cm.getStyleAtPos({styles, pos: ch}) || {}; const {style, index} = cm.getStyleAtPos({styles, pos: ch}) || {};
const isLessLang = cm.doc.mode.helperType === 'less';
const isStylusLang = cm.doc.mode.name === 'stylus'; const isStylusLang = cm.doc.mode.name === 'stylus';
const type = style && style.split(' ', 1)[0] || 'prop?'; const type = style && style.split(' ', 1)[0] || 'prop?';
if (!type || type === 'comment' || type === 'string') { if (!type || type === 'comment' || type === 'string') {
@ -87,7 +86,6 @@
'@supports', '@supports',
'@viewport', '@viewport',
]; ];
if (isLessLang) list = findAllCssVars(cm, left, '\\s*:').concat(list);
break; break;
case '#': // prevents autocomplete for #hex colors case '#': // prevents autocomplete for #hex colors
@ -144,16 +142,16 @@
leftLC = leftLC.replace(/^[^\w\s]\s*/, ''); leftLC = leftLC.replace(/^[^\w\s]\s*/, '');
} }
if (prop.startsWith('--')) prop = 'color'; // assuming 90% of variables are colors if (prop.startsWith('--')) prop = 'color'; // assuming 90% of variables are colors
if (!cssProps) await initCssProps(); if (!cssValues) cssValues = await linterMan.worker.getCssPropsValues();
list = [...new Set([...cssValues.all[prop] || [], ...cssValues.global])]; list = [...new Set([...cssValues.own[prop] || [], ...cssValues.global])];
end = prev + execAt(/(\s*[-a-z(]+)?/y, prev, text)[0].length; end = prev + execAt(/(\s*[-a-z(]+)?/y, prev, text)[0].length;
} }
} }
// properties and media features // properties and media features
if (!list && if (!list &&
/^(prop(erty|\?)|atom|error|tag)/.test(type) && /^(prop(erty|\?)|atom|error)/.test(type) &&
/^(block|atBlock_parens|maybeprop)/.test(getTokenState())) { /^(block|atBlock_parens|maybeprop)/.test(getTokenState())) {
if (!cssProps) await initCssProps(); if (!cssProps) initCssProps();
if (type === 'prop?') { if (type === 'prop?') {
prev += leftLC.length; prev += leftLC.length;
leftLC = ''; leftLC = '';
@ -176,9 +174,8 @@
}; };
} }
async function initCssProps() { function initCssProps() {
cssValues = await linterMan.worker.getCssPropsValues(); cssProps = addSuffix(cssMime.propertyKeywords);
cssProps = addSuffix(cssValues.all);
cssMedia = [].concat(...Object.entries(cssMime).map(getMediaKeys).filter(Boolean)).sort(); cssMedia = [].concat(...Object.entries(cssMime).map(getMediaKeys).filter(Boolean)).sort();
} }
@ -198,15 +195,13 @@
!style.startsWith(USO_VALID_VAR) && !style.startsWith(USO_INVALID_VAR); !style.startsWith(USO_VALID_VAR) && !style.startsWith(USO_INVALID_VAR);
} }
function findAllCssVars(cm, leftPart, rightPart = '') { function findAllCssVars(cm, leftPart) {
// simplified regex without CSS escapes // simplified regex without CSS escapes
const [, prefixed, named] = leftPart.match(/^(--|@)?(\S)?/);
const rx = new RegExp( const rx = new RegExp(
'(?:^|[\\s/;{])(' + '(?:^|[\\s/;{])(' +
(prefixed ? leftPart : '--') + (leftPart.startsWith('--') ? leftPart : '--') +
(named ? '' : '[a-zA-Z_\u0080-\uFFFF]') + (leftPart.length <= 2 ? '[a-zA-Z_\u0080-\uFFFF]' : '') +
'[-0-9a-zA-Z_\u0080-\uFFFF]*)' + '[-0-9a-zA-Z_\u0080-\uFFFF]*)',
rightPart,
'g'); 'g');
const list = new Set(); const list = new Set();
cm.eachLine(({text}) => { cm.eachLine(({text}) => {

View File

@ -1,44 +1,43 @@
/* global $$ $ $create messageBoxProxy setInputValue setupLivePrefs */// dom.js /* global $ $$ $create setupLivePrefs waitForSelector */// dom.js
/* global API */// msg.js /* global API */// msg.js
/* global CODEMIRROR_THEMES */ /* global CODEMIRROR_THEMES */
/* global CodeMirror */ /* global CodeMirror */
/* global MozDocMapper */// sections-util.js /* global MozDocMapper */// sections-util.js
/* global chromeSync */// storage-util.js
/* global initBeautifyButton */// beautify.js /* global initBeautifyButton */// beautify.js
/* global prefs */ /* global prefs */
/* global t */// localization.js /* global t */// localization.js
/* global FIREFOX getOwnTab sessionStore tryJSONparse tryURL */// toolbox.js /* global
FIREFOX
debounce
getOwnTab
sessionStore
tryJSONparse
tryURL
*/// toolbox.js
/* global EventEmitter */
'use strict'; 'use strict';
/** /**
* @type Editor * @type Editor
* @namespace Editor * @namespace Editor
*/ */
const editor = { const editor = Object.assign(EventEmitter(), {
style: null, style: null,
dirty: DirtyReporter(), dirty: DirtyReporter(),
isUsercss: false, isUsercss: false,
isWindowed: false, isWindowed: false,
livePreview: LivePreview(), lazyKeymaps: {
emacs: '/vendor/codemirror/keymap/emacs',
vim: '/vendor/codemirror/keymap/vim',
},
livePreview: null,
/** @type {'customName'|'name'} */ /** @type {'customName'|'name'} */
nameTarget: 'name', nameTarget: 'name',
previewDelay: 200, // Chrome devtools uses 200 previewDelay: 200, // Chrome devtools uses 200
saving: false,
scrollInfo: null, scrollInfo: null,
cancel: () => location.assign('/manage.html'), onStyleUpdated() {
document.documentElement.classList.toggle('is-new-style', !editor.style.id);
updateClass() {
$.rootCL.toggle('is-new-style', !editor.style.id);
},
updateTheme(name) {
if (!CODEMIRROR_THEMES[name]) {
name = 'default';
prefs.set('editor.theme', name);
}
$('#cm-theme').dataset.theme = name;
$('#cm-theme').textContent = CODEMIRROR_THEMES[name] || '';
}, },
updateTitle(isDirty = editor.dirty.isDirty()) { updateTitle(isDirty = editor.dirty.isDirty()) {
@ -49,24 +48,37 @@ const editor = {
customName || name || t('styleMissingName') customName || name || t('styleMissingName')
} - Stylus`; // the suffix enables external utilities to process our windows e.g. pin on top } - Stylus`; // the suffix enables external utilities to process our windows e.g. pin on top
}, },
}; });
//#region pre-init //#region pre-init
(() => { const baseInit = (() => {
const mqCompact = matchMedia('(max-width: 850px)'); const domReady = waitForSelector('#sections');
const toggleCompact = mq => $.rootCL.toggle('compact-layout', mq.matches);
mqCompact.on('change', toggleCompact); return {
toggleCompact(mqCompact); domReady,
Object.assign(editor, /** @namespace Editor */ { ready: Promise.all([
mqCompact, domReady,
styleReady: prefs.ready.then(loadStyle), loadStyle(),
}); prefs.ready.then(() =>
Promise.all([
loadTheme(),
loadKeymaps(),
])),
]),
};
/** Preloads vim/emacs keymap only if it's the active one, otherwise will load later */
function loadKeymaps() {
const km = prefs.get('editor.keyMap');
return /emacs/i.test(km) && require([editor.lazyKeymaps.emacs]) ||
/vim/i.test(km) && require([editor.lazyKeymaps.vim]);
}
async function loadStyle() { async function loadStyle() {
const params = new URLSearchParams(location.search); const params = new URLSearchParams(location.search);
let id = Number(params.get('id')); const id = Number(params.get('id'));
const style = id && await API.styles.get(id) || { const style = id && await API.styles.get(id) || {
id: id = null, // resetting the non-existent id
name: params.get('domain') || name: params.get('domain') ||
tryURL(params.get('url-prefix')).hostname || tryURL(params.get('url-prefix')).hostname ||
'', '',
@ -76,37 +88,90 @@ const editor = {
], ],
}; };
// switching the mode here to show the correct page ASAP, usually before DOMContentLoaded // switching the mode here to show the correct page ASAP, usually before DOMContentLoaded
const isUC = Boolean(style.usercssData || !id && prefs.get('newStyleAsUsercss')); editor.isUsercss = Boolean(style.usercssData || !style.id && prefs.get('newStyleAsUsercss'));
Object.assign(editor, /** @namespace Editor */ { editor.style = style;
style, editor.onStyleUpdated();
isUsercss: isUC,
template: isUC && !id && chromeSync.getLZValue(chromeSync.LZ_KEY.usercssTemplate), // promise
});
editor.updateClass();
editor.updateTheme(prefs.get('editor.theme'));
editor.updateTitle(false); editor.updateTitle(false);
$.rootCL.add(isUC ? 'usercss' : 'sectioned'); document.documentElement.classList.toggle('usercss', editor.isUsercss);
sessionStore.justEditedStyleId = id || ''; sessionStore.justEditedStyleId = style.id || '';
// no such style so let's clear the invalid URL parameters // no such style so let's clear the invalid URL parameters
if (!id) history.replaceState({}, '', location.pathname); if (!style.id) history.replaceState({}, '', location.pathname);
}
/** Preloads the theme so CodeMirror can use the correct metrics in its first render */
async function loadTheme() {
const theme = prefs.get('editor.theme');
if (!CODEMIRROR_THEMES.includes(theme)) {
prefs.set('editor.theme', 'default');
return;
}
if (theme !== 'default') {
const el = $('#cm-theme');
const el2 = await require([`/vendor/codemirror/theme/${theme}.css`]);
el2.id = el.id;
el.remove();
// FF containers take more time to load CSS
for (let retry = 0; !el2.sheet && ++retry <= 10;) {
await new Promise(requestAnimationFrame);
}
}
} }
})(); })();
//#endregion
//#region init layout/resize
baseInit.domReady.then(() => {
let headerHeight;
detectLayout(true);
window.on('resize', () => detectLayout());
function detectLayout(now) {
const compact = window.innerWidth <= 850;
if (compact) {
document.body.classList.add('compact-layout');
if (!editor.isUsercss) {
if (now) fixedHeader();
else debounce(fixedHeader, 250);
window.on('scroll', fixedHeader, {passive: true});
}
} else {
document.body.classList.remove('compact-layout', 'fixed-header');
window.off('scroll', fixedHeader);
}
for (const el of $$('details[data-pref]')) {
el.open = compact ? false : prefs.get(el.dataset.pref);
}
}
function fixedHeader() {
const headerFixed = $('.fixed-header');
if (!headerFixed) headerHeight = $('#header').clientHeight;
const scrollPoint = headerHeight - 43;
if (window.scrollY >= scrollPoint && !headerFixed) {
$('body').style.setProperty('--fixed-padding', ` ${headerHeight}px`);
$('body').classList.add('fixed-header');
} else if (window.scrollY < scrollPoint && headerFixed) {
$('body').classList.remove('fixed-header');
}
}
});
//#endregion //#endregion
//#region init header //#region init header
/* exported EditorHeader */ baseInit.ready.then(() => {
function EditorHeader() {
initBeautifyButton($('#beautify')); initBeautifyButton($('#beautify'));
initKeymapElement(); initKeymapElement();
initNameArea(); initNameArea();
initThemeElement(); initThemeElement();
setupLivePrefs(); setupLivePrefs();
window.on('load', () => { require(Object.values(editor.lazyKeymaps), () => {
initKeymapElement();
prefs.subscribe('editor.keyMap', showHotkeyInTooltip, {runNow: true}); prefs.subscribe('editor.keyMap', showHotkeyInTooltip, {runNow: true});
window.on('showHotkeyInTooltip', showHotkeyInTooltip); window.on('showHotkeyInTooltip', showHotkeyInTooltip);
}, {once: true}); });
function findKeyForCommand(command, map) { function findKeyForCommand(command, map) {
if (typeof map === 'string') map = CodeMirror.keyMap[map]; if (typeof map === 'string') map = CodeMirror.keyMap[map];
@ -132,12 +197,19 @@ function EditorHeader() {
nameEl.title = isCustomName ? t('customNameHint') : ''; nameEl.title = isCustomName ? t('customNameHint') : '';
nameEl.on('input', () => { nameEl.on('input', () => {
editor.updateName(true); editor.updateName(true);
resetEl.hidden = !editor.style.customName; resetEl.hidden = false;
}); });
resetEl.hidden = !editor.style.customName; resetEl.hidden = !editor.style.customName;
resetEl.onclick = () => { resetEl.onclick = () => {
editor.style.customName = null; // to delete it from db const {style} = editor;
setInputValue(nameEl, editor.style.name); nameEl.focus();
nameEl.select();
// trying to make it undoable via Ctrl-Z
if (!document.execCommand('insertText', false, style.name)) {
nameEl.value = style.name;
editor.updateName(true);
}
style.customName = null; // to delete it from db
resetEl.hidden = true; resetEl.hidden = true;
}; };
const enabledEl = $('#enabled'); const enabledEl = $('#enabled');
@ -147,7 +219,7 @@ function EditorHeader() {
function initThemeElement() { function initThemeElement() {
$('#editor.theme').append(...[ $('#editor.theme').append(...[
$create('option', {value: 'default'}, t('defaultTheme')), $create('option', {value: 'default'}, t('defaultTheme')),
...Object.keys(CODEMIRROR_THEMES).map(s => $create('option', s)), ...CODEMIRROR_THEMES.map(s => $create('option', s)),
]); ]);
// move the theme after built-in CSS so that its same-specificity selectors win // move the theme after built-in CSS so that its same-specificity selectors win
document.head.appendChild($('#cm-theme')); document.head.appendChild($('#cm-theme'));
@ -203,7 +275,7 @@ function EditorHeader() {
} }
} }
} }
} });
//#endregion //#endregion
//#region init windowed mode //#region init windowed mode
@ -220,17 +292,22 @@ function EditorHeader() {
} }
} }
getOwnTab().then(tab => { getOwnTab().then(async tab => {
ownTabId = tab.id; ownTabId = tab.id;
// use browser history back when 'back to manage' is clicked
if (sessionStore['manageStylesHistory' + ownTabId] === location.href) { if (sessionStore['manageStylesHistory' + ownTabId] === location.href) {
editor.cancel = () => history.back(); await baseInit.domReady;
$('#cancel-button').onclick = event => {
event.stopPropagation();
event.preventDefault();
history.back();
};
} }
}); });
async function initWindowedMode() { async function initWindowedMode() {
chrome.tabs.onAttached.addListener(onTabAttached); chrome.tabs.onAttached.addListener(onTabAttached);
// Chrome 96+ bug: the type is 'app' for a window that was restored via Ctrl-Shift-T const isSimple = (await browser.windows.getCurrent()).type === 'popup';
const isSimple = ['app', 'popup'].includes((await browser.windows.getCurrent()).type);
if (isSimple) require(['/edit/embedded-popup']); if (isSimple) require(['/edit/embedded-popup']);
editor.isWindowed = isSimple || ( editor.isWindowed = isSimple || (
history.length === 1 && history.length === 1 &&
@ -266,15 +343,9 @@ function EditorHeader() {
function DirtyReporter() { function DirtyReporter() {
const data = new Map(); const data = new Map();
const listeners = new Set(); const listeners = new Set();
const dataListeners = new Set();
const notifyChange = wasDirty => { const notifyChange = wasDirty => {
const isDirty = data.size > 0; if (wasDirty !== (data.size > 0)) {
const flipped = isDirty !== wasDirty; listeners.forEach(cb => cb());
if (flipped) {
listeners.forEach(cb => cb(isDirty));
}
if (flipped || isDirty) {
dataListeners.forEach(cb => cb(isDirty));
} }
}; };
/** @namespace DirtyReporter */ /** @namespace DirtyReporter */
@ -291,19 +362,17 @@ function DirtyReporter() {
saved.newValue = value; saved.newValue = value;
saved.type = 'modify'; saved.type = 'modify';
} }
} else {
return;
} }
notifyChange(wasDirty); notifyChange(wasDirty);
}, },
clear(...objs) { clear(obj) {
if (data.size && ( const wasDirty = data.size > 0;
objs.length if (obj === undefined) {
? objs.map(data.delete, data).includes(true) data.clear();
: (data.clear(), true) } else {
)) { data.delete(obj);
notifyChange(true);
} }
notifyChange(wasDirty);
}, },
has(key) { has(key) {
return data.has(key); return data.has(key);
@ -317,8 +386,6 @@ function DirtyReporter() {
if (!saved) { if (!saved) {
if (oldValue !== newValue) { if (oldValue !== newValue) {
data.set(obj, {type: 'modify', savedValue: oldValue, newValue}); data.set(obj, {type: 'modify', savedValue: oldValue, newValue});
} else {
return;
} }
} else if (saved.type === 'modify') { } else if (saved.type === 'modify') {
if (saved.savedValue === newValue) { if (saved.savedValue === newValue) {
@ -328,17 +395,12 @@ function DirtyReporter() {
} }
} else if (saved.type === 'add') { } else if (saved.type === 'add') {
saved.newValue = newValue; saved.newValue = newValue;
} else {
return;
} }
notifyChange(wasDirty); notifyChange(wasDirty);
}, },
onChange(cb, add = true) { onChange(cb, add = true) {
listeners[add ? 'add' : 'delete'](cb); listeners[add ? 'add' : 'delete'](cb);
}, },
onDataChange(cb, add = true) {
dataListeners[add ? 'add' : 'delete'](cb);
},
remove(obj, value) { remove(obj, value) {
const wasDirty = data.size > 0; const wasDirty = data.size > 0;
const saved = data.get(obj); const saved = data.get(obj);
@ -348,80 +410,10 @@ function DirtyReporter() {
data.delete(obj); data.delete(obj);
} else if (saved.type === 'modify') { } else if (saved.type === 'modify') {
saved.type = 'remove'; saved.type = 'remove';
} else {
return;
} }
notifyChange(wasDirty); notifyChange(wasDirty);
}, },
}; };
} }
function LivePreview() {
let el;
let data;
let port;
let preprocess;
let enabled = prefs.get('editor.livePreview');
prefs.subscribe('editor.livePreview', (key, value) => {
if (!value) {
if (port) {
port.disconnect();
port = null;
}
} else if (data && data.id && (data.enabled || editor.dirty.has('enabled'))) {
createPreviewer();
updatePreviewer(data);
}
enabled = value;
});
return {
/**
* @param {Function} [fn] - preprocessor
*/
init(fn) {
preprocess = fn;
},
update(newData) {
data = newData;
if (!port) {
if (!data.id || !data.enabled || !enabled) {
return;
}
createPreviewer();
}
updatePreviewer(data);
},
};
function createPreviewer() {
port = chrome.runtime.connect({name: 'livePreview'});
port.onDisconnect.addListener(err => {
throw err;
});
el = $('#preview-errors');
el.onclick = () => messageBoxProxy.alert(el.title, 'pre');
}
async function updatePreviewer(data) {
try {
port.postMessage(preprocess ? await preprocess(data) : data);
el.hidden = true;
} catch (err) {
if (Array.isArray(err)) {
err = err.map(e => e.message || e).join('\n');
} else if (err && err.index != null) {
// FIXME: this would fail if editors[0].getValue() !== data.sourceCode
const pos = editor.getEditors()[0].posFromIndex(err.index);
err.message = `${pos.line}:${pos.ch} ${err.message || err}`;
}
el.title = err.message || `${err}`;
el.hidden = false;
}
}
}
//#endregion //#endregion

View File

@ -65,7 +65,7 @@ function beautifyEditor(cm, options, ui) {
window.scrollTo(scrollX, scrollY); window.scrollTo(scrollX, scrollY);
cm.beautifyChange[cm.changeGeneration()] = true; cm.beautifyChange[cm.changeGeneration()] = true;
if (ui) { if (ui) {
$('button[role="close"]', helpPopup.div).disabled = false; $('#help-popup button[role="close"]').disabled = false;
} }
} }
} }
@ -82,13 +82,12 @@ function createBeautifyUI(scope, options) {
$createOption('}', 'newline_between_rules'), $createOption('}', 'newline_between_rules'),
$createLabeledCheckbox('preserve_newlines', 'styleBeautifyPreserveNewlines'), $createLabeledCheckbox('preserve_newlines', 'styleBeautifyPreserveNewlines'),
$createLabeledCheckbox('indent_conditional', 'styleBeautifyIndentConditional'), $createLabeledCheckbox('indent_conditional', 'styleBeautifyIndentConditional'),
editor.isUsercss && $createLabeledCheckbox('indent_mozdoc', '', '... @-moz-document'),
]), ]),
$create('p.beautify-hint', [ $create('p.beautify-hint', [
$create('span', t('styleBeautifyHint') + '\u00A0'), $create('span', t('styleBeautifyHint') + '\u00A0'),
createHotkeyInput('editor.beautify.hotkey', { createHotkeyInput('editor.beautify.hotkey', {
buttons: false, buttons: false,
onDone: () => moveFocus(helpPopup.div, 0), onDone: () => moveFocus($('#help-popup'), 0),
}), }),
]), ]),
$create('.buttons', [ $create('.buttons', [
@ -114,16 +113,16 @@ function createBeautifyUI(scope, options) {
}, },
}, t(scope.length === 1 ? 'undo' : 'undoGlobal')), }, t(scope.length === 1 ? 'undo' : 'undoGlobal')),
]), ]),
]), ]));
{
className: 'wide', $('#help-popup').className = 'wide';
});
$('.beautify-options').onchange = ({target}) => { $('.beautify-options').onchange = ({target}) => {
const value = target.type === 'checkbox' ? target.checked : target.selectedIndex > 0; const value = target.type === 'checkbox' ? target.checked : target.selectedIndex > 0;
const elLine = target.closest('[newline]');
if (elLine) elLine.setAttribute('newline', value);
prefs.set('editor.beautify', Object.assign(options, {[target.dataset.option]: value})); prefs.set('editor.beautify', Object.assign(options, {[target.dataset.option]: value}));
if (target.parentNode.hasAttribute('newline')) {
target.parentNode.setAttribute('newline', value.toString());
}
beautify(scope, false); beautify(scope, false);
}; };
@ -149,7 +148,7 @@ function createBeautifyUI(scope, options) {
); );
} }
function $createLabeledCheckbox(optionName, i18nKey, text) { function $createLabeledCheckbox(optionName, i18nKey) {
return ( return (
$create('label', {style: 'display: block; clear: both;'}, [ $create('label', {style: 'display: block; clear: both;'}, [
$create('input', { $create('input', {
@ -159,7 +158,7 @@ function createBeautifyUI(scope, options) {
}), }),
$create('SVG:svg.svg-icon.checked', $create('SVG:svg.svg-icon.checked',
$create('SVG:use', {'xlink:href': '#svg-icon-checked'})), $create('SVG:use', {'xlink:href': '#svg-icon-checked'})),
i18nKey ? t(i18nKey) : text, t(i18nKey),
]) ])
); );
} }

76
dist/edit/codemirror-default.css vendored Normal file
View File

@ -0,0 +1,76 @@
/* Built-in CodeMirror and addon customization */
.CodeMirror-hints {
z-index: 999;
}
.CodeMirror-hint:hover {
color: white;
background: #08f;
}
.CodeMirror {
border: solid #CCC 1px;
transition: box-shadow .1s;
}
#stylus#stylus .CodeMirror {
/* Using a specificity hack to override userstyles */
/* Not using the ring-color hack as it became ugly in new Chrome */
outline: none !important;
}
.CodeMirror-dialog {
animation: highlight 3s cubic-bezier(.18, .02, 0, .94);
}
.CodeMirror-search-field {
width: 10em;
}
.CodeMirror-jump-field {
width: 5em;
}
.CodeMirror-search-hint {
color: #888;
}
.CodeMirror-activeline .applies-to:before {
background-color: hsla(214, 100%, 90%, 0.15);
content: "";
top: 1em;
left: 0;
right: 0;
bottom: 1em;
position: absolute;
pointer-events: none;
}
.CodeMirror-activeline .applies-to ul {
z-index: 2;
}
.CodeMirror-foldgutter-open::after,
.CodeMirror-foldgutter-folded::after {
top: 5px;
width: 0;
height: 0;
content: "";
position: absolute;
border-style: solid;
opacity: .5;
left: 1px;
}
.CodeMirror-foldgutter-open::after {
border-width: 5px 3px 0 3px;
border-color: currentColor transparent transparent transparent;
}
.CodeMirror-foldgutter-folded::after {
margin-top: -2px;
margin-left: 1px;
border-width: 4px 0 4px 5px;
border-color: transparent transparent transparent currentColor;
}
.CodeMirror-linenumber {
cursor: pointer; /* for bookmarking */
}
/* Custom stuff we add to CodeMirror */
.cm-uso-variable {
font-weight: bold;
}
.gutter-bookmark {
background: linear-gradient(0deg, hsla(180, 100%, 30%, .75) 2px, hsla(180, 100%, 30%, .2) 2px);
}

View File

@ -1,6 +1,5 @@
/* global $ */// dom.js /* global $ */// dom.js
/* global CodeMirror */ /* global CodeMirror */
/* global UA */// toolbox.js
/* global editor */ /* global editor */
/* global prefs */ /* global prefs */
/* global t */// localization.js /* global t */// localization.js
@ -26,7 +25,7 @@
matchBrackets: true, matchBrackets: true,
hintOptions: {}, hintOptions: {},
lintReportDelay: prefs.get('editor.lintReportDelay'), lintReportDelay: prefs.get('editor.lintReportDelay'),
styleActiveLine: {nonEmpty: true}, styleActiveLine: true,
theme: prefs.get('editor.theme'), theme: prefs.get('editor.theme'),
keyMap: prefs.get('editor.keyMap'), keyMap: prefs.get('editor.keyMap'),
extraKeys: Object.assign(CodeMirror.defaults.extraKeys || {}, { extraKeys: Object.assign(CodeMirror.defaults.extraKeys || {}, {
@ -42,7 +41,7 @@
Object.assign(CodeMirror.defaults, defaults, prefs.get('editor.options')); Object.assign(CodeMirror.defaults, defaults, prefs.get('editor.options'));
// Adding hotkeys to some keymaps except 'basic' which is primitive by design // Adding hotkeys to some keymaps except 'basic' which is primitive by design
{ require(Object.values(typeof editor === 'object' && editor.lazyKeymaps || {}), () => {
const KM = CodeMirror.keyMap; const KM = CodeMirror.keyMap;
const extras = Object.values(CodeMirror.defaults.extraKeys); const extras = Object.values(CodeMirror.defaults.extraKeys);
if (!extras.includes('jumpToLine')) { if (!extras.includes('jumpToLine')) {
@ -63,7 +62,7 @@
if (!extras.includes('blockComment')) { if (!extras.includes('blockComment')) {
KM.sublime['Shift-Ctrl-/'] = 'commentSelection'; KM.sublime['Shift-Ctrl-/'] = 'commentSelection';
} }
if (UA.windows) { if (navigator.appVersion.includes('Windows')) {
// 'pcDefault' keymap on Windows should have F3/Shift-F3/Ctrl-R // 'pcDefault' keymap on Windows should have F3/Shift-F3/Ctrl-R
if (!extras.includes('findNext')) KM.pcDefault['F3'] = 'findNext'; if (!extras.includes('findNext')) KM.pcDefault['F3'] = 'findNext';
if (!extras.includes('findPrev')) KM.pcDefault['Shift-F3'] = 'findPrev'; if (!extras.includes('findPrev')) KM.pcDefault['Shift-F3'] = 'findPrev';
@ -90,7 +89,7 @@
} }
} }
} }
} });
Object.assign(CodeMirror.prototype, { Object.assign(CodeMirror.prototype, {
/** /**
@ -102,7 +101,6 @@
const m = this.doc.mode; const m = this.doc.mode;
if (force || (m.helperType ? m.helperType !== pp : m.name !== name)) { if (force || (m.helperType ? m.helperType !== pp : m.name !== name)) {
this.setOption('mode', name); this.setOption('mode', name);
this.doc.mode.lineComment = ''; // stylelint chokes on line comments a lot
} }
}, },
/** Superfast GC-friendly check that runs until the first non-space line */ /** Superfast GC-friendly check that runs until the first non-space line */

View File

@ -1,3 +1,4 @@
/* global $ */// dom.js
/* global CodeMirror */ /* global CodeMirror */
/* global editor */ /* global editor */
/* global prefs */ /* global prefs */
@ -31,7 +32,7 @@
globalSetOption(key, value) { globalSetOption(key, value) {
CodeMirror.defaults[key] = value; CodeMirror.defaults[key] = value;
if (cms.size > 4 && lazyOpt.names.includes(key)) { if (cms.size > 4 && lazyOpt && lazyOpt.names.includes(key)) {
lazyOpt.set(key, value); lazyOpt.set(key, value);
} else { } else {
cms.forEach(cm => cm.setOption(key, value)); cms.forEach(cm => cm.setOption(key, value));
@ -47,10 +48,8 @@
cm.lastActive = Date.now(); cm.lastActive = Date.now();
}; };
const onCmBlur = cm => { const onCmBlur = cm => {
setTimeout(() => {
/* Delaying to next tick to avoid double-processing of the currently processed keyboard event
* when it bubbles up from CodeMirror to `document` where the rerouter listens */
rerouteHotkeys.toggle(true); rerouteHotkeys.toggle(true);
setTimeout(() => {
const {wrapper} = cm.display; const {wrapper} = cm.display;
wrapper.classList.toggle('CodeMirror-active', wrapper.contains(document.activeElement)); wrapper.classList.toggle('CodeMirror-active', wrapper.contains(document.activeElement));
}); });
@ -67,7 +66,6 @@
k.slice('editor.'.length); k.slice('editor.'.length);
const prefKeys = prefs.knownKeys.filter(k => const prefKeys = prefs.knownKeys.filter(k =>
k !== 'editor.colorpicker' && // handled in colorpicker-helper.js k !== 'editor.colorpicker' && // handled in colorpicker-helper.js
k !== 'editor.arrowKeysTraverse' && // handled in sections-editor.js
prefToCmOpt(k) in CodeMirror.defaults); prefToCmOpt(k) in CodeMirror.defaults);
const {insertTab, insertSoftTab} = CodeMirror.commands; const {insertTab, insertSoftTab} = CodeMirror.commands;
@ -83,7 +81,6 @@
const opt = (showToken || value === 'selection') && { const opt = (showToken || value === 'selection') && {
showToken, showToken,
annotateScrollbar: true, annotateScrollbar: true,
delay: 0,
onUpdate: updateMatchHighlightCount, onUpdate: updateMatchHighlightCount,
}; };
cm.setOption('highlightSelectionMatches', opt || null); cm.setOption('highlightSelectionMatches', opt || null);
@ -97,13 +94,17 @@
} }
prefs.subscribe(prefKeys, (key, val) => { prefs.subscribe(prefKeys, (key, val) => {
if (key === 'editor.theme') editor.updateTheme(val); const name = prefToCmOpt(key);
cmFactory.globalSetOption(prefToCmOpt(key), val); if (name === 'theme') {
loadCmTheme(val);
} else {
cmFactory.globalSetOption(name, val);
}
}); });
// lazy propagation // lazy propagation
lazyOpt = { lazyOpt = window.IntersectionObserver && {
names: ['theme', 'lineWrapping'], names: ['theme', 'lineWrapping'],
set(key, value) { set(key, value) {
const {observer, queue} = lazyOpt; const {observer, queue} = lazyOpt;
@ -177,6 +178,25 @@
//#endregion //#endregion
//#region CM option handlers //#region CM option handlers
async function loadCmTheme(name) {
let el2;
const el = $('#cm-theme');
if (name === 'default') {
el.href = '';
} else {
const path = `/vendor/codemirror/theme/${name}.css`;
if (el.href !== location.origin + path) {
// avoid flicker: wait for the second stylesheet to load, then apply the theme
el2 = await require([path]);
}
}
cmFactory.globalSetOption('theme', name);
if (el2) {
el.remove();
el2.id = el.id;
}
}
function updateMatchHighlightCount(cm, state) { function updateMatchHighlightCount(cm, state) {
cm.display.wrapper.dataset.matchHighlightCount = state.matchesonscroll.matches.length; cm.display.wrapper.dataset.matchHighlightCount = state.matchesonscroll.matches.length;
} }

71
dist/edit/codemirror-themes.js vendored Normal file
View File

@ -0,0 +1,71 @@
/* Do not edit. This file is auto-generated by build-vendor.js */
'use strict';
/* exported CODEMIRROR_THEMES */
const CODEMIRROR_THEMES = [
'3024-day',
'3024-night',
'abbott',
'abcdef',
'ambiance',
'ambiance-mobile',
'ayu-dark',
'ayu-mirage',
'base16-dark',
'base16-light',
'bespin',
'blackboard',
'cobalt',
'colorforth',
'darcula',
'dracula',
'duotone-dark',
'duotone-light',
'eclipse',
'elegant',
'erlang-dark',
'gruvbox-dark',
'hopscotch',
'icecoder',
'idea',
'isotope',
'juejin',
'lesser-dark',
'liquibyte',
'lucario',
'material',
'material-darker',
'material-ocean',
'material-palenight',
'mbo',
'mdn-like',
'midnight',
'monokai',
'moxer',
'neat',
'neo',
'night',
'nord',
'oceanic-next',
'panda-syntax',
'paraiso-dark',
'paraiso-light',
'pastel-on-dark',
'railscasts',
'rubyblue',
'seti',
'shadowfox',
'solarized',
'ssms',
'the-matrix',
'tomorrow-night-bright',
'tomorrow-night-eighties',
'ttcn',
'twilight',
'vibrant-ink',
'xq-dark',
'xq-light',
'yeti',
'yonce',
'zenburn',
];

View File

@ -1,15 +1,15 @@
:root { :root {
--pad: 1rem; /* Edge padding for modals/blocks/whatnot. TODO: reuse it in more places */ --fixed-padding: unset;
--pad05: calc(0.5 * var(--pad));
--popup-button-width: 16px;
} }
body { body {
margin: 0;
height: 100vh; height: 100vh;
font: 12px arial,sans-serif;
} }
a { a {
color: var(--fg); color: #000;
transition: color .5s; transition: color .5s;
} }
a:hover { a:hover {
@ -34,7 +34,6 @@ a:hover {
} }
html.is-new-style #preview-label, html.is-new-style #preview-label,
html.is-new-style #style-settings-btn,
html.is-new-style #publish, html.is-new-style #publish,
.hidden { .hidden {
display: none !important; display: none !important;
@ -42,7 +41,7 @@ html.is-new-style #publish,
html.is-new-style #heading::after { html.is-new-style #heading::after {
content: attr(data-add); content: attr(data-add);
} }
html:not(.is-new-style) #heading::before { html:not(.is-new-style) #heading::after {
content: attr(data-edit); content: attr(data-edit);
} }
@ -54,8 +53,8 @@ html:not(.is-new-style) #heading::before {
top: 0; top: 0;
z-index: 1001; z-index: 1001;
border: none; border: none;
background: var(--bg); background: #fff;
box-shadow: 0 0 30px var(--fg); box-shadow: 0 0 30px #000;
} }
#popup-iframe:not([data-loaded]) { #popup-iframe:not([data-loaded]) {
opacity: 0; opacity: 0;
@ -71,12 +70,12 @@ html:not(.is-new-style) #heading::before {
#popup-button:hover { #popup-button:hover {
filter: drop-shadow(0 0 3px hsl(180, 70%, 50%)); filter: drop-shadow(0 0 3px hsl(180, 70%, 50%));
} }
.usercss:not(.compact-layout) #popup-button { .usercss body:not(.compact-layout) #popup-button {
right: 24px; right: 24px;
} }
/************ checkbox & select************/ /************ checkbox & select************/
.options-column > .option { .options-column > div[class="option"] {
margin-bottom: .25rem; margin-bottom: 4px;
} }
.options-column > .usercss-only { .options-column > .usercss-only {
@ -90,13 +89,14 @@ label {
/************ header ************/ /************ header ************/
#header { #header {
width: var(--header-width); width: 280px;
height: 100vh; height: 100vh;
overflow: hidden; overflow: auto;
position: fixed; position: fixed;
top: 0; top: 0;
padding-top: var(--pad); padding: 1rem;
box-shadow: 0 0 3rem -1.2rem #000; border-right: 1px dashed #AAA;
box-shadow: 0 0 3rem -1.2rem black;
box-sizing: border-box; box-sizing: border-box;
z-index: 10; z-index: 10;
display: flex; display: flex;
@ -105,10 +105,11 @@ label {
#header h1 { #header h1 {
margin-top: 0; margin-top: 0;
} }
#sections { .main {
padding-left: var(--header-width); padding-left: 280px;
height: 100%;
} }
.usercss #sections { #sections {
min-height: 0; min-height: 0;
height: 100%; height: 100%;
} }
@ -150,7 +151,6 @@ label {
margin-top: 2px; margin-top: 2px;
display: flex; display: flex;
align-items: center; align-items: center;
flex-wrap: wrap;
line-height: 16px; line-height: 16px;
} }
@ -175,7 +175,7 @@ label {
#preview-errors { #preview-errors {
background-color: red; background-color: red;
color: var(--bg); color: white;
padding: 0 6px; padding: 0 6px;
border-radius: 9px; border-radius: 9px;
margin-left: -.5em; margin-left: -.5em;
@ -195,6 +195,9 @@ label {
#colorpicker-settings.svg-inline-wrapper { #colorpicker-settings.svg-inline-wrapper {
margin: -2px 0 0 .1rem; margin: -2px 0 0 .1rem;
} }
.svg-inline-wrapper.applies-to-help {
margin: 0 0 0 .25rem;
}
.aligned .svg-inline-wrapper { .aligned .svg-inline-wrapper {
margin: -2px 0 0 .3rem; margin: -2px 0 0 .3rem;
} }
@ -208,75 +211,88 @@ label {
.svg-icon:hover, .svg-icon:hover,
.svg-icon.info, .svg-icon.info,
.svg-icon.settings { .svg-icon.settings {
fill: var(--c40); fill: #666;
} }
.svg-icon, .svg-icon,
.svg-icon.info:hover, .svg-icon.info:hover,
.svg-icon.settings:hover { .svg-icon.settings:hover {
fill: var(--fg); fill: #000;
} }
#options span .svg-icon { #options span .svg-icon {
margin-top: -3px; /* inline info and config icons */ margin-top: -3px; /* inline info and config icons */
} }
input:invalid {
background-color: rgba(255, 0, 0, 0.1);
color: darkred;
}
#enabled { #enabled {
margin-left: 0; margin-left: 0;
} }
/* collapsibles */ /* collapsibles */
#header details {
margin-right: var(--header-resizer-width);
}
#header details[open] {
overflow-y: auto;
margin-top: calc(1.5*var(--pad));
}
#header details[open] summary {
position: absolute;
margin-top: calc(-1.5*var(--pad));
}
#header summary { #header summary {
align-items: center; align-items: center;
margin-left: .25em; margin-left: -13px;
cursor: pointer; cursor: pointer;
white-space: nowrap; }
#header summary + * {
padding: .5rem 0;
} }
#header summary h2 { #header summary h2 {
display: inline-block; display: inline-block;
border-bottom: 1px dotted transparent; border-bottom: 1px dotted transparent;
margin: 0 0 0 -13px; margin-top: .1em;
margin-bottom: .1em;
margin-left: -13px;
padding-left: 13px; /* clicking directly on details-marker doesn't set pref so we cover it with h2 */ padding-left: 13px; /* clicking directly on details-marker doesn't set pref so we cover it with h2 */
max-width: calc(var(--header-width) - 2*var(--pad));
vertical-align: middle;
}
#header summary h2,
#header summary h2 > :first-child {
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
} }
#header summary:hover h2 { #header summary:hover h2 {
border-color: var(--c70); border-color: #bbb;
} }
#header summary svg { #header summary svg {
margin-top: -3px; margin-top: -3px;
} }
#header details > :not(summary) {
margin: 0 0 0 var(--pad);
padding: calc(var(--pad)/4) 0;
}
#details-wrapper { #details-wrapper {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
flex-grow: 1; flex-grow: 1;
overflow: hidden;
margin-top: var(--pad05);
} }
#actions .buttons {
#header details[open] + details[open] {
margin-top: .5rem;
}
#actions > * {
display: inline-flex; display: inline-flex;
flex-wrap: wrap; flex-wrap: wrap;
}
#mozilla-format-buttons {
display: flex;
flex-wrap: wrap;
align-items: center; align-items: center;
} }
#actions .buttons > * {
margin: 0 .25rem var(--pad05) 0; #actions > div > a {
height: min-content;
}
#actions button,
#actions > div > a {
margin: 0 .2rem .5rem 0;
}
#actions #cancel-button {
margin: 0;
}
#options:not([open]) + #lint h2 {
margin-top: 0;
}
#lint:not([open]) h2 {
margin-bottom: 0;
} }
#publish > div > * { #publish > div > * {
@ -378,12 +394,12 @@ label {
box-shadow: none !important; box-shadow: none !important;
pointer-events: auto !important; /* FF bug */ pointer-events: auto !important; /* FF bug */
} }
.sectioned .section { .section-editor .section {
margin: 0 0.7rem; margin: 0 0.7rem;
padding: 1rem; padding: 1rem;
} }
.sectioned .section:not(:first-child) { .section-editor .section:not(:first-child) {
border-top: 2px solid var(--c80); border-top: 2px solid hsl(0, 0%, 80%);
} }
.add-section:after { .add-section:after {
content: attr(short-text); content: attr(short-text);
@ -401,6 +417,10 @@ label {
.edit-actions button { .edit-actions button {
margin-right: .2rem; margin-right: .2rem;
} }
.dirty > label::before {
content: "*";
font-weight: bold;
}
#sections { #sections {
counter-reset: codebox; counter-reset: codebox;
} }
@ -418,19 +438,20 @@ label {
content: counter(codebox) ": " attr(data-text); content: counter(codebox) ": " attr(data-text);
margin-left: 0.25rem; margin-left: 0.25rem;
} }
.usercss .applies-to { .single-editor .applies-to {
border-width: 1px 0; border-width: 1px 0;
} }
.usercss .applies-to > label::before { .single-editor .applies-to > label::before {
content: attr(data-index) ":"; content: attr(data-index) ":";
margin-right: 0.25rem; margin-right: 0.25rem;
font-size: 12px;
} }
.code-label[data-text] { .code-label[data-text] {
font-weight: bold; font-weight: bold;
} }
#toc { #toc {
counter-reset: codelabel; counter-reset: codelabel;
margin: 0;
padding: .5rem 0;
} }
#toc li { #toc li {
white-space: nowrap; white-space: nowrap;
@ -457,7 +478,7 @@ label {
display: none; display: none;
} }
.section .CodeMirror { .section .CodeMirror {
margin-bottom: .5em; margin-bottom: .875rem;
box-sizing: border-box; box-sizing: border-box;
} }
/* deleted section */ /* deleted section */
@ -476,20 +497,19 @@ label {
.section.removed .CodeMirror { .section.removed .CodeMirror {
display: none; display: none;
} }
.move-section-down:after,
.move-section-up:after { .move-section-up:after {
content: ""; content: "";
display: block; display: block;
border: var(--side) solid transparent; border-style: solid;
--side: .4em; border-width: 0 .3em .5em .3em;
} border-color: transparent transparent currentColor transparent;
.move-section-up:after {
border-bottom: calc(var(--side) * 1.3) solid currentColor;
margin: -50% 0 0 0;
} }
.move-section-down:after { .move-section-down:after {
border-top: calc(var(--side) * 1.3) solid currentColor; content: "";
margin: 0 0 -60% 0; display: block;
border-style: solid;
border-width: .5em .3em 0 .3em;
border-color: currentColor transparent transparent transparent;
} }
/* code */ /* code */
.code { .code {
@ -513,6 +533,15 @@ label {
.resize-grip-enabled .CodeMirror-scrollbar-filler { .resize-grip-enabled .CodeMirror-scrollbar-filler {
bottom: 7px; /* make space for resize-grip */ bottom: 7px; /* make space for resize-grip */
} }
body:not(.find-open) .cm-matchhighlight,
body:not(.find-open) .CodeMirror-selection-highlight-scrollbar {
animation: fadein-match-highlighter 1s cubic-bezier(.97,.01,.42,.98);
animation-fill-mode: both;
}
body:not(.find-open) [data-match-highlight-count="1"] .cm-matchhighlight,
body:not(.find-open) [data-match-highlight-count="1"] .CodeMirror-selection-highlight-scrollbar {
animation: none;
}
@keyframes highlight { @keyframes highlight {
from { from {
background-color: #ff9; background-color: #ff9;
@ -529,6 +558,10 @@ label {
opacity: 1; opacity: 1;
} }
} }
@keyframes fadein-match-highlighter {
from { background-color: transparent; }
to { background-color: rgba(1, 151, 193, 0.1); }
}
.resize-grip { .resize-grip {
position: absolute; position: absolute;
display: block; display: block;
@ -568,21 +601,12 @@ label {
border-color: #f008; border-color: #f008;
} }
.applies-to label { .applies-to label {
font-size: 0; display: flex;
padding: 0; padding: 0;
line-height: var(--input-height); height: 22px;
} align-items: center;
.sectioned .applies-to label { margin: 0 .2em 0 0;
margin-left: -24px; white-space: nowrap;
position: absolute;
}
.compact-layout .applies-to label {
display: none;
}
.compact-layout .applies-to[data-all] label {
position: static;
display: inline;
margin: 0 6px 0 -4px;
} }
.applies-to ul { .applies-to ul {
flex: auto; flex: auto;
@ -595,17 +619,16 @@ label {
flex-wrap: wrap; flex-wrap: wrap;
list-style-type: none; list-style-type: none;
align-items: center; align-items: center;
margin-bottom: .5em;
} }
.applies-to li.applies-to-everything { .applies-to li.applies-to-everything {
align-items: unset; align-items: unset;
line-height: var(--input-height); line-height: 22px;
} }
.applies-to li > input { .applies-to li > input {
min-height: 1.4rem; min-height: 1.4rem;
} }
.applies-to .select-resizer { .applies-to li:not(.applies-to-everything) > * {
margin-right: .5em; margin: 0 .2rem .5rem 0;
} }
.applies-to li .add-applies-to:first-child { .applies-to li .add-applies-to:first-child {
margin-left: 1rem; margin-left: 1rem;
@ -624,8 +647,8 @@ label {
.add-applies-to, .add-applies-to,
.remove-applies-to { .remove-applies-to {
font-size: 0; font-size: 0;
height: var(--input-height); height: 22px;
width: var(--input-height); width: 22px;
display: inline-flex; display: inline-flex;
align-items: center; align-items: center;
justify-content: center; justify-content: center;
@ -634,14 +657,14 @@ label {
.add-applies-to .svg-icon, .add-applies-to .svg-icon,
.remove-applies-to .svg-icon { .remove-applies-to .svg-icon {
pointer-events: none; pointer-events: none;
fill: var(--c60); fill: hsl(0, 0%, 60%);
height: 12px; height: 12px;
width: 12px; width: 12px;
} }
.add-applies-to:hover .svg-icon, .add-applies-to:hover .svg-icon,
.remove-applies-to:hover .svg-icon { .remove-applies-to:hover .svg-icon {
pointer-events: none; pointer-events: none;
fill: var(--fg); fill: hsl(0, 0%, 0%);
} }
.test-regexp { .test-regexp {
display: none; display: none;
@ -673,7 +696,7 @@ label {
color: darkgreen; color: darkgreen;
} }
.regexp-report details[data-type="partial"] { .regexp-report details[data-type="partial"] {
color: var(--c65); color: darkgray;
} }
.regexp-report details[data-type="invalid"] { .regexp-report details[data-type="invalid"] {
color: maroon; color: maroon;
@ -706,24 +729,21 @@ label {
margin-right: .5em; margin-right: .5em;
} }
.regexp-report-note { .regexp-report-note {
color: var(--c60); color: #999;
position: absolute; position: absolute;
bottom: 0; bottom: 0;
hyphens: auto; hyphens: auto;
} }
/************ help popup ************/ /************ help popup ************/
#help-popup { #help-popup {
--pad-x: 1.5rem;
--pad-y: 1rem;
--pad-y2: calc(var(--pad-y) / 1.5);
top: 3rem; top: 3rem;
right: 3rem; right: 3rem;
max-width: 50vw; max-width: 50vw;
position: fixed; position: fixed;
display: none; display: none;
background-color: var(--bg); background-color: white;
box-shadow: 3px 3px 30px rgba(0, 0, 0, 0.5); box-shadow: 3px 3px 30px rgba(0, 0, 0, 0.5);
padding: var(--pad-y) var(--pad-x) 0; padding: 0.5rem;
z-index: 99; z-index: 99;
} }
#help-popup.big, #help-popup.big,
@ -732,7 +752,7 @@ label {
} }
#help-popup.big { #help-popup.big {
box-shadow: rgba(0, 0, 0, 0.45) 0px 0px 0px 100000px !important; box-shadow: rgba(0, 0, 0, 0.45) 0px 0px 0px 100000px !important;
left: calc(var(--header-width) - 3rem); left: calc(280px - 3rem);
} }
#help-popup.big .CodeMirror { #help-popup.big .CodeMirror {
min-height: 2rem; min-height: 2rem;
@ -740,20 +760,23 @@ label {
} }
#help-popup .title { #help-popup .title {
font-weight: bold; font-weight: bold;
background-color: rgba(128, 128, 128, .15); background-color: rgba(0,0,0,0.05);
margin: calc(-1 * var(--pad-y)) calc(-1 * var(--pad-x)) 0; margin: -0.5rem -0.5rem 0.5rem;
padding: var(--pad-y2) var(--pad-x); padding: .5rem 32px .5rem .5rem;
} }
#help-popup .contents { #help-popup .contents {
max-height: calc(100vh - 8rem); max-height: calc(100vh - 8rem);
overflow-y: auto; overflow-y: auto;
padding: var(--pad-y) 0; }
#help-popup .settings {
min-width: 500px;
min-height: 200px;
max-width: 48vw;
} }
#help-popup .dismiss { #help-popup .dismiss {
position: absolute; position: absolute;
right: 0; right: 4px;
top: 0; top: .5em;
padding: var(--pad-y2) .5em;
} }
#help-popup input[type="search"], #help-popup input[type="search"],
#help-popup .CodeMirror { #help-popup .CodeMirror {
@ -781,17 +804,17 @@ label {
} }
#help-popup .buttons { #help-popup .buttons {
display: flex; text-align: center;
justify-content: center; margin-top: .75em;
align-items: center;
margin: var(--pad-y2) 0 calc(var(--pad-y2) - var(--pad-y)) 0;
} }
.non-windows #help-popup .buttons { .non-windows #help-popup .buttons {
direction: rtl; direction: rtl;
justify-content: start; text-align: right;
} }
#help-popup button[name^="import"] { #help-popup button[name^="import"] {
line-height: 1.5rem; line-height: 1.5rem;
padding: 0 0.5rem;
margin: 0.5rem 0 0 0.5rem;
pointer-events: none; pointer-events: none;
opacity: 0.5; opacity: 0.5;
} }
@ -808,26 +831,48 @@ label {
#help-popup .rules p { #help-popup .rules p {
margin: .25em 0; margin: .25em 0;
} }
#help-popup .buttons > :nth-child(n + 2) { #help-popup .buttons button:nth-child(n + 2) {
margin: 0 0 0 .5em; margin-left: .5em;
}
.non-windows #help-popup .buttons > :nth-child(n + 2) {
margin: 0 .5em 0 0;
} }
/************ lint ************/ /************ lint ************/
#lint { #lint {
overflow-x: hidden; overflow: hidden;
margin-left: -1rem;
margin-right: -1rem;
padding: 0;
box-sizing: border-box;
display: flex;
flex-grow: 1;
position: relative;
}
#lint > summary {
position: relative;
margin-left: 0;
padding-left: 4px;
}
#lint[open]:not(.hidden-unless-compact) {
min-height: 102px;
} }
#lint summary h2 { #lint summary h2 {
display: inline-flex; text-indent: -2px;
}
#lint > .lint-scroll-container {
margin: 1rem 10px 0;
position: absolute;
top: 0;
bottom: 0;
left: 0;
right: 0;
overflow-y: auto;
overflow-x: hidden;
} }
#lint table { #lint table {
font-size: 100%;
border-spacing: 0; border-spacing: 0;
margin-bottom: 1rem;
line-height: 1.0; line-height: 1.0;
width: 100%; width: 100%;
font-size: 85%;
cursor: pointer;
} }
#lint tr td:last-child { #lint tr td:last-child {
width: 100%; width: 100%;
@ -835,28 +880,26 @@ label {
#lint td[role="line"] { #lint td[role="line"] {
padding-left: 0.25rem; padding-left: 0.25rem;
} }
#lint .report { #lint table:last-child {
display: flex; margin-bottom: 0;
} }
#lint .empty { #lint table.empty {
display: none; display: none;
} }
#lint .caption { #lint caption {
vertical-align: top; text-align: left;
line-height: 16px;
font-weight: bold; font-weight: bold;
padding-bottom: 6px;
} }
#lint .report:not(.empty) ~ :not(.empty) { #lint tbody {
border-top: 1px dotted rgba(128, 128, 128, .5); font-size: 85%;
margin-top: .25em; cursor: pointer;
padding-top: .25em;
} }
#lint tr:hover { #lint tr:hover {
background-color: hsla(180, 50%, 36%, .2); background-color: hsla(180, 50%, 36%, .2);
} }
#lint td { #lint td {
padding: 0; padding: 0;
line-height: 16px;
} }
#lint td[role="severity"] { #lint td[role="severity"] {
font-size: 0; font-size: 0;
@ -873,9 +916,6 @@ label {
text-align: left; text-align: left;
white-space: nowrap; white-space: nowrap;
} }
#issue-count::before {
content: ':\A0';
}
#message-box.center.lint-config #message-box-contents { #message-box.center.lint-config #message-box-contents {
text-align: left; text-align: left;
} }
@ -916,7 +956,7 @@ label {
} }
.beautify-options select { .beautify-options select {
border: none; border: none;
background: var(--c95); background: linear-gradient(90deg, rgba(0, 0, 0, .05) 18px, rgba(0, 0, 0, .02) 24px);
font-family: monospace; font-family: monospace;
font-weight: bold; font-weight: bold;
padding-left: 4px; padding-left: 4px;
@ -933,11 +973,16 @@ label {
box-shadow: none; box-shadow: none;
} }
.sectioned .usercss-only, html:not(.usercss) .usercss-only,
.usercss .sectioned-only { .usercss .sectioned-only {
display: none !important; /* hide during page init */ display: none !important; /* hide during page init */
} }
body:not(.compact-layout) .hidden-unless-compact,
body.linter-disabled .hidden-unless-compact {
display: none !important;
}
#options:not([open]) + #lint { #options:not([open]) + #lint {
margin-top: 0; margin-top: 0;
} }
@ -963,182 +1008,187 @@ label {
} }
#footer a { #footer a {
color: var(--c20); color: #333;
transition: color .5s; transition: color .5s;
} }
#footer a:hover { #footer a:hover {
color: var(--c40); color: #666;
} }
/************ line widget *************/ /************ line widget *************/
.CodeMirror-linewidget .applies-to { .CodeMirror-linewidget .applies-to {
margin: 1em 0; margin: 1em 0;
padding: .75rem calc(.25rem + var(--cm-bar-width, 0)) .25rem .75rem; padding: .75rem calc(.25rem + var(--cm-bar-width, 0)) .25rem .75rem;
font-family: var(--family);
} }
.CodeMirror-linewidget .applies-to li {
margin: 0;
}
.CodeMirror-linewidget .applies-to li[data-type="regexp"] .test-regexp { .CodeMirror-linewidget .applies-to li[data-type="regexp"] .test-regexp {
display: inline; display: inline;
} }
.CodeMirror-linewidget li.applies-to-everything { .CodeMirror-linewidget li.applies-to-everything {
margin-left: 0.2rem; margin-left: 0.2rem;
} }
/************ full-width only ************/
/* TODO: maybe move more rules here so we don't need to reset them in @media(max-width: 850px) */
@media (min-width: 851px) {
#header > :not(#details-wrapper):not(#header-resizer) {
margin-left: var(--pad);
margin-right: var(--pad);
}
#publish[open],
#header details:not([open]) {
flex: 0 0 auto;
}
#header details[open]:not(:last-child) {
margin-bottom: var(--pad05);
}
#header details:last-child {
padding-bottom: var(--pad);
}
}
/************ reponsive layouts ************/ /************ reponsive layouts ************/
@media(max-width: 850px) { @media(max-width: 850px) {
.sectioned body { body {
height: 100%;
}
.usercss body {
display: flex; display: flex;
flex-direction: column; flex-direction: column;
} }
.usercss #header,
#header:not(.sticky) {
position: static;
}
#header:not(.sticky) {
display: block;
}
#header { #header {
flex: 0 0 auto; flex: 0 1 auto;
height: unset; height: unset;
width: 100%; width: unset;
overflow: visible; position: inherit;
background: var(--bg);
border-right: none; border-right: none;
border-bottom: 1px dashed var(--c65); border-bottom: 1px dashed #AAA;
padding: var(--pad05) var(--pad05) 0; padding: .5rem 1rem .5rem .5rem;
} }
#header.sticky { .fixed-header {
flex-direction: row; --fixed-height: 40px;
box-shadow: 0 0 3rem -.75rem; padding-top: var(--fixed-padding);
}
.fixed-header #header {
min-height: var(--fixed-height);
position: fixed;
top: 0;
left: 0;
right: 0;
padding: 0;
background-color: #fff;
}
.fixed-header #header > *:not(#details-wrapper),
.fixed-header #options {
display: none !important;
}
.fixed-header #details-wrapper {
padding-top: calc((var(--fixed-height) - 1.2rem) / 2); /* 1.2 is the normal line height */
}
#header summary + *,
#lint > .lint-scroll-container {
margin-left: 1rem;
padding: .25rem 0 .5rem;
}
#actions {
display: flex;
flex-wrap: wrap;
white-space: nowrap;
margin: 0;
box-sizing: border-box;
}
#header input[type="checkbox"] {
vertical-align: middle;
}
#header details {
margin: 0;
} }
#header.sticky #basic-info-name,
#header.sticky #basic-info-enabled > :not(#preview-errors),
#header.sticky #mozilla-format-buttons,
#header.sticky .buttons > button,
#header.sticky .split-btn-pedal,
#heading, #heading,
h2 { h2 {
display: none; display: none;
} }
#preview-errors {
margin: -2px .75em 0 -.25em !important;
}
#header.sticky #basic-info {
margin: 0;
}
#header.sticky #preview-errors {
position: absolute;
top: var(--pad05);
right: var(--pad05);
margin: .25em 0 0 !important;
}
.popup-window #header.sticky #preview-errors {
right: calc(var(--pad05) + 24px);
}
.popup-window #details-wrapper {
margin-right: var(--popup-button-width);
}
#basic-info { #basic-info {
margin: 0 var(--popup-button-width) var(--pad05) 0; margin-bottom: .5rem;
box-sizing: border-box; box-sizing: border-box;
display: flex; display: flex;
flex-wrap: wrap; flex-wrap: wrap;
} }
#basic-info #name,
#basic-info > *:first-child { #basic-info > *:first-child {
flex-grow: 1; flex-grow: 1;
} }
#basic-info > *:not(:last-child) { #basic-info > *:not(:last-child) {
margin-right: 0.8rem; margin-right: 0.8rem;
} }
#header details > :not(summary) { #basic-info #name {
margin-left: var(--pad05); flex-grow: 1;
} }
#header h2 { #options-wrapper {
font-size: 14px; display: flex;
} flex-wrap: wrap;
#actions { box-sizing: border-box;
display: inline-block;
} }
#details-wrapper { #details-wrapper {
display: inline-flex;
vertical-align: top;
max-width: 100%;
flex-direction: row; flex-direction: row;
margin: .25em 0 var(--pad05); flex-wrap: wrap;
} }
#details-wrapper details[open] { #options[open] {
margin-top: 0; width: 100%;
z-index: 1000;
} }
#details-wrapper details[open]:hover, #sections-list[open] {
#details-wrapper details[open]:focus-within { max-height: 102px;
z-index: 1001;
} }
#details-wrapper details[open] > summary { #sections-list[open] #toc {
position: static; max-height: 60px;
overflow-y: auto;
}
#header details:not(#options) {
max-width: 50%;
}
.options-column {
flex-grow: 1;
padding-right: .5rem;
box-sizing: border-box;
}
.options-column > .usercss-only {
margin-bottom: 0;
}
#options-wrapper .options-column:nth-child(2) {
margin-top: 0; margin-top: 0;
} }
#details-wrapper details[open] > summary + * { #options:not([open]),
position: absolute;
overflow: hidden auto;
max-height: var(--max-height, 10vh);
max-width: calc(100vw - 2 * var(--pad));
background: var(--bg);
box-shadow: 0 6px 20px rgba(0, 0, 0, .3);
padding: var(--pad);
margin-top: var(--pad05);
}
@media (max-height: 500px) {
#details-wrapper {
--max-height: 50px;
}
}
#sections-list[open] > #toc {
left: 0;
margin-left: auto;
margin-right: auto;
width: min-content;
}
#sections-list[open] > #toc,
#lint[open] > .lint-report-container {
max-width: 50vw;
right: 0;
}
#options[open],
#publish[open],
#lint:not([open]) { #lint:not([open]) {
flex: 0 0 auto; overflow: initial;
overflow-x: hidden;
} }
#publish dd { #options:not([open]) + #lint:not([open]) {
white-space: normal; margin-top: 0;
max-height: 1.25em;
} }
#sections { #lint summary {
position: static;
margin-bottom: 0;
}
#header summary {
margin-left: 0;
padding-left: 4px;
}
#header summary h2 {
margin: 0;
padding: 0;
}
.option label {
margin: 0;
}
#options [type="number"] {
text-align: left; /* workaround the column flow bug in webkit */
padding-left: 0.2rem;
}
#options #tabSize-label {
position: relative;
top: 0.2rem;
}
#lint > .lint-scroll-container {
padding-top: 0;
margin-right: 0;
}
#lint {
padding: 0;
margin: .5rem 0 0;
}
#lint:not([open]) + #footer {
margin: .25em 0 -1em .25em;
}
.main {
height: unset !important;
padding-left: 0; padding-left: 0;
display: flex;
flex-direction: column;
flex: 1;
}
.tab-bar {
margin-top: var(--fixed-height);
} }
#sections > :not(.single-editor) { #sections > :not(.single-editor) {
margin: 0 .5rem; margin: 0 .5rem;
@ -1148,6 +1198,10 @@ label {
overflow: hidden; overflow: hidden;
flex: 1; flex: 1;
} }
.usercss #options:not([open]) ~ #lint.hidden ~ #footer,
.usercss #lint:not([open]) + #footer {
margin-top: -.25em;
}
#help-popup.big[style="display: block;"], #help-popup.big[style="display: block;"],
#help-popup[style="display: block;"] { #help-popup[style="display: block;"] {
width: max-content; width: max-content;

View File

@ -1,34 +1,37 @@
/* global $$ $ $create */// dom.js /* global $ $create messageBoxProxy waitForSheet */// dom.js
/* global API msg */// msg.js /* global msg API */// msg.js
/* global CodeMirror */ /* global CodeMirror */
/* global SectionsEditor */ /* global SectionsEditor */
/* global SourceEditor */ /* global SourceEditor */
/* global baseInit */
/* global clipString createHotkeyInput helpPopup */// util.js /* global clipString createHotkeyInput helpPopup */// util.js
/* global closeCurrentTab deepEqual mapObj sessionStore tryJSONparse */// toolbox.js /* global closeCurrentTab deepEqual sessionStore tryJSONparse */// toolbox.js
/* global cmFactory */ /* global cmFactory */
/* global editor EditorHeader */// base.js /* global editor */
/* global linterMan */ /* global linterMan */
/* global prefs */ /* global prefs */
/* global t */// localization.js /* global t */// localization.js
/* global StyleSettings */// settings.js
'use strict'; 'use strict';
//#region init //#region init
document.body.appendChild(t.template.body); baseInit.ready.then(async () => {
await waitForSheet();
EditorMethods(); (editor.isUsercss ? SourceEditor : SectionsEditor)();
editor.styleReady.then(async () => { StyleSettings(editor);
EditorHeader(); await editor.ready;
dispatchEvent(new Event('domReady')); editor.ready = true;
await (editor.isUsercss ? SourceEditor : SectionsEditor)();
editor.dirty.onChange(editor.updateDirty); editor.dirty.onChange(editor.updateDirty);
prefs.subscribe('editor.linter', () => linterMan.run());
prefs.subscribe('editor.linter', (key, value) => {
document.body.classList.toggle('linter-disabled', value === '');
linterMan.run();
});
// enabling after init to prevent flash of validation failure on an empty name // enabling after init to prevent flash of validation failure on an empty name
$('#name').required = !editor.isUsercss; $('#name').required = !editor.isUsercss;
$('#save-button').onclick = editor.save; $('#save-button').onclick = editor.save;
$('#cancel-button').onclick = editor.cancel;
const elSec = $('#sections-list'); const elSec = $('#sections-list');
// editor.toc.expanded pref isn't saved in compact-layout so prefs.subscribe won't work // editor.toc.expanded pref isn't saved in compact-layout so prefs.subscribe won't work
@ -45,65 +48,36 @@ editor.styleReady.then(async () => {
require(['/edit/linter-dialogs'], () => linterMan.showLintConfig()); require(['/edit/linter-dialogs'], () => linterMan.showLintConfig());
$('#lint-help').onclick = () => $('#lint-help').onclick = () =>
require(['/edit/linter-dialogs'], () => linterMan.showLintHelp()); require(['/edit/linter-dialogs'], () => linterMan.showLintHelp());
$('#style-settings-btn').onclick = () => require([
'/edit/settings.css',
'/edit/settings', /* global StyleSettings */
], () => StyleSettings());
require([ require([
'/edit/autocomplete', '/edit/autocomplete',
'/edit/drafts',
'/edit/global-search', '/edit/global-search',
]); ]);
}); });
editor.styleReady.then(async () => {
// Set up mini-header on scroll
const {isUsercss} = editor;
const el = $create({
style: `
top: 0;
height: 1px;
position: absolute;
visibility: hidden;
`.replace(/;/g, '!important;'),
});
const scroller = isUsercss ? $('.CodeMirror-scroll') : document.body;
const xoRoot = isUsercss ? scroller : undefined;
const xo = new IntersectionObserver(onScrolled, {root: xoRoot});
scroller.appendChild(el);
onCompactToggled(editor.mqCompact);
editor.mqCompact.on('change', onCompactToggled);
/** @param {MediaQueryList} mq */
function onCompactToggled(mq) {
for (const el of $$('details[data-pref]')) {
el.open = mq.matches ? false : prefs.get(el.dataset.pref);
}
if (mq.matches) {
xo.observe(el);
} else {
xo.disconnect();
}
}
/** @param {IntersectionObserverEntry[]} entries */
function onScrolled(entries) {
const h = $('#header');
const sticky = !entries.pop().isIntersecting;
if (!isUsercss) scroller.style.paddingTop = sticky ? h.offsetHeight + 'px' : '';
h.classList.toggle('sticky', sticky);
}
});
//#endregion //#endregion
//#region events //#region events
const IGNORE_UPDATE_REASONS = [
'editPreview',
'editPreviewEnd',
'editSave',
// https://github.com/openstyles/stylus/issues/807 is closed without fix
// 'config,
];
msg.onExtension(request => { msg.onExtension(request => {
const {style} = request; const {style} = request;
switch (request.method) { switch (request.method) {
case 'styleUpdated': case 'styleUpdated':
if (editor.style.id === style.id) { if (editor.style.id === style.id && !IGNORE_UPDATE_REASONS.includes(request.reason)) {
handleExternalUpdate(request); if (request.reason === 'toggle') {
editor.emit('styleToggled', request.style);
} else {
API.styles.get(request.style.id)
.then(style => {
editor.emit('styleChange', style, request.reason);
});
}
} }
break; break;
case 'styleDeleted': case 'styleDeleted':
@ -111,47 +85,17 @@ msg.onExtension(request => {
closeCurrentTab(); closeCurrentTab();
} }
break; break;
case 'editDeleteText':
document.execCommand('delete');
break;
} }
}); });
async function handleExternalUpdate({style, reason}) {
if (reason === 'editPreview' ||
reason === 'editPreviewEnd') {
return;
}
if (reason === 'editSave' && editor.saving) {
editor.saving = false;
return;
}
if (reason === 'toggle') {
if (editor.dirty.isDirty()) {
editor.toggleStyle(style.enabled);
} else {
Object.assign(editor.style, style);
}
editor.updateMeta();
editor.updateLivePreview();
return;
}
style = await API.styles.get(style.id);
if (reason === 'config') {
delete style.sourceCode;
delete style.sections;
delete style.name;
delete style.enabled;
Object.assign(editor.style, style);
} else {
await editor.replaceStyle(style);
}
window.dispatchEvent(new Event('styleSettings'));
}
window.on('beforeunload', e => { window.on('beforeunload', e => {
let pos; let pos;
if (editor.isWindowed && if (editor.isWindowed &&
document.visibilityState === 'visible' && document.visibilityState === 'visible' &&
prefs.get('openEditInWindow') && prefs.get('openEditInWindow') &&
screenX !== -32000 && // Chrome uses this value for minimized windows
( // only if not maximized ( // only if not maximized
screenX > 0 || outerWidth < screen.availWidth || screenX > 0 || outerWidth < screen.availWidth ||
screenY > 0 || outerHeight < screen.availHeight || screenY > 0 || outerHeight < screen.availHeight ||
@ -168,7 +112,16 @@ window.on('beforeunload', e => {
prefs.set('windowPosition', pos); prefs.set('windowPosition', pos);
} }
sessionStore.windowPos = JSON.stringify(pos || {}); sessionStore.windowPos = JSON.stringify(pos || {});
sessionStore['editorScrollInfo' + editor.style.id] = JSON.stringify(editor.makeScrollInfo()); sessionStore['editorScrollInfo' + editor.style.id] = JSON.stringify({
scrollY: window.scrollY,
cms: editor.getEditors().map(cm => /** @namespace EditorScrollInfo */({
bookmarks: (cm.state.sublimeBookmarks || []).map(b => b.find()),
focus: cm.hasFocus(),
height: cm.display.wrapper.style.height.replace('100vh', ''),
parentHeight: cm.display.wrapper.parentElement.offsetHeight,
sel: cm.isClean() && [cm.doc.sel.ranges, cm.doc.sel.primIndex],
})),
});
const activeElement = document.activeElement; const activeElement = document.activeElement;
if (activeElement) { if (activeElement) {
// blurring triggers 'change' or 'input' event if needed // blurring triggers 'change' or 'input' event if needed
@ -185,7 +138,7 @@ window.on('beforeunload', e => {
//#endregion //#endregion
//#region editor methods //#region editor methods
function EditorMethods() { (() => {
const toc = []; const toc = [];
const {dirty} = editor; const {dirty} = editor;
let {style} = editor; let {style} = editor;
@ -207,35 +160,15 @@ function EditorMethods() {
applyScrollInfo(cm, si = (editor.scrollInfo.cms || [])[0]) { applyScrollInfo(cm, si = (editor.scrollInfo.cms || [])[0]) {
if (si && si.sel) { if (si && si.sel) {
const bmOpts = {sublimeBookmark: true, clearWhenEmpty: false}; // copied from sublime.js const bmOpts = {sublimeBookmark: true, clearWhenEmpty: false}; // copied from sublime.js
cm.operation(() => {
cm.setSelections(...si.sel, {scroll: false}); cm.setSelections(...si.sel, {scroll: false});
cm.scrollIntoView(cm.getCursor(), si.parentHeight / 2);
cm.state.sublimeBookmarks = si.bookmarks.map(b => cm.markText(b.from, b.to, bmOpts)); cm.state.sublimeBookmarks = si.bookmarks.map(b => cm.markText(b.from, b.to, bmOpts));
Object.assign(cm.display.scroller, si.scroll); // for source editor });
Object.assign(cm.doc, si.scroll); // for sectioned editor
} }
}, },
makeScrollInfo() { toggleStyle(enabled = style.enabled) {
return {
scrollY: window.scrollY,
cms: editor.getEditors().map(cm => /** @namespace EditorScrollInfo */({
bookmarks: (cm.state.sublimeBookmarks || []).map(b => b.find()),
focus: cm.hasFocus(),
height: cm.display.wrapper.style.height.replace('100vh', ''),
parentHeight: cm.display.wrapper.parentElement.offsetHeight,
scroll: mapObj(cm.doc, null, ['scrollLeft', 'scrollTop']),
sel: [cm.doc.sel.ranges, cm.doc.sel.primIndex],
})),
};
},
async save() {
if (dirty.isDirty()) {
editor.saving = true;
await editor.saveImpl();
}
},
toggleStyle(enabled = !style.enabled) {
$('#enabled').checked = enabled; $('#enabled').checked = enabled;
editor.updateEnabledness(enabled); editor.updateEnabledness(enabled);
}, },
@ -303,18 +236,79 @@ function EditorMethods() {
el.classList.add(cls); el.classList.add(cls);
} }
}, },
useSavedStyle(newStyle) {
if (style.id !== newStyle.id) {
history.replaceState({}, '', `?id=${newStyle.id}`);
}
sessionStore.justEditedStyleId = newStyle.id;
Object.assign(style, newStyle);
editor.updateClass();
editor.updateMeta();
},
}); });
} })();
//#endregion
//#region editor livePreview
editor.livePreview = (() => {
let data;
let port;
let preprocess;
let enabled = prefs.get('editor.livePreview');
prefs.subscribe('editor.livePreview', (key, value) => {
if (!value) {
if (port) {
port.disconnect();
port = null;
}
} else if (data && data.id && (data.enabled || editor.dirty.has('enabled'))) {
createPreviewer();
updatePreviewer(data);
}
enabled = value;
});
return {
/**
* @param {Function} [fn] - preprocessor
*/
init(fn) {
preprocess = fn;
},
update(newData) {
data = newData;
if (!port) {
if (!data.id || !data.enabled || !enabled) {
return;
}
createPreviewer();
}
updatePreviewer(data);
},
};
function createPreviewer() {
port = chrome.runtime.connect({name: 'livePreview'});
port.onDisconnect.addListener(err => {
throw err;
});
}
async function updatePreviewer(data) {
const errorContainer = $('#preview-errors');
try {
port.postMessage(preprocess ? await preprocess(data) : data);
errorContainer.classList.add('hidden');
} catch (err) {
if (Array.isArray(err)) {
err = err.join('\n');
} else if (err && err.index != null) {
// FIXME: this would fail if editors[0].getValue() !== data.sourceCode
const pos = editor.getEditors()[0].posFromIndex(err.index);
err.message = `${pos.line}:${pos.ch} ${err.message || err}`;
}
errorContainer.classList.remove('hidden');
errorContainer.onclick = () => {
messageBoxProxy.alert(err.message || `${err}`, 'pre');
};
}
}
})();
//#endregion //#endregion
//#region colorpickerHelper //#region colorpickerHelper
@ -372,6 +366,8 @@ function EditorMethods() {
cmFactory.globalSetOption('colorpicker', defaults.colorpicker); cmFactory.globalSetOption('colorpicker', defaults.colorpicker);
}, {runNow: true}); }, {runNow: true});
await baseInit.domReady;
$('#colorpicker-settings').onclick = function (event) { $('#colorpicker-settings').onclick = function (event) {
event.preventDefault(); event.preventDefault();
const input = createHotkeyInput('editor.colorpicker.hotkey', {onDone: () => helpPopup.close()}); const input = createHotkeyInput('editor.colorpicker.hotkey', {onDone: () => helpPopup.close()});

View File

@ -2,7 +2,9 @@
'use strict'; 'use strict';
(() => { (() => {
let sugarss = false; const hasCurlyBraceError = warning =>
warning.text === 'Unnecessary curly bracket (CssSyntaxError)';
let sugarssFallback;
/** @namespace EditorWorker */ /** @namespace EditorWorker */
createWorkerApi({ createWorkerApi({
@ -26,7 +28,6 @@
// moving vendor-prefixed props to the end // moving vendor-prefixed props to the end
const cmp = (a, b) => a[0] === '-' && b[0] !== '-' ? 1 : a < b ? -1 : a > b; const cmp = (a, b) => a[0] === '-' && b[0] !== '-' ? 1 : a < b ? -1 : a > b;
for (const [k, v] of Object.entries(Properties)) { for (const [k, v] of Object.entries(Properties)) {
res[k] = false;
if (typeof v === 'string') { if (typeof v === 'string') {
let last = ''; let last = '';
const uniq = []; const uniq = [];
@ -43,7 +44,7 @@
if (uniq.length) res[k] = uniq; if (uniq.length) res[k] = uniq;
} }
} }
return {all: res, global: GlobalKeywords}; return {own: res, global: GlobalKeywords};
}, },
getRules(linter) { getRules(linter) {
@ -65,32 +66,23 @@
async stylelint(opts) { async stylelint(opts) {
require(['/vendor/stylelint-bundle/stylelint-bundle.min']); /* global stylelint */ require(['/vendor/stylelint-bundle/stylelint-bundle.min']); /* global stylelint */
// Stylus-lang allows a trailing ";" but sugarss doesn't, so we monkeypatch it
stylelint.SugarSSParser.prototype.checkSemicolon = tt => {
while (tt.length && tt[tt.length - 1][0] === ';') tt.pop();
};
for (const pass of opts.mode === 'stylus' ? [sugarss, !sugarss] : [-1]) {
/* We try sugarss (for indented stylus-lang), then css mode, switching them on failure,
* so that the succeeding syntax will be used next time first. */
opts.config.customSyntax = !pass ? 'sugarss' : '';
try { try {
const res = await stylelint.createLinter(opts)._lintSource(opts); let res;
if (pass !== -1) sugarss = pass; let pass = 0;
return collectStylelintResults(res, opts); /* sugarss is used for stylus-lang by default,
but it fails on normal css syntax so we retry in css mode. */
const isSugarSS = opts.syntax === 'sugarss';
if (sugarssFallback && isSugarSS) opts.syntax = sugarssFallback;
while (
++pass <= 2 &&
(res = (await stylelint.lint(opts)).results[0]) &&
isSugarSS && res.warnings.some(hasCurlyBraceError)
) sugarssFallback = opts.syntax = 'css';
delete res._postcssResult; // huge and unused
return res;
} catch (e) { } catch (e) {
const fatal = pass === -1 || delete e.postcssNode; // huge, unused, non-transferable
!pass && !/^CssSyntaxError:.+?Unnecessary curly bracket/.test(e) || throw e;
pass && !/^CssSyntaxError:.+?Unknown word[\s\S]*?\.decl\s/.test(`${e}${e.stack}`);
if (fatal) {
return [{
from: {line: e.line - 1, ch: e.column - 1},
to: {line: e.line - 1, ch: e.column - 1},
message: e.reason,
severity: 'error',
rule: e.name,
}];
}
}
} }
}, },
}); });
@ -145,32 +137,4 @@
return options; return options;
}, },
}; };
function collectStylelintResults({messages}, {mode}) {
/* We hide nonfatal "//" warnings since we lint with sugarss without applying @preprocessor.
* We can't easily pre-remove "//" comments which may be inside strings, comments, url(), etc.
* And even if we did, it'd be wrong to hide potential bugs in stylus-lang like #1460 */
const isLess = mode === 'text/x-less';
const slashCommentAllowed = isLess || mode === 'stylus';
const res = [];
for (const m of messages) {
if (/deprecation|invalidOption/.test(m.stylelintType)) {
continue;
}
const {rule} = m;
const msg = m.text.replace(/^Unexpected\s+/, '').replace(` (${rule})`, '');
if (slashCommentAllowed && msg.includes('"//"') ||
isLess && /^unknown at-rule "@[-\w]+:"/.test(msg) /* LESS variables */) {
continue;
}
res.push({
from: {line: m.line - 1, ch: m.column - 1},
to: {line: m.endLine - 1, ch: m.endColumn - 1},
message: msg[0].toUpperCase() + msg.slice(1),
severity: m.severity,
rule,
});
}
return res;
}
})(); })();

View File

@ -1,5 +1,6 @@
/* global $ $create $remove getEventKeyName */// dom.js /* global $ $create $remove getEventKeyName */// dom.js
/* global CodeMirror */ /* global CodeMirror */
/* global baseInit */// base.js
/* global prefs */ /* global prefs */
/* global t */// localization.js /* global t */// localization.js
'use strict'; 'use strict';
@ -19,13 +20,12 @@
title: t('optionsCustomizePopup') + '\n' + POPUP_HOTKEY, title: t('optionsCustomizePopup') + '\n' + POPUP_HOTKEY,
onclick: embedPopup, onclick: embedPopup,
}); });
$.root.appendChild(btn); document.documentElement.appendChild(btn);
$.rootCL.add('popup-window'); baseInit.domReady.then(() => {
window.on('domReady', () => {
document.body.appendChild(btn); document.body.appendChild(btn);
// Adding a dummy command to show in keymap help popup // Adding a dummy command to show in keymap help popup
CodeMirror.defaults.extraKeys[POPUP_HOTKEY] = 'openStylusPopup'; CodeMirror.defaults.extraKeys[POPUP_HOTKEY] = 'openStylusPopup';
}, {once: true}); });
prefs.subscribe('iconset', (_, val) => { prefs.subscribe('iconset', (_, val) => {
const prefix = `images/icon/${val ? 'light/' : ''}`; const prefix = `images/icon/${val ? 'light/' : ''}`;
@ -60,9 +60,14 @@
const body = pw.document.body; const body = pw.document.body;
pw.on('keydown', removePopupOnEsc); pw.on('keydown', removePopupOnEsc);
pw.close = removePopup; pw.close = removePopup;
if (pw.IntersectionObserver) {
new pw.IntersectionObserver(onIntersect).observe(body.appendChild( new pw.IntersectionObserver(onIntersect).observe(body.appendChild(
$create('div', {style: {height: '1px', marginTop: '-1px'}}) $create('div', {style: {height: '1px', marginTop: '-1px'}})
)); ));
} else {
frame.dataset.loaded = '';
frame.height = body.scrollHeight;
}
new pw.MutationObserver(onMutation).observe(body, { new pw.MutationObserver(onMutation).observe(body, {
attributes: true, attributes: true,
attributeFilter: ['style'], attributeFilter: ['style'],

View File

@ -1,4 +1,4 @@
/* global $ $$ $create $remove focusAccessibility setInputValue toggleDataset */// dom.js /* global $ $$ $create $remove focusAccessibility toggleDataset */// dom.js
/* global CodeMirror */ /* global CodeMirror */
/* global chromeLocal */// storage-util.js /* global chromeLocal */// storage-util.js
/* global colorMimicry */ /* global colorMimicry */
@ -54,7 +54,7 @@
undoHistory: [], undoHistory: [],
searchInApplies: !editor.isUsercss, searchInApplies: !document.documentElement.classList.contains('usercss'),
}; };
//endregion //endregion
@ -588,7 +588,7 @@
input: colorMimicry($('input:not(:disabled)'), {bg: 'backgroundColor'}), input: colorMimicry($('input:not(:disabled)'), {bg: 'backgroundColor'}),
icon: colorMimicry($$('svg.info')[1], {fill: 'fill'}), icon: colorMimicry($$('svg.info')[1], {fill: 'fill'}),
}; };
$.root.appendChild( document.documentElement.appendChild(
$(DIALOG_STYLE_SELECTOR) || $(DIALOG_STYLE_SELECTOR) ||
$create('style' + DIALOG_STYLE_SELECTOR) $create('style' + DIALOG_STYLE_SELECTOR)
).textContent = ` ).textContent = `
@ -607,10 +607,10 @@
} }
#search-replace-dialog[data-type="replace"] button:hover svg, #search-replace-dialog[data-type="replace"] button:hover svg,
#search-replace-dialog svg:hover { #search-replace-dialog svg:hover {
fill: var(--cmin); fill: inherit;
} }
#search-replace-dialog [data-action="case"]:hover { #search-replace-dialog [data-action="case"]:hover {
color: var(--cmin); color: inherit;
} }
#search-replace-dialog [data-action="clear"] { #search-replace-dialog [data-action="clear"] {
background-color: ${colors.input.bg.replace(/[^,]+$/, '') + '.75)'}; background-color: ${colors.input.bg.replace(/[^,]+$/, '') + '.75)'};
@ -930,5 +930,18 @@
}))); })));
} }
function setInputValue(input, value) {
input.focus();
input.select();
// using execCommand to add to the input's undo history
document.execCommand(value ? 'insertText' : 'delete', false, value);
// some versions of Firefox ignore execCommand
if (input.value !== value) {
input.value = value;
input.dispatchEvent(new Event('input', {bubbles: true}));
}
}
//endregion //endregion
})(); })();

View File

@ -202,7 +202,35 @@ linterMan.DEFAULTS = {
getConfig: config => ({ getConfig: config => ({
rules: Object.assign({}, DEFAULTS.stylelint.rules, config && config.rules), rules: Object.assign({}, DEFAULTS.stylelint.rules, config && config.rules),
}), }),
lint: (code, config, mode) => worker.stylelint({code, config, mode}), async lint(code, config, mode) {
const isLess = mode === 'text/x-less';
const isStylus = mode === 'stylus';
const syntax = isLess ? 'less' : isStylus ? 'sugarss' : 'css';
const raw = await worker.stylelint({code, config, syntax});
if (!raw) {
return [];
}
// Hiding the errors about "//" comments as we're preprocessing only when saving/applying
// and we can't just pre-remove the comments since "//" may be inside a string token
const slashCommentAllowed = isLess || isStylus;
const res = [];
for (const w of raw.warnings) {
const msg = w.text.match(/^(?:Unexpected\s+)?(.*?)\s*\([^()]+\)$|$/)[1] || w.text;
if (!slashCommentAllowed || !(
w.rule === 'no-invalid-double-slash-comments' ||
w.rule === 'property-no-unknown' && msg.includes('"//"')
)) {
res.push({
from: {line: w.line - 1, ch: w.column - 1},
to: {line: w.line - 1, ch: w.column},
message: msg.slice(0, 1).toUpperCase() + msg.slice(1),
severity: w.severity,
rule: w.rule,
});
}
}
return res;
},
}, },
}; };
@ -287,7 +315,7 @@ linterMan.DEFAULTS = {
function updateCount() { function updateCount() {
const issueCount = Array.from(tables.values()) const issueCount = Array.from(tables.values())
.reduce((sum, table) => sum + table.trs.length, 0); .reduce((sum, table) => sum + table.trs.length, 0);
$('#lint').hidden = !issueCount; $('#lint').classList.toggle('hidden-unless-compact', issueCount === 0);
$('#issue-count').textContent = issueCount; $('#issue-count').textContent = issueCount;
} }
@ -303,20 +331,19 @@ linterMan.DEFAULTS = {
} }
function createTable(cm) { function createTable(cm) {
const caption = $create('.caption'); const caption = $create('caption');
const table = $create('table'); const tbody = $create('tbody');
const report = $create('.report', [caption, table]); const table = $create('table', [caption, tbody]);
const trs = []; const trs = [];
return { return {
element: report, element: table,
trs, trs,
updateAnnotations, updateAnnotations,
updateCaption, updateCaption,
}; };
function updateCaption() { function updateCaption() {
const t = editor.getEditorTitle(cm); caption.textContent = editor.getEditorTitle(cm);
Object.assign(caption, typeof t == 'string' ? {textContent: t} : t);
} }
function updateAnnotations(lines) { function updateAnnotations(lines) {
@ -328,20 +355,20 @@ linterMan.DEFAULTS = {
} else { } else {
tr = createTr(); tr = createTr();
trs.push(tr); trs.push(tr);
table.appendChild(tr.element); tbody.append(tr.element);
} }
tr.update(anno); tr.update(anno);
i++; i++;
} }
if (i === 0) { if (i === 0) {
trs.length = 0; trs.length = 0;
table.textContent = ''; tbody.textContent = '';
} else { } else {
while (trs.length > i) { while (trs.length > i) {
trs.pop().element.remove(); trs.pop().element.remove();
} }
} }
report.classList.toggle('empty', !trs.length); table.classList.toggle('empty', trs.length === 0);
function *getAnnotations() { function *getAnnotations() {
for (const line of lines.filter(Boolean)) { for (const line of lines.filter(Boolean)) {

View File

@ -4,7 +4,9 @@
/* global colorMimicry */ /* global colorMimicry */
/* global editor */ /* global editor */
/* global msg */ /* global msg */
/* global prefs */
/* global t */// localization.js /* global t */// localization.js
/* global tryCatch */// toolbox.js
'use strict'; 'use strict';
/* exported MozSectionWidget */ /* exported MozSectionWidget */
@ -38,12 +40,11 @@ function MozSectionWidget(cm, finder = MozSectionFinder(cm)) {
}; };
function init() { function init() {
const hint = {title: t('appliesHelp')};
enabled = true; enabled = true;
TPL = { TPL = {
container: container:
$create('div' + C_CONTAINER, [ $create('div' + C_CONTAINER, [
$create(C_LABEL, hint, t('appliesLabel')), $create(C_LABEL, t('appliesLabel')),
$create('ul' + C_LIST), $create('ul' + C_LIST),
]), ]),
listItem: listItem:
@ -52,9 +53,8 @@ function MozSectionWidget(cm, finder = MozSectionFinder(cm)) {
$create('li.applies-to-everything', t('appliesToEverything')), $create('li.applies-to-everything', t('appliesToEverything')),
}; };
Object.assign($(C_TYPE, TPL.listItem), hint);
$(C_VALUE, TPL.listItem).after( $(C_VALUE, TPL.listItem).after(
$create('button.test-regexp', t('genericTest'))); $create('button.test-regexp', t('styleRegexpTestButton')));
CLICK_ROUTE = { CLICK_ROUTE = {
'.test-regexp': showRegExpTester, '.test-regexp': showRegExpTester,
@ -156,7 +156,6 @@ function MozSectionWidget(cm, finder = MozSectionFinder(cm)) {
} }
if (msg.style || msg.styles || if (msg.style || msg.styles ||
msg.prefs && 'disableAll' in msg.prefs || msg.prefs && 'disableAll' in msg.prefs ||
msg.method === 'colorScheme' ||
msg.method === 'styleDeleted') { msg.method === 'styleDeleted') {
requestAnimationFrame(updateWidgetStyle); requestAnimationFrame(updateWidgetStyle);
} }
@ -164,6 +163,11 @@ function MozSectionWidget(cm, finder = MozSectionFinder(cm)) {
function updateWidgetStyle() { function updateWidgetStyle() {
funcHeight = 0; funcHeight = 0;
if (prefs.get('editor.theme') !== 'default' &&
!tryCatch(() => $('#cm-theme').sheet.cssRules)) {
requestAnimationFrame(updateWidgetStyle);
return;
}
const MIN_LUMA = .05; const MIN_LUMA = .05;
const MIN_LUMA_DIFF = .4; const MIN_LUMA_DIFF = .4;
const color = { const color = {
@ -199,6 +203,7 @@ function MozSectionWidget(cm, finder = MozSectionFinder(cm)) {
color: ${fore}; color: ${fore};
} }
${C_CONTAINER} input, ${C_CONTAINER} input,
${C_CONTAINER} button,
${C_CONTAINER} select { ${C_CONTAINER} select {
background: rgba(255, 255, 255, ${ background: rgba(255, 255, 255, ${
Math.max(MIN_LUMA, Math.pow(Math.max(0, color.gutter.bgLuma - MIN_LUMA * 2), 2)).toFixed(2) Math.max(MIN_LUMA, Math.pow(Math.max(0, color.gutter.bgLuma - MIN_LUMA * 2), 2)).toFixed(2)
@ -212,7 +217,7 @@ function MozSectionWidget(cm, finder = MozSectionFinder(cm)) {
transition: none; transition: none;
} }
`; `;
$.root.appendChild(actualStyle); document.documentElement.appendChild(actualStyle);
} }
/** /**
@ -269,14 +274,13 @@ function MozSectionWidget(cm, finder = MozSectionFinder(cm)) {
let widget = old && old.widget; let widget = old && old.widget;
const height = Math.round(funcHeight * (sec.funcs.length || 1)) || undefined; const height = Math.round(funcHeight * (sec.funcs.length || 1)) || undefined;
const node = renderContainer(sec, widget); const node = renderContainer(sec, widget);
if (widget && widget.line.lineNo() === sec.start.line) { if (widget) {
widget.node = node; widget.node = node;
if (height && height !== widget.height) { if (height && height !== widget.height) {
widget.height = height; widget.height = height;
widget.changed(); widget.changed();
} }
} else { } else {
if (widget) widget.clear();
widget = cm.addLineWidget(sec.start.line, node, { widget = cm.addLineWidget(sec.start.line, node, {
coverGutter: true, coverGutter: true,
noHScroll: true, noHScroll: true,

View File

@ -5,6 +5,7 @@
'use strict'; 'use strict';
const regexpTester = (() => { const regexpTester = (() => {
const GET_FAVICON_URL = 'https://www.google.com/s2/favicons?domain=';
const OWN_ICON = chrome.runtime.getManifest().icons['16']; const OWN_ICON = chrome.runtime.getManifest().icons['16'];
const cachedRegexps = new Map(); const cachedRegexps = new Map();
let currentRegexps = []; let currentRegexps = [];
@ -92,7 +93,7 @@ const regexpTester = (() => {
for (const [url, match] of urls.entries()) { for (const [url, match] of urls.entries()) {
const faviconUrl = url.startsWith(URLS.ownOrigin) const faviconUrl = url.startsWith(URLS.ownOrigin)
? OWN_ICON ? OWN_ICON
: URLS.favicon(new URL(url).hostname); : GET_FAVICON_URL + new URL(url).hostname;
const icon = $create('img', {src: faviconUrl}); const icon = $create('img', {src: faviconUrl});
if (match.text.length === url.length) { if (match.text.length === url.length) {
full.push($create('a', {tabIndex: 0}, [ full.push($create('a', {tabIndex: 0}, [

View File

@ -1,4 +1,4 @@
/* global $ toggleDataset */// dom.js /* global $ */// dom.js
/* global MozDocMapper trimCommentLabel */// util.js /* global MozDocMapper trimCommentLabel */// util.js
/* global cmFactory */ /* global cmFactory */
/* global debounce tryRegExp */// toolbox.js /* global debounce tryRegExp */// toolbox.js
@ -23,7 +23,7 @@ function createSection(originalSection, genId, si) {
const elLabel = $('.code-label', el); const elLabel = $('.code-label', el);
const cm = cmFactory.create(wrapper => { const cm = cmFactory.create(wrapper => {
// making it tall during initial load so IntersectionObserver sees only one adjacent CM // making it tall during initial load so IntersectionObserver sees only one adjacent CM
if (editor.loading) { if (editor.ready !== true) {
wrapper.style.height = si ? si.height : '100vh'; wrapper.style.height = si ? si.height : '100vh';
} }
elLabel.after(wrapper); elLabel.after(wrapper);
@ -31,13 +31,11 @@ function createSection(originalSection, genId, si) {
value: originalSection.code, value: originalSection.code,
}); });
el.CodeMirror = cm; // used by getAssociatedEditor el.CodeMirror = cm; // used by getAssociatedEditor
cm.el = el;
editor.applyScrollInfo(cm, si); editor.applyScrollInfo(cm, si);
const changeListeners = new Set(); const changeListeners = new Set();
const appliesToContainer = $('.applies-to', el); const appliesToContainer = $('.applies-to-list', el);
const appliesToList = $('.applies-to-list', el);
const appliesTo = []; const appliesTo = [];
MozDocMapper.forEachProp(originalSection, (type, value) => MozDocMapper.forEachProp(originalSection, (type, value) =>
insertApplyAfter({type, value})); insertApplyAfter({type, value}));
@ -115,21 +113,66 @@ function createSection(originalSection, genId, si) {
changeGeneration = newGeneration; changeGeneration = newGeneration;
emitSectionChange('code'); emitSectionChange('code');
}); });
cm.display.wrapper.on('keydown', event => handleKeydown(cm, event), true);
$('.test-regexp', el).onclick = () => updateRegexpTester(true); $('.test-regexp', el).onclick = () => updateRegexpTester(true);
initBeautifyButton($('.beautify-section', el), [cm]); initBeautifyButton($('.beautify-section', el), [cm]);
} }
function handleKeydown(cm, event) {
if (event.shiftKey || event.altKey || event.metaKey) {
return;
}
const {key} = event;
const {line, ch} = cm.getCursor();
switch (key) {
case 'ArrowLeft':
if (line || ch) {
return;
}
// fallthrough
case 'ArrowUp':
cm = line === 0 && editor.prevEditor(cm, false);
if (!cm) {
return;
}
event.preventDefault();
event.stopPropagation();
cm.setCursor(cm.doc.size - 1, key === 'ArrowLeft' ? 1e20 : ch);
break;
case 'ArrowRight':
if (line < cm.doc.size - 1 || ch < cm.getLine(line).length - 1) {
return;
}
// fallthrough
case 'ArrowDown':
cm = line === cm.doc.size - 1 && editor.nextEditor(cm, false);
if (!cm) {
return;
}
event.preventDefault();
event.stopPropagation();
cm.setCursor(0, 0);
break;
}
}
async function updateRegexpTester(toggle) { async function updateRegexpTester(toggle) {
const isLoaded = typeof regexpTester === 'object' || const isLoaded = typeof regexpTester === 'object';
toggle && await require(['/edit/regexp-tester']); /* global regexpTester */ if (toggle && !isLoaded) {
if (toggle != null) { await require(['/edit/regexp-tester']); /* global regexpTester */
}
if (toggle != null && isLoaded) {
regexpTester.toggle(toggle); regexpTester.toggle(toggle);
} }
const regexps = appliesTo.filter(a => a.type === 'regexp') const regexps = appliesTo.filter(a => a.type === 'regexp')
.map(a => a.value); .map(a => a.value);
const hasRe = regexps.length > 0; if (regexps.length) {
if (hasRe && isLoaded) regexpTester.update(regexps); el.classList.add('has-regexp');
el.classList.toggle('has-regexp', hasRe); if (isLoaded) regexpTester.update(regexps);
} else {
el.classList.remove('has-regexp');
if (isLoaded) regexpTester.toggle(false);
}
} }
function updateTocEntry(origin) { function updateTocEntry(origin) {
@ -197,13 +240,11 @@ function createSection(originalSection, genId, si) {
function insertApplyAfter(init, base) { function insertApplyAfter(init, base) {
const apply = createApply(init); const apply = createApply(init);
appliesTo.splice(base ? appliesTo.indexOf(base) + 1 : appliesTo.length, 0, apply); appliesTo.splice(base ? appliesTo.indexOf(base) + 1 : appliesTo.length, 0, apply);
appliesToList.insertBefore(apply.el, base ? base.el.nextSibling : null); appliesToContainer.insertBefore(apply.el, base ? base.el.nextSibling : null);
toggleDataset(appliesToContainer, 'all', init.all);
dirty.add(apply, apply); dirty.add(apply, apply);
if (appliesTo.length > 1 && appliesTo[0].all) { if (appliesTo.length > 1 && appliesTo[0].all) {
removeApply(appliesTo[0]); removeApply(appliesTo[0]);
} }
if (base) requestAnimationFrame(shrinkSectionBy1);
emitSectionChange('apply'); emitSectionChange('apply');
return apply; return apply;
} }
@ -312,15 +353,6 @@ function createSection(originalSection, genId, si) {
dirty.add(`${dirtyPrefix}.value`, value); dirty.add(`${dirtyPrefix}.value`, value);
} }
} }
function shrinkSectionBy1() {
const cmEl = cm.display.wrapper;
const cmH = cmEl.offsetHeight;
const viewH = el.parentElement.offsetHeight;
if (el.offsetHeight > viewH && cmH > Math.min(viewH / 2, cm.display.sizer.offsetHeight + 30)) {
cmEl.style.height = (cmH - appliesToContainer.offsetHeight / (appliesTo.length || 1) | 0) + 'px';
}
}
} }
function createResizeGrip(cm) { function createResizeGrip(cm) {
@ -349,7 +381,8 @@ function createResizeGrip(cm) {
cm.display.lineDiv.offsetParent.offsetTop + cm.display.lineDiv.offsetParent.offsetTop +
/* borders */ /* borders */
wrapper.offsetHeight - wrapper.clientHeight; wrapper.offsetHeight - wrapper.clientHeight;
document.body.classList.add('resizing-v'); wrapper.style.pointerEvents = 'none';
document.body.style.cursor = 's-resize';
document.on('mousemove', resize); document.on('mousemove', resize);
document.on('mouseup', resizeStop); document.on('mouseup', resizeStop);
@ -365,7 +398,8 @@ function createResizeGrip(cm) {
function resizeStop() { function resizeStop() {
document.off('mouseup', resizeStop); document.off('mouseup', resizeStop);
document.off('mousemove', resize); document.off('mousemove', resize);
document.body.classList.remove('resizing-v'); wrapper.style.pointerEvents = '';
document.body.style.cursor = '';
} }
}; };

Some files were not shown because too many files have changed in this diff Show More