Proper support for alternate selectors

This commit is contained in:
Henry Jameson 2024-10-21 23:10:54 +03:00
parent 21cb168dfd
commit 7e4fe93c7f
4 changed files with 59 additions and 58 deletions

View file

@ -1,6 +1,6 @@
export default { export default {
name: 'Modals', name: 'Modals',
selector: '.modal-view', selector: ['.modal-view', '#modal', '.shout-panel'],
lazy: true, lazy: true,
notEditable: true, notEditable: true,
validInnerComponents: [ validInnerComponents: [

View file

@ -1,6 +1,6 @@
export default { export default {
name: 'Scrollbar', name: 'Scrollbar',
selector: '::-webkit-scrollbar', selector: ['::-webkit-scrollbar-button', '::-webkit-scrollbar-thumb', '::-webkit-resizer'],
notEditable: true, // for now notEditable: true, // for now
defaultRules: [ defaultRules: [
{ {

View file

@ -56,22 +56,7 @@ export const generateTheme = (inputRuleset, callbacks, debug) => {
getCssRules(themes3.eager, debug).forEach(rule => { getCssRules(themes3.eager, debug).forEach(rule => {
// Hacks to support multiple selectors on same component // Hacks to support multiple selectors on same component
if (rule.match(/::-webkit-scrollbar-button/)) { onNewRule(rule, false)
const parts = rule.split(/[{}]/g)
const newRule = [
parts[0],
', ',
parts[0].replace(/button/, 'thumb'),
', ',
parts[0].replace(/scrollbar-button/, 'resizer'),
' {',
parts[1],
'}'
].join('')
onNewRule(newRule, false)
} else {
onNewRule(rule, false)
}
}) })
onEagerFinished() onEagerFinished()
@ -85,22 +70,7 @@ export const generateTheme = (inputRuleset, callbacks, debug) => {
const chunk = chunks[counter] const chunk = chunks[counter]
Promise.all(chunk.map(x => x())).then(result => { Promise.all(chunk.map(x => x())).then(result => {
getCssRules(result.filter(x => x), debug).forEach(rule => { getCssRules(result.filter(x => x), debug).forEach(rule => {
if (rule.match(/\.modal-view/)) { onNewRule(rule, true)
const parts = rule.split(/[{}]/g)
const newRule = [
parts[0],
', ',
parts[0].replace(/\.modal-view/, '#modal'),
', ',
parts[0].replace(/\.modal-view/, '.shout-panel'),
' {',
parts[1],
'}'
].join('')
onNewRule(newRule, true)
} else {
onNewRule(rule, true)
}
}) })
// const t1 = performance.now() // const t1 = performance.now()
// console.debug('Chunk ' + counter + ' took ' + (t1 - t0) + 'ms') // console.debug('Chunk ' + counter + ' took ' + (t1 - t0) + 'ms')

View file

@ -56,43 +56,74 @@ export const getAllPossibleCombinations = (array) => {
* *
* @returns {String} CSS selector (or path) * @returns {String} CSS selector (or path)
*/ */
export const genericRuleToSelector = components => (rule, ignoreOutOfTreeSelector, isParent) => { export const genericRuleToSelector = components => (rule, ignoreOutOfTreeSelector, liteMode, children) => {
const isParent = !!children
if (!rule && !isParent) return null if (!rule && !isParent) return null
const component = components[rule.component] const component = components[rule.component]
const { states = {}, variants = {}, selector, outOfTreeSelector } = component const { states = {}, variants = {}, outOfTreeSelector } = component
const applicableStates = ((rule.state || []).filter(x => x !== 'normal')).map(state => states[state]) const expand = (array = [], subArray = []) => {
if (array.length === 0) return subArray.map(x => [x])
if (subArray.length === 0) return array.map(x => [x])
return array.map(a => {
return subArray.map(b => [a, b])
}).flat()
}
let componentSelectors = Array.isArray(component.selector) ? component.selector : [component.selector]
if (ignoreOutOfTreeSelector || liteMode) componentSelectors = [componentSelectors[0]]
componentSelectors = componentSelectors.map(selector => {
if (selector === ':root') {
return ''
} else if (isParent) {
return selector
} else {
if (outOfTreeSelector && !ignoreOutOfTreeSelector) return outOfTreeSelector
return selector
}
})
const applicableVariantName = (rule.variant || 'normal') const applicableVariantName = (rule.variant || 'normal')
let applicableVariant = '' let variantSelectors = null
if (applicableVariantName !== 'normal') { if (applicableVariantName !== 'normal') {
applicableVariant = variants[applicableVariantName] variantSelectors = variants[applicableVariantName]
} else { } else {
applicableVariant = variants?.normal ?? '' variantSelectors = variants?.normal ?? ''
} }
variantSelectors = Array.isArray(variantSelectors) ? variantSelectors : [variantSelectors]
if (ignoreOutOfTreeSelector || liteMode) variantSelectors = [variantSelectors[0]]
let realSelector const applicableStates = (rule.state || []).filter(x => x !== 'normal')
if (selector === ':root') { // const applicableStates = (rule.state || [])
realSelector = '' const statesSelectors = applicableStates.map(state => {
} else if (isParent) { const selector = states[state] || ''
realSelector = selector let arraySelector = Array.isArray(selector) ? selector : [selector]
} else { if (ignoreOutOfTreeSelector || liteMode) arraySelector = [arraySelector[0]]
if (outOfTreeSelector && !ignoreOutOfTreeSelector) realSelector = outOfTreeSelector arraySelector
else realSelector = selector .sort((a, b) => {
} if (a.startsWith(':')) return 1
if (/^[a-z]/.exec(a)) return -1
else return 0
})
.join('')
return arraySelector
})
const selectors = [realSelector, applicableVariant, ...applicableStates] const statesSelectorsFlat = statesSelectors.reduce((acc, s) => {
.sort((a, b) => { return expand(acc, s).map(st => st.join(''))
if (a.startsWith(':')) return 1 }, [])
if (/^[a-z]/.exec(a)) return -1
else return 0 const componentVariant = expand(componentSelectors, variantSelectors).map(cv => cv.join(''))
}) const componentVariantStates = expand(componentVariant, statesSelectorsFlat).map(cvs => cvs.join(''))
.join('') const selectors = expand(componentVariantStates, children).map(cvsc => cvsc.join(' '))
/*
*/
if (rule.parent) { if (rule.parent) {
return (genericRuleToSelector(components)(rule.parent, ignoreOutOfTreeSelector, true) + ' ' + selectors).trim() return genericRuleToSelector(components)(rule.parent, ignoreOutOfTreeSelector, liteMode, selectors)
} }
return selectors.trim()
return selectors.join(', ').trim()
} }
/** /**