/* * Copyright (C) 2023-2024 Yomitan Authors * Copyright (C) 2021-2022 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/>. */ import css from 'css'; import fs from 'fs'; import path from 'path'; import {fileURLToPath} from 'url'; const dirname = path.dirname(fileURLToPath(import.meta.url)); /** * @returns {{cssFilePath: string, overridesCssFilePath: string, outputPath: string}[]} */ export function getTargets() { return [ { cssFilePath: path.join(dirname, '..', 'ext/css/structured-content.css'), overridesCssFilePath: path.join(dirname, 'data/structured-content-overrides.css'), outputPath: path.join(dirname, '..', 'ext/data/structured-content-style.json') }, { cssFilePath: path.join(dirname, '..', 'ext/css/display-pronunciation.css'), overridesCssFilePath: path.join(dirname, 'data/display-pronunciation-overrides.css'), outputPath: path.join(dirname, '..', 'ext/data/pronunciation-style.json') } ]; } /** * @param {import('css-style-applier').RawStyleData} rules * @param {string[]} selectors * @returns {number} */ 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; } /** * @param {import('css-style-applier').RawStyleDataStyleArray} styles * @param {string} property * @param {Map<string, number>} removedProperties * @returns {number} */ 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; } /** * Manually formats JSON for easier CSS parseability. * @param {import('css-style-applier').RawStyleData} rules CSS ruleset. * @returns {string} */ export function formatRulesJson(rules) { // This is similar to the following code, but formatted a but more succinctly: // return JSON.stringify(rules, null, 4); const indent1 = ' '; const indent2 = indent1.repeat(2); const indent3 = indent1.repeat(3); let result = ''; result += '['; let ruleIndex = 0; for (const {selectors, styles} of rules) { if (ruleIndex > 0) { result += ','; } result += `\n${indent1}{\n${indent2}"selectors": `; result += ( selectors.length === 1 ? `[${JSON.stringify(selectors[0], null, 4)}]` : JSON.stringify(selectors, null, 4).replace(/\n/g, '\n' + indent2) ); result += `,\n${indent2}"styles": [`; let styleIndex = 0; for (const [key, value] of styles) { if (styleIndex > 0) { result += ','; } result += `\n${indent3}[${JSON.stringify(key)}, ${JSON.stringify(value)}]`; ++styleIndex; } if (styleIndex > 0) { result += `\n${indent2}`; } result += `]\n${indent1}}`; ++ruleIndex; } if (ruleIndex > 0) { result += '\n'; } result += ']'; result += '\n'; return result; } /** * Generates a CSS ruleset. * @param {string} cssFilePath * @param {string} overridesCssFilePath * @returns {import('css-style-applier').RawStyleData} * @throws {Error} */ export function generateRules(cssFilePath, overridesCssFilePath) { const cssFileContent = fs.readFileSync(cssFilePath, {encoding: 'utf8'}); const overridesCssFileContent = fs.readFileSync(overridesCssFilePath, {encoding: 'utf8'}); const defaultStylesheet = /** @type {css.StyleRules} */ (css.parse(cssFileContent, {}).stylesheet); const overridesStylesheet = /** @type {css.StyleRules} */ (css.parse(overridesCssFileContent, {}).stylesheet); const removePropertyPattern = /^remove-property\s+([\w\W]+)$/; const removeRulePattern = /^remove-rule$/; const propertySeparator = /\s+/; /** @type {import('css-style-applier').RawStyleData} */ const rules = []; for (const rule of defaultStylesheet.rules) { if (rule.type !== 'rule') { continue; } const {selectors, declarations} = /** @type {css.Rule} */ (rule); if (typeof selectors === 'undefined') { continue; } /** @type {import('css-style-applier').RawStyleDataStyleArray} */ const styles = []; if (typeof declarations !== 'undefined') { for (const declaration of declarations) { if (declaration.type !== 'declaration') { console.log(declaration); continue; } const {property, value} = /** @type {css.Declaration} */ (declaration); if (typeof property !== 'string' || typeof value !== 'string') { continue; } styles.push([property, value]); } } if (styles.length > 0) { rules.push({selectors, styles}); } } for (const rule of overridesStylesheet.rules) { if (rule.type !== 'rule') { continue; } const {selectors, declarations} = /** @type {css.Rule} */ (rule); if (typeof selectors === 'undefined' || typeof declarations === 'undefined') { continue; } /** @type {Map<string, number>} */ 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} = /** @type {css.Declaration} */ (declaration); if (typeof property === 'string' && typeof value === 'string') { 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 = (/** @type {css.Comment} */ (declaration).comment || '').trim(); let m; if ((m = removePropertyPattern.exec(comment)) !== null) { for (const property of m[1].split(propertySeparator)) { const removeCount = removeProperty(rules[index].styles, property, removedProperties); if (removeCount === 0) { throw new Error(`Property removal is unnecessary; ${property} does not exist`); } } } else if (removeRulePattern.test(comment)) { 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; }