2017-08-17 22:00:37 +05:30
|
|
|
<script>
|
2020-11-24 15:15:51 +05:30
|
|
|
/* eslint-disable vue/no-v-html */
|
2020-01-01 13:55:28 +05:30
|
|
|
import katex from 'katex';
|
2021-03-11 19:13:27 +05:30
|
|
|
import marked from 'marked';
|
2021-01-03 14:25:43 +05:30
|
|
|
import { sanitize } from '~/lib/dompurify';
|
2021-06-08 01:23:25 +05:30
|
|
|
import { hasContent } from '~/lib/utils/text_utility';
|
2018-12-13 13:39:08 +05:30
|
|
|
import Prompt from './prompt.vue';
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
const renderer = new marked.Renderer();
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
/*
|
2017-08-17 22:00:37 +05:30
|
|
|
Regex to match KaTex blocks.
|
|
|
|
|
|
|
|
Supports the following:
|
|
|
|
|
|
|
|
\begin{equation}<math>\end{equation}
|
|
|
|
$$<math>$$
|
|
|
|
inline $<math>$
|
|
|
|
|
|
|
|
The matched text then goes through the KaTex renderer & then outputs the HTML
|
|
|
|
*/
|
2018-12-13 13:39:08 +05:30
|
|
|
const katexRegexString = `(
|
2017-08-17 22:00:37 +05:30
|
|
|
^\\\\begin{[a-zA-Z]+}\\s
|
|
|
|
|
|
|
|
|
^\\$\\$
|
|
|
|
|
|
|
|
|
\\s\\$(?!\\$)
|
|
|
|
)
|
2017-09-10 17:25:29 +05:30
|
|
|
((.|\\n)+?)
|
2017-08-17 22:00:37 +05:30
|
|
|
(
|
|
|
|
\\s\\\\end{[a-zA-Z]+}$
|
|
|
|
|
|
|
|
|
\\$\\$$
|
|
|
|
|
|
|
|
|
\\$
|
|
|
|
)
|
2018-12-13 13:39:08 +05:30
|
|
|
`
|
|
|
|
.replace(/\s/g, '')
|
|
|
|
.trim();
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2021-04-29 21:17:54 +05:30
|
|
|
function deHTMLify(t) {
|
|
|
|
// get some specific characters back, that are allowed for KaTex rendering
|
|
|
|
const text = t.replace(/'/g, "'").replace(/</g, '<').replace(/>/g, '>');
|
|
|
|
return text;
|
|
|
|
}
|
2020-05-24 23:13:21 +05:30
|
|
|
function renderKatex(t) {
|
2018-12-13 13:39:08 +05:30
|
|
|
let text = t;
|
2020-05-24 23:13:21 +05:30
|
|
|
let numInline = 0; // number of successfull converted math formulas
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
if (typeof katex !== 'undefined') {
|
|
|
|
const katexString = text
|
|
|
|
.replace(/&/g, '&')
|
2020-04-22 19:07:51 +05:30
|
|
|
.replace(/&=&/g, '\\space=\\space') // eslint-disable-line @gitlab/require-i18n-strings
|
2018-12-13 13:39:08 +05:30
|
|
|
.replace(/<(\/?)em>/g, '_');
|
|
|
|
const regex = new RegExp(katexRegexString, 'gi');
|
|
|
|
const matchLocation = katexString.search(regex);
|
|
|
|
const numberOfMatches = katexString.match(regex);
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
if (numberOfMatches && numberOfMatches.length !== 0) {
|
2020-05-24 23:13:21 +05:30
|
|
|
let matches = regex.exec(katexString);
|
2018-12-13 13:39:08 +05:30
|
|
|
if (matchLocation > 0) {
|
2020-05-24 23:13:21 +05:30
|
|
|
numInline += 1;
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
while (matches !== null) {
|
2020-05-24 23:13:21 +05:30
|
|
|
try {
|
2021-04-29 21:17:54 +05:30
|
|
|
const renderedKatex = katex.renderToString(deHTMLify(matches[0].replace(/\$/g, '')));
|
2020-05-24 23:13:21 +05:30
|
|
|
text = `${text.replace(matches[0], ` ${renderedKatex}`)}`;
|
|
|
|
} catch {
|
|
|
|
numInline -= 1;
|
|
|
|
}
|
2018-12-13 13:39:08 +05:30
|
|
|
matches = regex.exec(katexString);
|
2017-08-17 22:00:37 +05:30
|
|
|
}
|
2018-12-13 13:39:08 +05:30
|
|
|
} else {
|
2020-05-24 23:13:21 +05:30
|
|
|
try {
|
2021-04-29 21:17:54 +05:30
|
|
|
text = katex.renderToString(deHTMLify(matches[2]));
|
2020-05-24 23:13:21 +05:30
|
|
|
} catch (error) {
|
|
|
|
numInline -= 1;
|
|
|
|
}
|
2017-08-17 22:00:37 +05:30
|
|
|
}
|
|
|
|
}
|
2018-12-13 13:39:08 +05:30
|
|
|
}
|
2020-05-24 23:13:21 +05:30
|
|
|
return [text, numInline > 0];
|
|
|
|
}
|
2021-03-08 18:12:59 +05:30
|
|
|
renderer.paragraph = (t) => {
|
2020-05-24 23:13:21 +05:30
|
|
|
const [text, inline] = renderKatex(t);
|
2018-12-13 13:39:08 +05:30
|
|
|
return `<p class="${inline ? 'inline-katex' : ''}">${text}</p>`;
|
|
|
|
};
|
2021-03-08 18:12:59 +05:30
|
|
|
renderer.listitem = (t) => {
|
2020-05-24 23:13:21 +05:30
|
|
|
const [text, inline] = renderKatex(t);
|
|
|
|
return `<li class="${inline ? 'inline-katex' : ''}">${text}</li>`;
|
|
|
|
};
|
2021-06-08 01:23:25 +05:30
|
|
|
renderer.originalImage = renderer.image;
|
|
|
|
|
|
|
|
renderer.image = function image(href, title, text) {
|
|
|
|
const attachmentHeader = `attachment:`; // eslint-disable-line @gitlab/require-i18n-strings
|
|
|
|
|
|
|
|
if (!this.attachments || !href.startsWith(attachmentHeader)) {
|
|
|
|
return this.originalImage(href, title, text);
|
|
|
|
}
|
|
|
|
|
|
|
|
let img = ``;
|
|
|
|
const filename = href.substring(attachmentHeader.length);
|
|
|
|
|
|
|
|
if (hasContent(filename)) {
|
|
|
|
const attachment = this.attachments[filename];
|
|
|
|
|
|
|
|
if (attachment) {
|
|
|
|
const imageType = Object.keys(attachment)[0];
|
|
|
|
|
|
|
|
if (hasContent(imageType)) {
|
|
|
|
const data = attachment[imageType];
|
|
|
|
const inlined = `data:${imageType};base64,${data}"`; // eslint-disable-line @gitlab/require-i18n-strings
|
|
|
|
img = this.originalImage(inlined, title, text);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!hasContent(img)) {
|
|
|
|
return this.originalImage(href, title, text);
|
|
|
|
}
|
|
|
|
|
|
|
|
return sanitize(img);
|
|
|
|
};
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
marked.setOptions({
|
|
|
|
renderer,
|
|
|
|
});
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
export default {
|
|
|
|
components: {
|
|
|
|
prompt: Prompt,
|
|
|
|
},
|
|
|
|
props: {
|
|
|
|
cell: {
|
|
|
|
type: Object,
|
|
|
|
required: true,
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
2018-12-13 13:39:08 +05:30
|
|
|
},
|
|
|
|
computed: {
|
|
|
|
markdown() {
|
2021-06-08 01:23:25 +05:30
|
|
|
renderer.attachments = this.cell.attachments;
|
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
return sanitize(marked(this.cell.source.join('').replace(/\\/g, '\\\\')), {
|
2020-01-01 13:55:28 +05:30
|
|
|
// allowedTags from GitLab's inline HTML guidelines
|
|
|
|
// https://docs.gitlab.com/ee/user/markdown.html#inline-html
|
2020-10-24 23:57:45 +05:30
|
|
|
ALLOWED_TAGS: [
|
|
|
|
'a',
|
|
|
|
'abbr',
|
|
|
|
'b',
|
|
|
|
'blockquote',
|
|
|
|
'br',
|
|
|
|
'code',
|
|
|
|
'dd',
|
|
|
|
'del',
|
|
|
|
'div',
|
|
|
|
'dl',
|
|
|
|
'dt',
|
|
|
|
'em',
|
2020-01-01 13:55:28 +05:30
|
|
|
'h1',
|
|
|
|
'h2',
|
|
|
|
'h3',
|
|
|
|
'h4',
|
|
|
|
'h5',
|
|
|
|
'h6',
|
2020-10-24 23:57:45 +05:30
|
|
|
'hr',
|
2020-01-01 13:55:28 +05:30
|
|
|
'i',
|
|
|
|
'img',
|
|
|
|
'ins',
|
|
|
|
'kbd',
|
2020-10-24 23:57:45 +05:30
|
|
|
'li',
|
|
|
|
'ol',
|
|
|
|
'p',
|
|
|
|
'pre',
|
2020-01-01 13:55:28 +05:30
|
|
|
'q',
|
|
|
|
'rp',
|
2020-10-24 23:57:45 +05:30
|
|
|
'rt',
|
|
|
|
'ruby',
|
2020-01-01 13:55:28 +05:30
|
|
|
's',
|
2020-10-24 23:57:45 +05:30
|
|
|
'samp',
|
2020-01-01 13:55:28 +05:30
|
|
|
'span',
|
2020-10-24 23:57:45 +05:30
|
|
|
'strike',
|
|
|
|
'strong',
|
|
|
|
'sub',
|
2020-01-01 13:55:28 +05:30
|
|
|
'summary',
|
2020-10-24 23:57:45 +05:30
|
|
|
'sup',
|
|
|
|
'table',
|
|
|
|
'tbody',
|
|
|
|
'td',
|
|
|
|
'tfoot',
|
|
|
|
'th',
|
|
|
|
'thead',
|
|
|
|
'tr',
|
|
|
|
'tt',
|
|
|
|
'ul',
|
|
|
|
'var',
|
2020-01-01 13:55:28 +05:30
|
|
|
],
|
2020-10-24 23:57:45 +05:30
|
|
|
ALLOWED_ATTR: ['class', 'style', 'href', 'src'],
|
2021-06-02 17:11:27 +05:30
|
|
|
ALLOW_DATA_ATTR: false,
|
2018-12-13 13:39:08 +05:30
|
|
|
});
|
2017-08-17 22:00:37 +05:30
|
|
|
},
|
2018-12-13 13:39:08 +05:30
|
|
|
},
|
|
|
|
};
|
2017-08-17 22:00:37 +05:30
|
|
|
</script>
|
|
|
|
|
2018-03-17 18:26:18 +05:30
|
|
|
<template>
|
|
|
|
<div class="cell text-cell">
|
|
|
|
<prompt />
|
2019-02-15 15:39:39 +05:30
|
|
|
<div class="markdown" v-html="markdown"></div>
|
2018-03-17 18:26:18 +05:30
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
2017-08-17 22:00:37 +05:30
|
|
|
<style>
|
2020-01-01 13:55:28 +05:30
|
|
|
/*
|
|
|
|
Importing the necessary katex stylesheet from the node_module folder rather
|
|
|
|
than copying the stylesheet into `app/assets/stylesheets/vendors` for
|
|
|
|
automatic importing via `app/assets/stylesheets/application.scss`. The reason
|
|
|
|
is that the katex stylesheet depends on many fonts that are in node_module
|
|
|
|
subfolders - moving all these fonts would make updating katex difficult.
|
|
|
|
*/
|
|
|
|
@import '~katex/dist/katex.min.css';
|
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
.markdown .katex {
|
|
|
|
display: block;
|
|
|
|
text-align: center;
|
|
|
|
}
|
2017-08-17 22:00:37 +05:30
|
|
|
|
2018-12-13 13:39:08 +05:30
|
|
|
.markdown .inline-katex .katex {
|
|
|
|
display: inline;
|
|
|
|
text-align: initial;
|
|
|
|
}
|
2017-08-17 22:00:37 +05:30
|
|
|
</style>
|