aboutsummaryrefslogtreecommitdiffstats
path: root/web_src/js/markup
diff options
context:
space:
mode:
author6543 <6543@obermui.de>2021-05-07 10:43:41 +0200
committerGitHub <noreply@github.com>2021-05-07 10:43:41 +0200
commit640066840e23367d9d13e92d786b877448ae9329 (patch)
treee419f42d40a904b0b16302f3710914902f5366c0 /web_src/js/markup
parent9b5185d3cc0bd63d7387655bedaeea6340695d95 (diff)
downloadgitea-640066840e23367d9d13e92d786b877448ae9329.tar.gz
gitea-640066840e23367d9d13e92d786b877448ae9329.zip
Use a generic markup class to display externally rendered files and diffs (#15735)
* creates and implements generic markup less class * How to give custom CSS to externally rendered html * Clarifies sources of CSS styling of markup * further clarification of sources of markup styling * rename _markdown to _markup * remove defunct import * fix orphaned reference * Update docs/content/doc/advanced/external-renderers.en-us.md * more renames markdown -> markup * do not suggest less customization * add back tokens * fix class whitespace, remove useless if-clause * remove unused csv-data rules * use named exports and rename functions * sort imports Co-authored-by: HarvsG <11440490+HarvsG@users.noreply.github.com> Co-authored-by: techknowlogick <techknowlogick@gitea.io> Co-authored-by: silverwind <me@silverwind.io>
Diffstat (limited to 'web_src/js/markup')
-rw-r--r--web_src/js/markup/anchors.js32
-rw-r--r--web_src/js/markup/content.js5
-rw-r--r--web_src/js/markup/mermaid.js56
3 files changed, 93 insertions, 0 deletions
diff --git a/web_src/js/markup/anchors.js b/web_src/js/markup/anchors.js
new file mode 100644
index 0000000000..cc2ed5db78
--- /dev/null
+++ b/web_src/js/markup/anchors.js
@@ -0,0 +1,32 @@
+import {svg} from '../svg.js';
+
+const headingSelector = '.markup h1, .markup h2, .markup h3, .markup h4, .markup h5, .markup h6';
+
+function scrollToAnchor() {
+ if (document.querySelector(':target')) return;
+ if (!window.location.hash || window.location.hash.length <= 1) return;
+ const id = decodeURIComponent(window.location.hash.substring(1));
+ const el = document.getElementById(`user-content-${id}`);
+ if (el) {
+ el.scrollIntoView();
+ } else if (id.startsWith('user-content-')) { // compat for links with old 'user-content-' prefixed hashes
+ const el = document.getElementById(id);
+ if (el) el.scrollIntoView();
+ }
+}
+
+export function initMarkupAnchors() {
+ if (!document.querySelector('.markup')) return;
+
+ for (const heading of document.querySelectorAll(headingSelector)) {
+ const originalId = heading.id.replace(/^user-content-/, '');
+ const a = document.createElement('a');
+ a.classList.add('anchor');
+ a.setAttribute('href', `#${encodeURIComponent(originalId)}`);
+ a.innerHTML = svg('octicon-link');
+ heading.prepend(a);
+ }
+
+ scrollToAnchor();
+ window.addEventListener('hashchange', scrollToAnchor);
+}
diff --git a/web_src/js/markup/content.js b/web_src/js/markup/content.js
new file mode 100644
index 0000000000..f06c9908f2
--- /dev/null
+++ b/web_src/js/markup/content.js
@@ -0,0 +1,5 @@
+import {renderMermaid} from './mermaid.js';
+
+export async function renderMarkupContent() {
+ await renderMermaid(document.querySelectorAll('code.language-mermaid'));
+}
diff --git a/web_src/js/markup/mermaid.js b/web_src/js/markup/mermaid.js
new file mode 100644
index 0000000000..d0aefd1aff
--- /dev/null
+++ b/web_src/js/markup/mermaid.js
@@ -0,0 +1,56 @@
+const MAX_SOURCE_CHARACTERS = 5000;
+
+function displayError(el, err) {
+ el.closest('pre').classList.remove('is-loading');
+ const errorNode = document.createElement('div');
+ errorNode.setAttribute('class', 'ui message error markup-block-error mono');
+ errorNode.textContent = err.str || err.message || String(err);
+ el.closest('pre').before(errorNode);
+}
+
+export async function renderMermaid(els) {
+ if (!els || !els.length) return;
+
+ const mermaid = await import(/* webpackChunkName: "mermaid" */'mermaid');
+
+ mermaid.initialize({
+ mermaid: {
+ startOnLoad: false,
+ },
+ flowchart: {
+ useMaxWidth: true,
+ htmlLabels: false,
+ },
+ theme: 'neutral',
+ securityLevel: 'strict',
+ });
+
+ for (const el of els) {
+ if (el.textContent.length > MAX_SOURCE_CHARACTERS) {
+ displayError(el, new Error(`Mermaid source of ${el.textContent.length} characters exceeds the maximum allowed length of ${MAX_SOURCE_CHARACTERS}.`));
+ continue;
+ }
+
+ let valid;
+ try {
+ valid = mermaid.parse(el.textContent);
+ } catch (err) {
+ displayError(el, err);
+ }
+
+ if (!valid) {
+ el.closest('pre').classList.remove('is-loading');
+ continue;
+ }
+
+ try {
+ mermaid.init(undefined, el, (id) => {
+ const svg = document.getElementById(id);
+ svg.classList.add('mermaid-chart');
+ svg.closest('pre').replaceWith(svg);
+ });
+ } catch (err) {
+ displayError(el, err);
+ }
+ }
+}