diff options
| author | toasted-nutbread <toasted-nutbread@users.noreply.github.com> | 2021-06-27 17:57:00 -0400 | 
|---|---|---|
| committer | GitHub <noreply@github.com> | 2021-06-27 17:57:00 -0400 | 
| commit | 002da9fba8800b796ed98dd83e2ab68b4d37a1cc (patch) | |
| tree | a500cec04f9a587926b2dc20290188a96b4bd209 /dev | |
| parent | 24ef820ba816411288d8bc739f6e69abb511deb0 (diff) | |
Structured content style json (#1771)
* Install css
* Remove unnecessary rule
* Create CSS overrides file
* Create script to generate CSS JSON file
* Generate JSON
* Add test
Diffstat (limited to 'dev')
| -rw-r--r-- | dev/data/structured-content-overrides.css | 50 | ||||
| -rw-r--r-- | dev/generate-structured-content-style.js | 191 | 
2 files changed, 241 insertions, 0 deletions
| diff --git a/dev/data/structured-content-overrides.css b/dev/data/structured-content-overrides.css new file mode 100644 index 00000000..31873760 --- /dev/null +++ b/dev/data/structured-content-overrides.css @@ -0,0 +1,50 @@ +/* + * Copyright (C) 2021  Yomichan Authors + * + * This program is free software: you can redistribute it and/or modify + * it under the entrys of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program.  If not, see <https://www.gnu.org/licenses/>. + */ + +.gloss-image-container { +    font-size: 1px; +} +.gloss-image-link[data-background=true]>.gloss-image-container { +    /* remove-property background-color */ +} +.gloss-image-container-overlay { +    font-size: initial; +    line-height: initial; +    color: initial; +} +.gloss-image-background { +    background-color: currentColor; +} +:root[data-browser=firefox] .gloss-image-link[data-image-rendering=crisp-edges] .gloss-image, +:root[data-browser=firefox] .gloss-image-link[data-image-rendering=crisp-edges] .gloss-image-background, +:root[data-browser=firefox-mobile] .gloss-image-link[data-image-rendering=crisp-edges] .gloss-image, +:root[data-browser=firefox-mobile] .gloss-image-link[data-image-rendering=crisp-edges] .gloss-image-background { +    /* remove-rule */ +} +.gloss-image-link-text { +    line-height: initial; +} +.gloss-sc-thead, +.gloss-sc-tfoot, +.gloss-sc-th { +    /* remove-property background-color */ +} +.gloss-sc-th, +.gloss-sc-td { +    border-width: 1px; +    border-color: currentColor; +} diff --git a/dev/generate-structured-content-style.js b/dev/generate-structured-content-style.js new file mode 100644 index 00000000..8335efc2 --- /dev/null +++ b/dev/generate-structured-content-style.js @@ -0,0 +1,191 @@ +/* + * Copyright (C) 2021  Yomichan Authors + * + * This program is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * This program is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with this program.  If not, see <https://www.gnu.org/licenses/>. + */ + +const fs = require('fs'); +const path = require('path'); +const css = require('css'); +const {testMain} = require('./util'); + +function indexOfRule(rules, selectors) { +    const jj = selectors.length; +    for (let i = 0, ii = rules.length; i < ii; ++i) { +        const ruleSelectors = rules[i].selectors; +        if (ruleSelectors.length !== jj) { continue; } +        let okay = true; +        for (let j = 0; j < jj; ++j) { +            if (selectors[j] !== ruleSelectors[j]) { +                okay = false; +                break; +            } +        } +        if (okay) { return i; } +    } +    return -1; +} + +function removeProperty(styles, property, removedProperties) { +    let removeCount = removedProperties.get(property); +    if (typeof removeCount !== 'undefined') { return removeCount; } +    removeCount = 0; +    for (let i = 0, ii = styles.length; i < ii; ++i) { +        const key = styles[i][0]; +        if (key !== property) { continue; } +        styles.splice(i, 1); +        --i; +        --ii; +        ++removeCount; +    } +    removedProperties.set(property, removeCount); +    return removeCount; +} + +function formatRulesJson(rules) { +    // Manually format JSON, for improved compactness +    // return JSON.stringify(rules, null, 4); +    const indent1 = '    '; +    const indent2 = indent1.repeat(2); +    const indent3 = indent1.repeat(3); +    let result = ''; +    result += '['; +    let index1 = 0; +    for (const {selectors, styles} of rules) { +        if (index1 > 0) { result += ','; } +        result += `\n${indent1}{\n${indent2}"selectors": `; +        if (selectors.length === 1) { +            result += `[${JSON.stringify(selectors[0], null, 4)}]`; +        } else { +            result += JSON.stringify(selectors, null, 4).replace(/\n/g, '\n' + indent2); +        } +        result += `,\n${indent2}"styles": [`; +        let index2 = 0; +        for (const [key, value] of styles) { +            if (index2 > 0) { result += ','; } +            result += `\n${indent3}[${JSON.stringify(key)}, ${JSON.stringify(value)}]`; +            ++index2; +        } +        if (index2 > 0) { result += `\n${indent2}`; } +        result += `]\n${indent1}}`; +        ++index1; +    } +    if (index1 > 0) { result += '\n'; } +    result += ']'; +    return result; +} + +function generateRules() { +    const content1 = fs.readFileSync(path.join(__dirname, '..', 'ext/css/structured-content.css'), {encoding: 'utf8'}); +    const content2 = fs.readFileSync(path.join(__dirname, 'data/structured-content-overrides.css'), {encoding: 'utf8'}); +    const stylesheet1 = css.parse(content1, {}).stylesheet; +    const stylesheet2 = css.parse(content2, {}).stylesheet; + +    const removePropertyPattern = /^remove-property\s+([a-zA-Z0-9-]+)$/; +    const removeRulePattern = /^remove-rule$/; + +    const rules = []; + +    // Default stylesheet +    for (const rule of stylesheet1.rules) { +        if (rule.type !== 'rule') { continue; } +        const {selectors, declarations} = rule; +        const styles = []; +        for (const declaration of declarations) { +            if (declaration.type !== 'declaration') { console.log(declaration); continue; } +            const {property, value} = declaration; +            styles.push([property, value]); +        } +        if (styles.length > 0) { +            rules.push({selectors, styles}); +        } +    } + +    // Overrides +    for (const rule of stylesheet2.rules) { +        if (rule.type !== 'rule') { continue; } +        const {selectors, declarations} = rule; +        const removedProperties = new Map(); +        for (const declaration of declarations) { +            switch (declaration.type) { +                case 'declaration': +                    { +                        const index = indexOfRule(rules, selectors); +                        let entry; +                        if (index >= 0) { +                            entry = rules[index]; +                        } else { +                            entry = {selectors, styles: []}; +                            rules.push(entry); +                        } +                        const {property, value} = declaration; +                        removeProperty(entry.styles, property, removedProperties); +                        entry.styles.push([property, value]); +                    } +                    break; +                case 'comment': +                    { +                        const index = indexOfRule(rules, selectors); +                        if (index < 0) { throw new Error('Could not find rule with matching selectors'); } +                        const comment = declaration.comment.trim(); +                        let m; +                        if ((m = removePropertyPattern.exec(comment)) !== null) { +                            const property = m[1]; +                            const removeCount = removeProperty(rules[index].styles, property, removedProperties); +                            if (removeCount === 0) { throw new Error(`Property removal is unnecessary; ${property} does not exist`); } +                        } else if ((m = removeRulePattern.exec(comment)) !== null) { +                            rules.splice(index, 1); +                        } +                    } +                    break; +            } +        } +    } + +    // Remove empty +    for (let i = 0, ii = rules.length; i < ii; ++i) { +        if (rules[i].styles.length > 0) { continue; } +        rules.splice(i, 1); +        --i; +        --ii; +    } + +    return rules; +} + +function generateRulesJson() { +    return formatRulesJson(generateRules()); +} + +function getOutputPath() { +    return path.join(__dirname, '..', 'ext/data/structured-content-style.json'); +} + +function main() { +    const outputFileName = getOutputPath(); +    const json = generateRulesJson(); +    fs.writeFileSync(outputFileName, json, {encoding: 'utf8'}); +} + + +if (require.main === module) { +    testMain(main, process.argv.slice(2)); +} + + +module.exports = { +    generateRules, +    generateRulesJson, +    getOutputPath +}; |