From ee316c6c24b5ee1f2d17d1606694ee9c19691c8d Mon Sep 17 00:00:00 2001 From: Trivernis Date: Tue, 30 Jul 2019 21:02:41 +0200 Subject: [PATCH] More plugins and better config - fixed puppeteer bug where MathJax message containers appeared - added plugins - added `extends` to `mdconfig.json` to extend a default config - added option to show all plugins `--plugins` --- CHANGELOG.md | 1 + README.md | 64 +++--- gulpfile.js | 13 +- package-lock.json | 35 +++ package.json | 7 + src/configs/full-mdconfig.json | 27 +++ src/index.ts | 26 ++- src/lib/MarkdownConfig.ts | 15 ++ src/lib/Renderer.ts | 19 +- src/lib/plugins.ts | 88 ++++++-- src/lib/utils.ts | 25 ++- src/styles/default.sass | 17 +- src/styles/includes/vars.sass | 1 + src/styles/includes/w3-math.css | 380 -------------------------------- 14 files changed, 278 insertions(+), 440 deletions(-) create mode 100644 src/configs/full-mdconfig.json delete mode 100644 src/styles/includes/w3-math.css diff --git a/CHANGELOG.md b/CHANGELOG.md index 101a08b..d3d07a1 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -15,3 +15,4 @@ - auto base64 converting for images for a standalone html when bundling flag is set - support for `mdconf.json`config file - MathJax script to html +- `mdconfig.json` extensions with `extends: "name"` diff --git a/README.md b/README.md index ad073f1..7c7f987 100644 --- a/README.md +++ b/README.md @@ -39,27 +39,8 @@ or [!use]: # (plugin1, plugin2, plugin3) ``` -The plugin names are listed in the following table. Basically it is just the package name with the markdown-it removed: - -| module | import name | -|----------------------|-------------| -| markdown-it-footnote | footnote -| markdown-it-anchor | anchor -| markdown-it-mark | mark -| markdown-it-sub | sub -| markdown-it-attrs | attrs -| markdown-it-abbr | abbr -| markdown-it-checkbox | checkbox -| markdown-it-imsize | imsize -| markdown-it-highlightjs | highlightjs -| markdown-it-toc-done-right | toc-done-right -| markdown-it-smartarrows | smartarrows -| markdown-it-plantuml | plantuml -| markdown-it-mathjax | mathjax -| markdown-it-math | math -| markdown-it-div | div -| markdown-it-multimd-table | multimd-table - +The plugin names are listed in the following table. Basically it is just the package name with the markdown-it removed. See [Plugins](#plugins) + For example you can declare the use of `markdown-it-emoji` the following: ```markdown @@ -97,8 +78,9 @@ You can include your own stylesheet. It is applied after the default style. The You can also define plugins, stylesheets and other stuff by using a `mdconf.json` file in the same directory as the main markdown file. Example config: -```json -{ +```json5 +{ // everything is optional + "extends": "full", // extend from predefined configurations "format": "A4", "plugins": [ "emoji", @@ -111,6 +93,36 @@ You can also define plugins, stylesheets and other stuff by using a `mdconf.json } ``` -## Other stuff - -Images and stylesheets are included within the html file. All image urls are converted into base64 urls. +### Predefined configurations + +config | type +-------|----- +full | includes all plugins + +## Plugins + +Plugin Name | Markdown-it plugin +-----------------|--------------------- +emoji | markdown-it-emoji +footnote | markdown-it-footnote +anchor | markdown-it-anchor +mark | markdown-it-mark +sub | markdown-it-sub +attrs | markdown-it-attrs +abbr | markdown-it-abbr +checkbox | markdown-it-checkbox +imsize | markdown-it-imsize +highlightjs | markdown-it-highlightjs +smartarrows | markdown-it-smartarrows +plantuml | markdown-it-plantuml +math | markdown-it-math +div | markdown-it-div +kbd | markdown-it-kbd +video | markdown-it-video +underline | markdown-it-underline +multimd-table | markdown-it-multimd-table +toc-done-right | markdown-it-toc-done-right +center-text | markdown-it-center-text +header-sections | markdown-it-header-sections +task-checkbox | markdown-it-task-checkbox +implicit-figures | markdown-it-implicit-figures diff --git a/gulpfile.js b/gulpfile.js index 8b653a8..b182944 100644 --- a/gulpfile.js +++ b/gulpfile.js @@ -22,10 +22,17 @@ function compileSass() { .pipe(dest('dist/styles')); } -task('cleanBuild', series(clearDist, compileTypescript, compileSass)); -task('default', series(compileTypescript, compileSass)); +function moveOther() { + return src(['src/**/*', '!src/**/*.ts', '!src/**/*.sass']) + .pipe(dest('dist')); +} + + +task('cleanBuild', series(clearDist, compileTypescript, compileSass, moveOther)); +task('default', series(compileTypescript, compileSass, moveOther)); task('watch', () => { - series(compileSass, compileTypescript); + series(compileSass, compileTypescript, moveOther); watch('src/**/*.sass', series(compileSass)); watch('**/*.ts', compileTypescript); + watch(['src/**/*', '!src/**/*.ts', '!src/**/*.sass'], moveOther); }); diff --git a/package-lock.json b/package-lock.json index 49003d2..4c59a3c 100644 --- a/package-lock.json +++ b/package-lock.json @@ -3396,6 +3396,11 @@ "resolved": "https://registry.npmjs.org/markdown-it-attrs/-/markdown-it-attrs-2.4.1.tgz", "integrity": "sha512-BASnIYS+JLpjlhDf7jLV8VOuccxjfDDnQcz5dLfgPsYw8OsgbASexADdIkF7tIdGn+jaQSA4qOZXM3v3W3JBCg==" }, + "markdown-it-center-text": { + "version": "1.0.4", + "resolved": "https://registry.npmjs.org/markdown-it-center-text/-/markdown-it-center-text-1.0.4.tgz", + "integrity": "sha512-7AA0gek4FYJ6gWB2kdGNx06k3GrlO97nEMY7uovQ70/6MtfqPznrgnKoWO3V4p36m3RT5GeQIFXqBH9Wr4iL/g==" + }, "markdown-it-checkbox": { "version": "1.1.0", "resolved": "https://registry.npmjs.org/markdown-it-checkbox/-/markdown-it-checkbox-1.1.0.tgz", @@ -3419,6 +3424,11 @@ "resolved": "https://registry.npmjs.org/markdown-it-footnote/-/markdown-it-footnote-3.0.2.tgz", "integrity": "sha512-JVW6fCmZWjvMdDQSbOT3nnOQtd9iAXmw7hTSh26+v42BnvXeVyGMDBm5b/EZocMed2MbCAHiTX632vY0FyGB8A==" }, + "markdown-it-header-sections": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/markdown-it-header-sections/-/markdown-it-header-sections-1.0.0.tgz", + "integrity": "sha512-GndZbs+iTo7NkNRf00VmT4mtFql7VoJNnSwB7hP5xkomGUTdQ78siVZIuRco4fbiWJd45JUemLxUS9vO8luPSA==" + }, "markdown-it-highlightjs": { "version": "3.0.0", "resolved": "https://registry.npmjs.org/markdown-it-highlightjs/-/markdown-it-highlightjs-3.0.0.tgz", @@ -3428,6 +3438,11 @@ "lodash.flow": "^3.1.0" } }, + "markdown-it-implicit-figures": { + "version": "0.9.0", + "resolved": "https://registry.npmjs.org/markdown-it-implicit-figures/-/markdown-it-implicit-figures-0.9.0.tgz", + "integrity": "sha512-ykLQZmhy8QnNEm6NgImIrr+/QMe7gSnHgLhNAjz9GGCNdSLR7C/SLFNwGdUZq7vK3a0myWiOYE0tpaZuB0rWWw==" + }, "markdown-it-imsize": { "version": "2.0.1", "resolved": "https://registry.npmjs.org/markdown-it-imsize/-/markdown-it-imsize-2.0.1.tgz", @@ -3438,6 +3453,11 @@ "resolved": "https://registry.npmjs.org/markdown-it-ins/-/markdown-it-ins-2.0.0.tgz", "integrity": "sha1-papqMPHi9x6Ul1Z8/f9A8f3mdIM=" }, + "markdown-it-kbd": { + "version": "2.0.0", + "resolved": "https://registry.npmjs.org/markdown-it-kbd/-/markdown-it-kbd-2.0.0.tgz", + "integrity": "sha512-wMEFAF6WIcitSLt2uGuqXjOB6EfmpML7FN6KI8WdhI2A5l1SsgtmaFPR1ss7BjIqUrtjSk4tNhSl0azt3quwEA==" + }, "markdown-it-mark": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/markdown-it-mark/-/markdown-it-mark-2.0.0.tgz", @@ -3493,11 +3513,26 @@ "resolved": "https://registry.npmjs.org/markdown-it-sub/-/markdown-it-sub-1.0.0.tgz", "integrity": "sha1-N1/WAm6ufdywEkl/ZBEZXqHjr+g=" }, + "markdown-it-task-checkbox": { + "version": "1.0.6", + "resolved": "https://registry.npmjs.org/markdown-it-task-checkbox/-/markdown-it-task-checkbox-1.0.6.tgz", + "integrity": "sha512-7pxkHuvqTOu3iwVGmDPeYjQg+AIS9VQxzyLP9JCg9lBjgPAJXGEkChK6A2iFuj3tS0GV3HG2u5AMNhcQqwxpJw==" + }, "markdown-it-toc-done-right": { "version": "4.0.2", "resolved": "https://registry.npmjs.org/markdown-it-toc-done-right/-/markdown-it-toc-done-right-4.0.2.tgz", "integrity": "sha512-LIdwU8coEdBa6JxhMmcst5Qc9/ptVDmUB6dJuJzlxaV75Dry+AziKpQRgZ42SMQ5m7hpb7z6b0erIRoYk+bMWA==" }, + "markdown-it-underline": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/markdown-it-underline/-/markdown-it-underline-1.0.1.tgz", + "integrity": "sha1-Q6VKVB2V1zm0MVdwHx2jBtfDALI=" + }, + "markdown-it-video": { + "version": "0.6.3", + "resolved": "https://registry.npmjs.org/markdown-it-video/-/markdown-it-video-0.6.3.tgz", + "integrity": "sha512-T4th1kwy0OcvyWSN4u3rqPGxvbDclpucnVSSaH3ZacbGsAts964dxokx9s/I3GYsrDCJs4ogtEeEeVP18DQj0Q==" + }, "matchdep": { "version": "2.0.0", "resolved": "https://registry.npmjs.org/matchdep/-/matchdep-2.0.0.tgz", diff --git a/package.json b/package.json index aa2531b..2cd3478 100644 --- a/package.json +++ b/package.json @@ -37,13 +37,17 @@ "markdown-it-abbr": "^1.0.4", "markdown-it-anchor": "^5.2.4", "markdown-it-attrs": "^2.4.1", + "markdown-it-center-text": "^1.0.4", "markdown-it-checkbox": "^1.1.0", "markdown-it-div": "^1.1.0", "markdown-it-emoji": "^1.4.0", "markdown-it-footnote": "^3.0.2", + "markdown-it-header-sections": "^1.0.0", "markdown-it-highlightjs": "^3.0.0", + "markdown-it-implicit-figures": "^0.9.0", "markdown-it-imsize": "^2.0.1", "markdown-it-ins": "^2.0.0", + "markdown-it-kbd": "^2.0.0", "markdown-it-mark": "^2.0.0", "markdown-it-math": "^4.1.1", "markdown-it-mathjax": "^2.0.0", @@ -51,7 +55,10 @@ "markdown-it-plantuml": "^1.4.1", "markdown-it-smartarrows": "^1.0.1", "markdown-it-sub": "^1.0.0", + "markdown-it-task-checkbox": "^1.0.6", "markdown-it-toc-done-right": "^4.0.2", + "markdown-it-underline": "^1.0.1", + "markdown-it-video": "^0.6.3", "node-fetch": "^2.6.0", "puppeteer": "^1.19.0", "uninstall": "0.0.0", diff --git a/src/configs/full-mdconfig.json b/src/configs/full-mdconfig.json new file mode 100644 index 0000000..1102ab7 --- /dev/null +++ b/src/configs/full-mdconfig.json @@ -0,0 +1,27 @@ +{ + "plugins": [ + "emoji", + "footnote", + "anchor", + "mark", + "sub", + "attrs", + "abbr", + "checkbox", + "imsize", + "highlightjs", + "smartarrows", + "plantuml", + "math", + "div", + "multimd-table", + "toc-done-right", + "kbd", + "center-text", + "video", + "header-sections", + "task-checkbox", + "underline", + "implicit-figures" + ] +} diff --git a/src/index.ts b/src/index.ts index 37bf66c..340fc7c 100644 --- a/src/index.ts +++ b/src/index.ts @@ -4,6 +4,7 @@ import {extname, dirname} from 'path'; import {ArgumentParser} from "argparse"; import {MarkdownConfig} from "./lib/MarkdownConfig"; import {logger} from "./lib/logger"; +import {markdownPlugins} from "./lib/plugins"; /** * Returns the filename without the extension. @@ -14,12 +15,13 @@ function noExt(file: string): string { } async function main() { - logger.profile('processing'); + let start = process.hrtime(); const parser = new ArgumentParser({ addHelp: true }); parser.addArgument('file',{ - help: 'The file to render' + help: 'The file to render', + required: false }); parser.addArgument( ['-w', '--watch'], @@ -45,7 +47,24 @@ async function main() { action: 'storeTrue' } ); + parser.addArgument( + ['--plugins'], + { + help: 'prints out the plugins, that can be used.', + required: false, + action: 'storeTrue' + } + ); let args = parser.parseArgs(); + + if (args.plugins) { + console.log('Plugin Name | Markdown-it plugin'); + console.log('-----------------|---------------------'); + console.log(Object.entries(markdownPlugins).map((x:any) => x[0] + .padEnd(16, ' ') + ' | ' + x[1].module).join('\n')); + process.exit(0); + } + let config = new MarkdownConfig(dirname(args.file)); config.bundle = args.bundle || args.pdf; let renderer = new Renderer(config); @@ -72,7 +91,8 @@ async function main() { await writeFile(outputFile, html); console.log(`File stored as ${outputFile}`); } - logger.profile('processing'); + let diff = process.hrtime(start); + logger.info(`Total: ${(diff[0]*1e9 + diff[1])/1e6} ms`); } 4 main(); diff --git a/src/lib/MarkdownConfig.ts b/src/lib/MarkdownConfig.ts index d420546..7d1ba7d 100644 --- a/src/lib/MarkdownConfig.ts +++ b/src/lib/MarkdownConfig.ts @@ -1,8 +1,17 @@ import * as fsx from 'fs-extra'; import * as path from 'path'; import {getMarkdownPlugin} from "./utils"; +import {logger} from "./logger"; const confName = 'mdconfig.json'; +const confDir = `${__dirname}../../configs`; + +/** + * Configs that can be extended + */ +const configs: any = { + "full": `${confDir}/full-${confName}` +}; /** * Markdown config class for easyer access on the configuration. @@ -33,6 +42,11 @@ export class MarkdownConfig { */ private loadData() { let configData = fsx.readJsonSync(this.filename); + + if (configData.extends && configs[configData.extends]) { // assign the base config to the config data + logger.info(`Extending config with ${configData.extends}`); + Object.assign(configData, fsx.readJsonSync(configs[configData.extends])); + } this.format = configData.format || this.format; this.bundle = configData.bundle; if (configData.plugins) @@ -48,5 +62,6 @@ export class MarkdownConfig { public save(filename?: string) { let confFile = filename || this.filename; fsx.writeJsonSync(confFile, this); + logger.info(`Config file saved as ${filename}.`); } } diff --git a/src/lib/Renderer.ts b/src/lib/Renderer.ts index 0b5e2bd..6ec5863 100644 --- a/src/lib/Renderer.ts +++ b/src/lib/Renderer.ts @@ -77,17 +77,26 @@ export class Renderer extends EventEmitter { let document = await fsx.readFile(filename, 'utf-8'); document = document.replace(/\r\n/g, '\n'); logger.verbose(`Applying ${this.beforeRendering.length} beforeRendering functions...`); + let start: any = process.hrtime(); + for (let func of this.beforeRendering) { document = await func(document, filename, this); } + let diff: any = process.hrtime(start); + logger.info(`Pre-Rendering: ${(diff[0]*1e9 + diff[1])/1e6} ms`); this.addConfigPlugins(); - logger.verbose(`Rendering with markdown-it and ${this.config.plugins.size} plugins`); + start = process.hrtime(); let result: string = this.md.render(document); + diff = process.hrtime(start); + logger.info(`Rendering: ${(diff[0]*1e9 + diff[1])/1e6} ms`); + + start = process.hrtime(); - logger.verbose(`Applying ${this.afterRendering.length} afterRendering functions...`); for (let func of this.afterRendering) { result = await func(result, filename, this); } + diff = process.hrtime(start); + logger.info(`Post-Rendering: ${(diff[0]*1e9 + diff[1])/1e6} ms`); logger.verbose('HTML rendered'); this.emit('rendered', result); return result; @@ -108,8 +117,8 @@ export class Renderer extends EventEmitter { const page = await browser.newPage(); logger.info('Setting and evaluating content'); await page.setContent(html); - await page.waitForFunction('window.MathJax.isReady === true'); - await delay(1000); + await page.waitForFunction('window.MathJax.isReady'); + await page.waitForFunction('!!document.querySelector("#MathJax_Message").style.display'); logger.info(`Starting PDF export (format: ${format}) to ${output}`); await page.pdf({ path: output, @@ -194,7 +203,7 @@ export class Renderer extends EventEmitter { private addConfigPlugins() { for (let plugin of this.config.plugins) { try { - this.md.use(plugin); + this.md.use(plugin.module, plugin.options); } catch (err) { logger.error(err); } diff --git a/src/lib/plugins.ts b/src/lib/plugins.ts index 30c62c1..9eb584f 100644 --- a/src/lib/plugins.ts +++ b/src/lib/plugins.ts @@ -1,18 +1,74 @@ export const markdownPlugins: any = { - emoji: 'markdown-it-emoji', - footnote: 'markdown-it-footnote', - anchor: 'markdown-it-anchor', - mark: 'markdown-it-mark', - sub: 'markdown-it-sub', - attrs: 'markdown-it-attrs', - abbr: 'markdown-it-abbr', - checkbox: 'markdown-it-checkbox', - imsize: 'markdown-it-imsize', - highlightjs: 'markdown-it-highlightjs', - smartarrows: 'markdown-it-smartarrows', - plantuml: 'markdown-it-plantuml', - mathjax: 'markdown-it-mathjax', - math: 'markdown-it-math', - div: 'markdown-it-div', - 'toc-done-right': 'markdown-it-toc-done-right' + emoji: { + module: 'markdown-it-emoji' + }, + footnote: { + module: 'markdown-it-footnote' + }, + anchor: { + module: 'markdown-it-anchor' + }, + mark: { + module: 'markdown-it-mark' + }, + sub: { + module: 'markdown-it-sub' + }, + attrs: { + module: 'markdown-it-attrs' + }, + abbr: { + module: 'markdown-it-abbr' + }, + checkbox: { + module: 'markdown-it-checkbox' + }, + imsize: { + module: 'markdown-it-imsize' + }, + highlightjs: { + module: 'markdown-it-highlightjs' + }, + smartarrows: { + module: 'markdown-it-smartarrows' + }, + plantuml: { + module: 'markdown-it-plantuml' + }, + math: { + module: 'markdown-it-math' + }, + div: { + module: 'markdown-it-div' + }, + kbd: { + module: 'markdown-it-kbd' + }, + video: { + module: 'markdown-it-video' + }, + underline: { + module: 'markdown-it-underline' + }, + 'multimd-table': { + module: 'markdown-it-multimd-table' + }, + 'toc-done-right': { + module: 'markdown-it-toc-done-right' + }, + 'center-text': { + module: 'markdown-it-center-text' + }, + 'header-sections': { + module: 'markdown-it-header-sections' + }, + 'task-checkbox': { + module: 'markdown-it-task-checkbox' + }, + 'implicit-figures': { + module: 'markdown-it-implicit-figures', + options: { + figcaption: true + } + } }; diff --git a/src/lib/utils.ts b/src/lib/utils.ts index 3981c30..085330a 100644 --- a/src/lib/utils.ts +++ b/src/lib/utils.ts @@ -55,19 +55,32 @@ export function includeMathJax(dom: JSDOM): JSDOM { /** * Returns the markdown plugin associated with the pluginName + * The plugin is first searched in the plugins definition. + * Then it is tried to require the plugin. * @param pluginName */ -export function getMarkdownPlugin(pluginName: string) { +export function getMarkdownPlugin(pluginName: any) { logger.debug(`Trying to resolve plugin ${pluginName}`); if (markdownPlugins[pluginName]) { - return require(markdownPlugins[pluginName]); + if (markdownPlugins[pluginName].module.constructor.name === 'String') + markdownPlugins[pluginName].module = require(markdownPlugins[pluginName].module); + return markdownPlugins[pluginName]; } else { try { - let plugin = require(pluginName); - if (plugin) - return plugin; + if (pluginName.module) { + if (pluginName.module.constructor.name === 'String') + pluginName.module = require(pluginName.module); + return pluginName + } else { + let plugin = require(pluginName); + if (plugin) + return { + module: plugin + }; + } } catch (err) { - console.error(`Module ${pluginName} not found.`); + console.error(`Module ${JSON.stringify(pluginName)} not found.`); + console.debug(err); } } } diff --git a/src/styles/default.sass b/src/styles/default.sass index 11ca402..f549476 100644 --- a/src/styles/default.sass +++ b/src/styles/default.sass @@ -1,5 +1,4 @@ @import includes/vars -//@import includes/w3-math @import includes/highlighjs body @@ -24,6 +23,19 @@ img max-width: 100% height: auto +hr + border: 0.1em solid $inactive + +kbd + border: 1px solid $inactive + padding: 0.1em + background-color: $backgroundSecondary + box-shadow: 0 0.1em 0.1em $inactive + +figcaption + margin: 0.5em + font-style: italic + .page page-break-before: always page-break-after: always @@ -31,3 +43,6 @@ img .newpage page-break-after: always + +.text-align-center + text-align: center diff --git a/src/styles/includes/vars.sass b/src/styles/includes/vars.sass index 5fc99c1..a67d81d 100644 --- a/src/styles/includes/vars.sass +++ b/src/styles/includes/vars.sass @@ -1 +1,2 @@ $backgroundSecondary: #eee +$inactive: #444 diff --git a/src/styles/includes/w3-math.css b/src/styles/includes/w3-math.css deleted file mode 100644 index 8018d79..0000000 --- a/src/styles/includes/w3-math.css +++ /dev/null @@ -1,380 +0,0 @@ -math -{line-height:1.3em; - text-indent:0;} -math[display="block"] -{display:block; - text-align:center; - page-break-inside:avoid;} -mfrac -{display:inline-table; - white-space:nowrap; - border-collapse:collapse; - text-align:center; - vertical-align:0.9em; - margin:0 2px; - font-size:1em;} -mfrac > * -{line-height:1.3em; - font-size:0.9em;} -mfrac > *:first-child -{display:inline-table; - vertical-align:text-bottom;} -mfrac > * + * -{border-top:solid thin; - display:table-row;} -mfrac[linethickness="0"] > * + * -{border-top:none;} -mfrac[linethickness="2"] > * + *, mfrac[linethickness="thick"] > * + * -{border-top:solid medium;} -mfrac[numalign="left"] > *:first-child, mfrac[denalign="left"] > * + * -{text-align:left;} -mfrac[numalign="right"] > *:first-child, mfrac[denalign="right"] > * + * -{text-align:right;} -msub, msup, msubsup, mmultiscripts -{display:inline-table; - line-height:0.4em;} -msubsup, msup, mmultiscripts -{margin-top:0.4ex; - table-baseline:2;} -msubsup, msub, mmultiscripts -{margin-bottom:0.4ex;} -msubsup, msup -{direction:rtl;} -msub > * -{display:table-row;} -none -{content:"\A0";} -msubsup > *, msup > * -{display:table-row; - direction:ltr; - text-align:left;} -mmultiscripts > * -{display:none;} -mmultiscripts > *:first-child, mmultiscripts > mprescripts + * -{display:table-row;} -mmultiscripts > mprescripts + * + * -{display:table-header-group;} -msub > *:first-child:after, msub > * + *:before, -msubsup > *:first-child:before, msup > *:first-child:before, -mmultiscripts > *:first-child:before -{display:table-cell; - content:"\A0";} -msubsup > * + * + *, msup > * + * -{display:table-header-group;} -msub > * + *, msup > * + *, msubsup > * + *, munder > * + *, mover > * + *, munderover > * + *, -mmultiscripts > * + * -{font-size:0.7em;} -munder, munderover, mover -{display:inline-table; - margin:1px; - text-align:center;} -munder > *, munderover > *, mover > * -{display:table-row;} -mover > * + *, munderover > * + * + * -{display:table-header-group;} -mover, munderover -{table-baseline:2;} -msqrt -{display:inline-block; - margin:1px 0 1px 22px; - border-top:solid 1px; - border-left:groove 2px; - padding:2px 5px 0 0;} -msqrt:before -{display:inline-block; - vertical-align:bottom; - content:''; - width:22px; - height:14px; - background-repeat:no-repeat; - margin:0 3px 0 -22px;} -msqrt:before, mroot > * + *:after -{background-image:url("");} -mroot -{display:inline-table; - direction:rtl;} -mroot > * -{display:table-cell; - direction:ltr; - text-align:left;} -mroot > *:first-child -{border-top:solid 1px; - border-left:groove 2px; - padding:2px 5px 0 3px;} -mroot > * + * -{vertical-align:bottom; - text-align:right; - font-size:0.7em; - line-height:1em;} -mroot > * + *:after -{display:block; - content:""; - width:22px; - height:14px; - margin-right:-1px; - margin-left:auto;} -mfenced -{display:inline-table; - border-collapse:separate; - border-spacing:0.2ex 0; - white-space:nowrap; - margin:1px;} -mfenced > *:first-child -{display:table-row;} -mfenced > * + * -{display:none;} -mfenced > *:before, mfenced > *:after -{display:table-cell; - content:"\A0"; - background-repeat:no-repeat; - background-size:100% 100%;} -mfenced > *:before -{background-image:url("");} -mfenced > *:after -{background-image:url("");} -mfenced[open="["] > *:before -{border-style:solid; - border-width:1px 0 1px 1px; - background-image:none;} -mfenced[close="]"] > *:after -{border-style:solid; - border-width:1px 1px 1px 0; - background-image:none;} -mfenced[open="\2016"] > *:before -{border-style:double; - border-width:0 0 0 3px; - background-image:none;} -mfenced[close="\2016"] > *:after -{border-style:double; - border-width:0 3px 0 0; - background-image:none;} -mfenced[open="\27e6"] > *:before -{border-style:double; - border-width:2px 0 2px 3px; - background-image:none;} -mfenced[close="\27e7"] > *:after -{border-style:double; - border-width:2px 3px 2px 0; - background-image:none;} -mfenced[open="|"] > *:before -{border-style:solid; - border-width:0 0 0 1px; - background-image:none;} -mfenced[close="|"] > *:after -{border-style:solid; - border-width:0 1px 0 0; - background-image:none;} -mfenced[open="\230a"] > *:before -{border-style:solid; - border-width:0 0 1px 1px; - background-image:none;} -mfenced[close="\230b"] > *:after -{border-style:solid; - border-width:0 1px 1px 0; - background-image:none;} -mfenced[open="\2308"] > *:before -{border-style:solid; - border-width:1px 0 0 1px; - background-image:none;} -mfenced[close="\2309"] > *:after -{border-style:solid; - border-width:1px 1px 0 0; - background-image:none;} -mfenced[open="{"] > *:before -{content:"\A0\A0"; - background-image:url("");} -mfenced[close="}"] > *:after -{content:"\A0\A0"; - background-image:url("");} -mfenced[open="\27e8"] > *:before, mfenced[open="\2329"] > *:before -{background-image:url("");} -mfenced[close="\27e9"] > *:after, mfenced[close="\232A"] > *:after -{background-image:url("");} -mfenced[open=""] > *:before, mfenced[close=""] > *:after -{content:normal;} -mover > * + mo[fence="true"], munder > * + mo[fence="true"] -{content:"\A0"; - line-height:1ex; - background-size:100% 100%; - background-image:url("");} -munder > * + mo[fence="true"] -{background-image:url("");} -menclose -{display:inline-table; - border-collapse:separate; - border-spacing:0.4ex 0;} -menclose[notation="top"] -{border-top:solid thin;} -menclose[notation="bottom"] -{border-bottom:solid thin;} -menclose[notation="right"] -{border-right:solid thin;} -menclose[notation="left"] -{border-left:solid thin;} -menclose[notation="box"] -{border:solid thin;} -menclose[notation="horizontalstrike"] -{text-decoration:line-through;} -mtable -{display:inline-table; - line-height:1.5em; - text-align:center; - vertical-align:middle;} -mtr -{display:table-row;} -mtd -{display:table-cell; - padding:0 0.5ex;} -mtable[columnalign="left"], mtr[columnalign="left"], mtd[columnalign="left"] -{text-align:left;} -mtable[columnalign="right"], mtr[columnalign="right"], mtd[columnalign="right"] -{text-align:right;} -mtable[rowalign="top"] mtd, mtable mtr[rowalign="top"] mtd, -math mtable mtr mtd[rowalign="top"] -{vertical-align:top} -mtable[rowalign="bottom"] mtd, mtable mtr[rowalign="bottom"] mtd, -math mtable mtr mtd[rowalign="bottom"] -{vertical-align:bottom} -mtable[rowalign="center"] mtd, mtable mtr[rowalign="center"] mtd, -math mtable mtr mtd[rowalign="center"] -{vertical-align:middle} -mtable[frame="solid"] -{border:solid thin;} -mtable[frame="dashed"] -{border:dashed thin;} -mtable[rowlines="solid"], mtable[rowlines="dashed"], -mtable[columnlines="solid"], mtable[columnlines="dashed"] -{border-collapse:collapse;} -mtable[rowlines="solid"] > mtr + mtr -{border-top:solid thin;} -mtable[rowlines="dashed"] > mtr + mtr -{border-top:dashed thin;} -mtable[columnlines="solid"] > mtr > mtd + mtd -{border-left:solid thin;} -mtable[columnlines="dashed"] > mtr > mtd + mtd -{border-left:dashed thin;} -mspace[linebreak="goodbreak"]:before -{content:"\200B"; - white-space:normal;} -mspace[linebreak="newline"]:before, mspace[linebreak="indentingnewline"]:before -{content:"\000A"; - white-space:pre;} -mspace[width]:before -{content:normal;} -mspace[width="verythinmathspace"] -{padding:0 0.05em;} -mspace[width="thinmathspace"] -{padding:0 0.08em;} -mspace[width="mediummathspace"] -{padding:0 0.11em;} -mspace[width="thickmathspace"] -{padding:0 0.14em;} -mspace[width="verythickmathspace"] -{padding:0 0.17em;} -mo[largeop="true"] -{font-size:1.3em; - vertical-align:-0.1ex;} -mo[form="infix"], * + mo -{padding:0 0.3ex;} -mo[form="prefix"] -{padding:0 0 0 0.5ex;} -mo[form="postfix"] -{padding:0 0.5ex 0 0;} -mo[fence="true"], mo[separator="true"] -{padding:0;} -mi[mathvariant="bold"], mi[mathvariant="bold-italic"], mi[mathvariant="bold-sans-serif"], -mi[mathvariant="sans-serif-bold-italic"],mn[mathvariant="bold"], mn[mathvariant="bold-italic"], -mn[mathvariant="bold-sans-serif"], mn[mathvariant="sans-serif-bold-italic"],mo[mathvariant="bold"], -mo[mathvariant="bold-italic"], mo[mathvariant="bold-sans-serif"], mo[mathvariant="sans-serif-bold-italic"], -ms[mathvariant="bold"], ms[mathvariant="bold-italic"], ms[mathvariant="bold-sans-serif"], -ms[mathvariant="sans-serif-bold-italic"],mtext[mathvariant="bold"], mtext[mathvariant="bold-italic"], -mtext[mathvariant="bold-sans-serif"], mtext[mathvariant="sans-serif-bold-italic"] -{font-weight:bold; - font-style:normal;} -mi[mathvariant="monospace"], mn[mathvariant="monospace"],mo[mathvariant="monospace"], -ms[mathvariant="monospace"],mtext[mathvariant="monospace"] -{font-family:monospace; - font-style:normal;} -mi[mathvariant="sans-serif"], mi[mathvariant="bold-sans-serif"], mi[mathvariant="bold-sans-serif"], -mi[mathvariant="sans-serif-italic"], mi[mathvariant="sans-serif-bold-italic"],mn[mathvariant="bold-sans-serif"], -mn[mathvariant="sans-serif"], mn[mathvariant="bold-sans-serif"], mn[mathvariant="sans-serif-italic"], -mn[mathvariant="sans-serif-bold-italic"], mo[mathvariant="sans-serif"], mo[mathvariant="bold-sans-serif"], -mo[mathvariant="bold-sans-serif"], mo[mathvariant="sans-serif-italic"], mo[mathvariant="sans-serif-bold-italic"], -ms[mathvariant="sans-serif"], ms[mathvariant="bold-sans-serif"], ms[mathvariant="bold-sans-serif"], -ms[mathvariant="sans-serif-italic"], ms[mathvariant="sans-serif-bold-italic"], mtext[mathvariant="sans-serif"], -mtext[mathvariant="bold-sans-serif"], mtext[mathvariant="bold-sans-serif"], mtext[mathvariant="sans-serif-italic"], -mtext[mathvariant="sans-serif-bold-italic"] -{font-family:sans-serif; - font-style:normal;} -mi, mi[mathvariant="italic"], mi[mathvariant="bold-italic"],mi[mathvariant="sans-serif-italic"], -mi[mathvariant="sans-serif-bold-italic"],mn[mathvariant="italic"], mn[mathvariant="bold-italic"], -mn[mathvariant="sans-serif-italic"], mn[mathvariant="sans-serif-bold-italic"],mo[mathvariant="italic"], -mo[mathvariant="bold-italic"],mo[mathvariant="sans-serif-italic"], mo[mathvariant="sans-serif-bold-italic"], -ms[mathvariant="italic"], ms[mathvariant="bold-italic"],ms[mathvariant="sans-serif-italic"], -ms[mathvariant="sans-serif-bold-italic"],mtext[mathvariant="italic"], mtext[mathvariant="bold-italic"], -mtext[mathvariant="sans-serif-italic"], mtext[mathvariant="sans-serif-bold-italic"] -{font-style:italic;} -mi[mathvariant="normal"], mn[mathvariant="normal"], mo[mathvariant="normal"], -ms[mathvariant="normal"], mtext[mathvariant="normal"] -{font-style:normal;} -ms:before, ms:after -{content:"\0022"} -ms[lquote]:before -{content:attr(lquote)} -ms[rquote]:after -{content:attr(rquote)} -mphantom -{visibility:hidden} -merror -{outline:solid thin red} -merror:before -{content:"Error: "} -mrow -{white-space:nowrap;} -math[display='block'] -{display:block; - margin:1em 0 1em 3em;} -mstack, mlongdiv -{display:inline-table; - font-family:monospace;} -mstack -{text-align:right; - border-collapse:collapse;} -mstack[align='top'], mlongdiv[align='top'] -{vertical-align:top;} -mstack[align='bottom'], mlongdiv[align='bottom'] -{vertical-align:bottom;} -mstack[align='center'], mlongdiv[align='center'] -{vertical-align:middle;} -msline -{display:block; - border-bottom:solid thin;} -mstack > *, mlongdiv > mn:first-child -{display:table-row;} -mlongdiv > *:first-child + * -{display:table-cell;} -mlongdiv > *:first-child + * + * -{border-top:solid thin;} -mlongdiv > *:first-child:before -{display:table-cell; - content:'\a0';} -mlongdiv > *:first-child + *:after -{content:')';} -mlongdiv > * -{display:block;} -mscarry -{display:none;} -maction > * + * -{display:none;} -maction[actiontype="tooltip"]:focus > * + *, -maction[actiontype="tooltip"]:hover > * + * -{position:fixed; - display:block; - top:0; - left:0; - background-color:InfoBackground; - color:InfoText; - padding:0.5ex; - border:solid 1px;} -annotation, annotation-xml -{display:none;}