extract fitSelectBox()

This commit is contained in:
tophf 2020-12-14 14:53:08 +03:00
parent 9ffa754246
commit df2e8221d2
8 changed files with 122 additions and 110 deletions

View File

@ -7,6 +7,7 @@ define(require => {
tryRegExp,
} = require('/js/toolbox');
const {API} = require('/js/msg');
const usercss = require('./usercss-api-helper');
// toLocaleLowerCase cache, autocleared after 1 minute
const cache = new Map();
@ -103,13 +104,13 @@ define(require => {
function extractMeta(style) {
return style.usercssData
? (style.sourceCode.match(API.usercss.rxMETA) || [''])[0]
? (style.sourceCode.match(usercss.rxMETA) || [''])[0]
: null;
}
function stripMeta(style) {
return style.usercssData
? style.sourceCode.replace(API.usercss.rxMETA, '')
? style.sourceCode.replace(usercss.rxMETA, '')
: null;
}

View File

@ -141,10 +141,6 @@ define(require => {
}
});
/**
* @type Prefs
* @namespace Prefs
*/
const prefs = {
STORAGE_KEY,

View File

@ -13,7 +13,8 @@ define(require => {
const debounceTimers = new Map();
let URLS, deepCopy, deepEqual, deepMerge;
const toolbox = {
/** @type {Toolbox} */
const toolbox = /** @namespace Toolbox */ {
CHROME,
FIREFOX,

View File

@ -217,7 +217,7 @@
</div>
</label>
<div class="select-resizer">
<select id="manage.onlyEnabled.invert">
<select id="manage.onlyEnabled.invert" class="fit-width">
<option i18n-text="manageOnlyEnabled" value="false"></option>
<option i18n-text="manageOnlyDisabled" value="true"></option>
</select>
@ -235,7 +235,7 @@
</div>
</label>
<div class="select-resizer">
<select id="manage.onlyLocal.invert" i18n-title="manageOnlyLocalTooltip">
<select id="manage.onlyLocal.invert" i18n-title="manageOnlyLocalTooltip" class="fit-width">
<option i18n-text="manageOnlyLocal" value="false"></option>
<option i18n-text="manageOnlyExternal" value="true"></option>
</select>
@ -253,7 +253,7 @@
</div>
</label>
<div class="select-resizer">
<select id="manage.onlyUsercss.invert">
<select id="manage.onlyUsercss.invert" class="fit-width">
<option i18n-text="manageOnlyUsercss" value="false"></option>
<option i18n-text="manageOnlyNonUsercss" value="true"></option>
</select>
@ -276,7 +276,7 @@
data-filter=":not(.not-matching)"
data-filter-hide=".not-matching">
<div class="select-wrapper">
<select id="searchMode">
<select id="searchMode" class="fit-width">
<option i18n-text="searchStylesName" value="name"></option>
<option i18n-text="searchStylesMeta" value="meta" selected></option>
<option i18n-text="searchStylesCode" value="code"></option>

View File

@ -35,17 +35,6 @@ define(require => {
const Events = {
ENTRY_ROUTES: {
'input, .enable, .disable': 'toggle',
'.style-name': 'name',
'.homepage': 'external',
'.check-update': 'check',
'.update': 'update',
'.delete': 'delete',
'.applies-to .expander': 'expandTargets',
'.configure-usercss': 'config',
},
addEntryTitle(link) {
const style = link.closest('.entry').styleMeta;
const ucd = style.usercssData;
@ -130,8 +119,7 @@ define(require => {
for (const selector in Events.ENTRY_ROUTES) {
for (let el = target; el && el !== entry; el = el.parentElement) {
if (el.matches(selector)) {
const handler = Events.ENTRY_ROUTES[selector];
return Events[handler].call(el, event, entry);
return Events.ENTRY_ROUTES[selector].call(el, event, entry);
}
}
}
@ -164,6 +152,17 @@ define(require => {
},
};
Events.ENTRY_ROUTES = {
'input, .enable, .disable': Events.toggle,
'.style-name': Events.name,
'.homepage': Events.external,
'.check-update': Events.check,
'.update': Events.update,
'.delete': Events.delete,
'.applies-to .expander': Events.expandTargets,
'.configure-usercss': Events.config,
};
async function handleUpdateForId(id, opts) {
handleUpdate(await API.styles.get(id), opts);
bulkChangeQueue.time = performance.now();

View File

@ -62,20 +62,6 @@ define(require => {
}
});
HTMLSelectElement.prototype.adjustWidth = function () {
const sel = this.selectedOptions[0];
if (!sel) return;
const wOld = parseFloat(this.style.width);
const opts = [...this];
opts.forEach(opt => opt !== sel && opt.remove());
this.style.width = '';
requestAnimationFrame(() => {
const w = this.offsetWidth;
if (w && wOld !== w) this.style.width = w + 'px';
this.append(...opts);
});
};
function initFilters() {
$('#search').oninput = $('#searchMode').oninput = function (e) {
router.updateSearch(this.id, e.target.value);
@ -112,10 +98,6 @@ define(require => {
slaveData.filter = filter;
slaveData.filterHide = valueMap.get(!value);
debounce(filterOnChange, 0, event);
// avoid triggering MutationObserver during page load
if (document.readyState === 'complete') {
el.adjustWidth();
}
};
el.onchange({target: el});
});
@ -150,14 +132,6 @@ define(require => {
router.updateSearch('search', '');
};
// Adjust width after selects are visible
prefs.subscribe('manage.filters.expanded', () => {
const el = $('#filters');
if (el.open) {
$$('.filter-selection select', el).forEach(select => select.adjustWidth());
}
});
filterOnChange({forceRefilter: true});
}

View File

@ -1,6 +1,6 @@
'use strict';
define(require => {
define(async require => {
const {API, msg} = require('/js/msg');
const {
CHROME,
@ -21,7 +21,8 @@ define(require => {
const {
BULK_THROTTLE_MS,
bulkChangeQueue,
waitingForContainer,
containerPromise,
fitSelectBoxInOpenDetails,
showStyles,
switchUI,
} = require('./render');
@ -32,54 +33,56 @@ define(require => {
handleVisibilityChange,
} = require('./events');
(async () => {
const query = router.getSearch('search');
const [styles, ids, container] = await Promise.all([
API.styles.getAll(),
query && API.searchDB({query, mode: router.getSearch('searchMode')}),
waitingForContainer,
prefs.initializing,
]);
container.onclick = Events.entryClicked;
$('#manage-options-button').onclick = () => router.updateHash('#stylus-options');
$('#sync-styles').onclick = () => router.updateHash('#stylus-options');
$$('#header a[href^="http"]').forEach(a => (a.onclick = Events.external));
// show date installed & last update on hover
container.on('mouseover', Events.lazyAddEntryTitle, {passive: true});
container.on('mouseout', Events.lazyAddEntryTitle, {passive: true});
document.on('visibilitychange', handleVisibilityChange);
// N.B. triggers existing onchange listeners
setupLivePrefs();
prefs.subscribe(newUI.ids.map(newUI.prefKeyForId), () => switchUI());
switchUI({styleOnly: true});
// translate CSS manually
document.styleSheets[0].insertRule(
`:root {${[
'genericDisabledLabel',
'updateAllCheckSucceededSomeEdited',
'filteredStylesAllHidden',
].map(id => `--${id}:"${CSS.escape(t(id))}";`).join('')
}}`);
if (!VIVALDI) {
$$('.filter-selection select').forEach(el => el.adjustWidth());
}
if (CHROME >= 80 && CHROME <= 88) {
// Wrong checkboxes are randomly checked after going back in history, https://crbug.com/1138598
window.on('pagehide', () => {
$$('input[type=checkbox]').forEach((el, i) => (el.name = `bug${i}`));
});
}
showStyles(styles, ids);
require([
'./import-export',
'./incremental-search',
]);
})();
msg.onExtension(onRuntimeMessage);
router.watch({hash: '#stylus-options'}, state => (state ? embedOptions : unembedOptions)());
router.watch({hash: '#stylus-options'}, toggleEmbeddedOptions);
window.on('closeOptions', () => router.updateHash(''));
const query = router.getSearch('search');
const [styles, ids, container] = await Promise.all([
API.styles.getAll(),
query && API.searchDB({query, mode: router.getSearch('searchMode')}),
containerPromise,
prefs.initializing,
]);
container.on('click', Events.entryClicked);
container.on('mouseover', Events.lazyAddEntryTitle, {passive: true});
container.on('mouseout', Events.lazyAddEntryTitle, {passive: true});
$('#manage-options-button').onclick = () => router.updateHash('#stylus-options');
$('#sync-styles').onclick = () => router.updateHash('#stylus-options');
$$('#header a[href^="http"]').forEach(a => (a.onclick = Events.external));
document.on('visibilitychange', handleVisibilityChange);
setupLivePrefs();
prefs.subscribe(newUI.ids.map(newUI.prefKeyForId), () => switchUI());
switchUI({styleOnly: true});
// translate CSS manually
document.styleSheets[0].insertRule(
`:root {${[
'genericDisabledLabel',
'updateAllCheckSucceededSomeEdited',
'filteredStylesAllHidden',
].map(id => `--${id}:"${CSS.escape(t(id))}";`).join('')
}}`);
if (!VIVALDI) {
fitSelectBoxInOpenDetails($('#filters'));
}
if (CHROME >= 80 && CHROME <= 88) {
// Wrong checkboxes are randomly checked after going back in history, https://crbug.com/1138598
window.on('pagehide', () => {
$$('input[type=checkbox]').forEach((el, i) => (el.name = `bug${i}`));
});
}
showStyles(styles, ids);
require([
'./import-export',
'./incremental-search',
]);
function onRuntimeMessage(msg) {
switch (msg.method) {
case 'styleUpdated':
@ -101,21 +104,18 @@ define(require => {
setTimeout(sorter.updateStripes, 0, {onlyWhenColumnsChanged: true});
}
function embedOptions() {
const options = $('#stylus-embedded-options') ||
document.documentElement.appendChild($create('iframe', {
async function toggleEmbeddedOptions(state) {
const el = $('#stylus-embedded-options') ||
state && document.documentElement.appendChild($create('iframe', {
id: 'stylus-embedded-options',
src: '/options.html',
}));
options.focus();
}
async function unembedOptions() {
const options = $('#stylus-embedded-options');
if (options) {
options.contentWindow.document.body.classList.add('scaleout');
await animateElement(options, 'fadeout');
options.remove();
if (state) {
el.focus();
} else if (el) {
el.contentDocument.body.classList.add('scaleout');
await animateElement(el, 'fadeout');
el.remove();
}
}
});

View File

@ -36,7 +36,7 @@ define(require => {
BULK_THROTTLE_MS: 100,
bulkChangeQueue: [],
// needed to avoid flicker due to an extra frame and layout shift
waitingForContainer: waitForSelector('#installed').then(el => (installed = el)),
containerPromise: waitForSelector('#installed').then(el => (installed = el)),
$entry(styleOrId, root = installed) {
return $(`#${ENTRY_ID_PREFIX_RAW}${styleOrId.id || styleOrId}`, root);
@ -184,6 +184,26 @@ define(require => {
entry._numTargets = numTargets;
},
/**
* @param {HTMLDetailsElement} el
* @param {string} targetSel
*/
fitSelectBoxInOpenDetails(el, targetSel = 'select.fit-width') {
const run = () => {
if (el.open) {
fitSelectBox(...$$(targetSel, el));
}
};
el.on('change', ({target}) => {
if (el.open && target.matches(targetSel)) {
fitSelectBox(target);
}
});
new MutationObserver(run)
.observe(el, {attributeFilter: ['open'], attributes: true});
run();
},
getFaviconImgSrc(container = installed) {
if (!newUI.enabled || !newUI.favicons) return;
const regexpRemoveNegativeLookAhead = /(\?!([^)]+\))|\(\?![\w(]+[^)]+[\w|)]+)/g;
@ -355,6 +375,26 @@ define(require => {
}
}
function fitSelectBox(...elems) {
const data = [];
for (const el of elems) {
const sel = el.selectedOptions[0];
if (!sel) return;
const oldWidth = parseFloat(el.style.width);
const elOpts = [...el];
data.push({el, elOpts, oldWidth});
elOpts.forEach(opt => opt !== sel && opt.remove());
el.style.width = '';
}
requestAnimationFrame(() => {
for (const {el, elOpts, oldWidth} of data) {
const w = el.offsetWidth;
if (w && oldWidth !== w) el.style.width = w + 'px';
el.append(...elOpts);
}
});
}
function highlightEditedStyle() {
if (!sessionStore.justEditedStyleId) return;
const entry = render.$entry(sessionStore.justEditedStyleId);
@ -375,6 +415,7 @@ define(require => {
const x = Math.max(0, left);
const y = Math.max(0, top);
const first = document.elementFromPoint(x, y);
if (!first) return requestAnimationFrame(loadFavicons.bind(null, ...arguments));
const lastOffset = first.offsetTop + window.innerHeight;
const numTargets = newUI.targets;
let entry = first && first.closest('.entry') || installed.children[0];