Add asciimath processing (#10)

This version sets the new asciimath parsing to be an option. It also deals with the old zoom_on_hover and enable_accessibility options a bit better.
This commit is contained in:
mcmcclur 2018-05-28 18:54:56 -04:00 committed by Sam
parent 5588d2ba65
commit 05d52f593f
4 changed files with 68 additions and 35 deletions

View File

@ -2,14 +2,13 @@ import { withPluginApi } from 'discourse/lib/plugin-api';
import loadScript from 'discourse/lib/load-script'; import loadScript from 'discourse/lib/load-script';
let initializedMathJax = false; let initializedMathJax = false;
let zoom_on_hover, enable_accessibility;
function initMathJax() { function initMathJax(opts) {
if (initializedMathJax) { return; } if (initializedMathJax) { return; }
var extensions = ["toMathML.js", "Safe.js"]; var extensions = ["toMathML.js", "Safe.js"];
if (enable_accessibility) { if (opts.enable_accessibility) {
extensions.push("[a11y]/accessibility-menu.js"); extensions.push("[a11y]/accessibility-menu.js");
} }
@ -21,7 +20,7 @@ function initMathJax() {
root: '/plugins/discourse-math/mathjax' root: '/plugins/discourse-math/mathjax'
}; };
if(zoom_on_hover) { if(opts.zoom_on_hover) {
settings.menuSettings = {zoom: "Hover"}; settings.menuSettings = {zoom: "Hover"};
settings.MathEvents = {hover: 750}; settings.MathEvents = {hover: 750};
} }
@ -29,12 +28,13 @@ function initMathJax() {
initializedMathJax = true; initializedMathJax = true;
} }
function ensureMathJax(){ function ensureMathJax(opts){
initMathJax(); initMathJax(opts);
return loadScript('/plugins/discourse-math/mathjax/MathJax.1.7.1.js'); return loadScript('/plugins/discourse-math/mathjax/MathJax.1.7.1.js');
} }
function decorate(elem, isPreview){ function decorate(elem, isPreview){
const $elem= $(elem); const $elem= $(elem);
if ($elem.data('applied-mathjax')){ if ($elem.data('applied-mathjax')){
@ -42,14 +42,20 @@ function decorate(elem, isPreview){
} }
$elem.data('applied-mathjax', true); $elem.data('applied-mathjax', true);
if($elem.hasClass('math')) {
const tag = elem.tagName === "DIV" ? "div" : "span"; const tag = elem.tagName === "DIV" ? "div" : "span";
const display = tag === "div" ? "; mode=display" : ""; const display = tag === "div" ? "; mode=display" : "";
var $mathWrapper = $(`<${tag} style="display: none;"><script type="math/tex${display}"></script></${tag}>`);
const $mathWrapper = $(`<${tag} style="display: none;"><script type="math/tex${display}"></script></${tag}>`); var $math = $mathWrapper.children();
const $math = $mathWrapper.children();
$math.html($elem.text()); $math.html($elem.text());
$elem.after($mathWrapper); $elem.after($mathWrapper);
}
else if($elem.hasClass('asciimath')) {
var $mathWrapper = $(`<span style="display: none;"><script type="math/asciimath"></script></span>`);
var $math = $mathWrapper.children();
$math.html($elem.text());
$elem.after($mathWrapper);
}
Em.run.later(this, ()=> { Em.run.later(this, ()=> {
window.MathJax.Hub.Queue(() => { window.MathJax.Hub.Queue(() => {
@ -64,35 +70,43 @@ function decorate(elem, isPreview){
}, isPreview ? 200 : 0); }, isPreview ? 200 : 0);
} }
function mathjax($elem) { function mathjax($elem, opts) {
if (!$elem || !$elem.find) { if (!$elem || !$elem.find) {
return; return;
} }
const mathElems = $elem.find('.math'); let mathElems;
if(opts.enable_asciimath) {
mathElems = $elem.find('.math, .asciimath');
}
else {
mathElems = $elem.find('.math');
}
if (mathElems.length > 0) { if (mathElems.length > 0) {
const isPreview = $elem.hasClass('d-editor-preview'); const isPreview = $elem.hasClass('d-editor-preview');
ensureMathJax().then(()=>{ ensureMathJax(opts).then(()=>{
mathElems.each((idx,elem) => decorate(elem, isPreview)); mathElems.each((idx,elem) => decorate(elem, isPreview));
}); });
} }
} }
function initializeMath(api) { function initializeMath(api, discourse_math_opts) {
api.decorateCooked(mathjax); api.decorateCooked(function(elem) {mathjax(elem,discourse_math_opts)});
} }
export default { export default {
name: "apply-math", name: "apply-math",
initialize(container) { initialize(container) {
const siteSettings = container.lookup('site-settings:main'); const siteSettings = container.lookup('site-settings:main');
zoom_on_hover = siteSettings.discourse_math_zoom_on_hover; let discourse_math_opts = {
enable_accessibility = siteSettings.discourse_math_enable_accessibility; zoom_on_hover: siteSettings.discourse_math_zoom_on_hover,
enable_accessibility: siteSettings.discourse_math_enable_accessibility,
enable_asciimath: siteSettings.discourse_math_enable_asciimath
};
if (siteSettings.discourse_math_enabled) { if (siteSettings.discourse_math_enabled) {
withPluginApi('0.5', initializeMath); withPluginApi('0.5', function(api) {initializeMath(api,discourse_math_opts)});
} }
} }
}; };

View File

@ -2,43 +2,44 @@
// //
// //
// //
function isSafeBoundary(code, md) {
if (code === 36) { function isSafeBoundary(character_code, delimiter_code, md) {
if (character_code === delimiter_code) {
return false; return false;
} }
if (md.utils.isWhiteSpace(code)) { if (md.utils.isWhiteSpace(character_code)) {
return true; return true;
} }
if (md.utils.isMdAsciiPunct(code)) { if (md.utils.isMdAsciiPunct(character_code)) {
return true; return true;
} }
if (md.utils.isPunctChar(code)) { if (md.utils.isPunctChar(character_code)) {
return true; return true;
} }
return false; return false;
} }
function inlineMath(state, silent) { function math_input(state, silent, delimiter_code) {
let pos = state.pos, let pos = state.pos,
posMax = state.posMax; posMax = state.posMax;
if (silent || state.src.charCodeAt(pos) !== 36 /* $ */ || posMax < pos+2) { if (silent || state.src.charCodeAt(pos) !== delimiter_code || posMax < pos+2) {
return false; return false;
} }
// too short // too short
if (state.src.charCodeAt(pos+1) === 36 /* $ */) { if (state.src.charCodeAt(pos+1) === delimiter_code) {
return false; return false;
} }
if (pos > 0) { if (pos > 0) {
let prev = state.src.charCodeAt(pos-1); let prev = state.src.charCodeAt(pos-1);
if (!isSafeBoundary(prev, state.md)) { if (!isSafeBoundary(prev, delimiter_code, state.md)) {
return false; return false;
} }
} }
@ -47,7 +48,7 @@ function inlineMath(state, silent) {
let found; let found;
for(let i=pos+1; i<posMax; i++) { for(let i=pos+1; i<posMax; i++) {
let code = state.src.charCodeAt(i); let code = state.src.charCodeAt(i);
if (code === 36 /* $ */ && state.src.charCodeAt(i-1) !== 92 /* \ */) { if (code === delimiter_code && state.src.charCodeAt(i-1) !== 92 /* \ */) {
found = i; found = i;
break; break;
} }
@ -59,7 +60,7 @@ function inlineMath(state, silent) {
if (found+1 <= posMax) { if (found+1 <= posMax) {
let next = state.src.charCodeAt(found+1); let next = state.src.charCodeAt(found+1);
if (next && !isSafeBoundary(next, state.md)) { if (next && !isSafeBoundary(next, delimiter_code, state.md)) {
return false; return false;
} }
} }
@ -68,11 +69,20 @@ function inlineMath(state, silent) {
let token = state.push('html_raw', '', 0); let token = state.push('html_raw', '', 0);
const escaped = state.md.utils.escapeHtml(data); const escaped = state.md.utils.escapeHtml(data);
token.content = `<span class='math'>${escaped}</span>`; let math_class = delimiter_code === 36 ? "'math'" : "'asciimath'";
token.content = `<span class=${math_class}>${escaped}</span>`;
state.pos = found+1; state.pos = found+1;
return true; return true;
} }
function inlineMath(state, silent) {
return math_input(state, silent, 36 /* $ */)
}
function asciiMath(state, silent) {
return math_input(state, silent, 37 /* % */)
}
function isBlockMarker(state, start, max, md) { function isBlockMarker(state, start, max, md) {
if (state.src.charCodeAt(start) !== 36 /* $ */) { if (state.src.charCodeAt(start) !== 36 /* $ */) {
@ -144,11 +154,16 @@ export function setup(helper) {
return; return;
} }
let enable_asciimath;
helper.registerOptions((opts, siteSettings) => { helper.registerOptions((opts, siteSettings) => {
opts.features.math = siteSettings.discourse_math_enabled; opts.features.math = siteSettings.discourse_math_enabled;
enable_asciimath = siteSettings.discourse_math_enable_asciimath;
}); });
helper.registerPlugin(md => { helper.registerPlugin(md => {
if(enable_asciimath) {
md.inline.ruler.after('escape', 'asciimath', asciiMath);
}
md.inline.ruler.after('escape', 'math', inlineMath); md.inline.ruler.after('escape', 'math', inlineMath);
md.block.ruler.after('code', 'math', blockMath, { md.block.ruler.after('code', 'math', blockMath, {
alt: ['paragraph', 'reference', 'blockquote', 'list'] alt: ['paragraph', 'reference', 'blockquote', 'list']

View File

@ -6,3 +6,4 @@ en:
discourse_math_enabled: 'Enable Discourse Math plugin (will add special processing to $ and $$ blocks)' discourse_math_enabled: 'Enable Discourse Math plugin (will add special processing to $ and $$ blocks)'
discourse_math_zoom_on_hover: 'Zoom 200% on hover' discourse_math_zoom_on_hover: 'Zoom 200% on hover'
discourse_math_enable_accessibility: 'Enable accessibility features' discourse_math_enable_accessibility: 'Enable accessibility features'
discourse_math_enable_asciimath: 'Enable asciimath (will add special processing to % delimited input)'

View File

@ -8,3 +8,6 @@ plugins:
discourse_math_enable_accessibility: discourse_math_enable_accessibility:
default: false default: false
client: true client: true
discourse_math_enable_asciimath:
default: false
client: true