Add loading spinners and mermaid error handling (#12358)
- Add loading spinners on editor and mermaid renderers - Add error handling and inline error box for mermaid - Fix Mermaid rendering by using the .init api
This commit is contained in:
parent
5e5c893555
commit
e61c09ed73
10 changed files with 148 additions and 27 deletions
|
@ -7,6 +7,7 @@ package markdown
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
|
"strings"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
"code.gitea.io/gitea/modules/log"
|
"code.gitea.io/gitea/modules/log"
|
||||||
|
@ -57,13 +58,33 @@ func render(body []byte, urlPrefix string, metas map[string]string, wikiMarkdown
|
||||||
chromahtml.PreventSurroundingPre(true),
|
chromahtml.PreventSurroundingPre(true),
|
||||||
),
|
),
|
||||||
highlighting.WithWrapperRenderer(func(w util.BufWriter, c highlighting.CodeBlockContext, entering bool) {
|
highlighting.WithWrapperRenderer(func(w util.BufWriter, c highlighting.CodeBlockContext, entering bool) {
|
||||||
language, _ := c.Language()
|
|
||||||
if language == nil {
|
|
||||||
language = []byte("text")
|
|
||||||
}
|
|
||||||
if entering {
|
if entering {
|
||||||
|
language, _ := c.Language()
|
||||||
|
if language == nil {
|
||||||
|
language = []byte("text")
|
||||||
|
}
|
||||||
|
|
||||||
|
languageStr := string(language)
|
||||||
|
|
||||||
|
preClasses := []string{}
|
||||||
|
if languageStr == "mermaid" {
|
||||||
|
preClasses = append(preClasses, "is-loading")
|
||||||
|
}
|
||||||
|
|
||||||
|
if len(preClasses) > 0 {
|
||||||
|
_, err := w.WriteString(`<pre class="` + strings.Join(preClasses, " ") + `">`)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
_, err := w.WriteString(`<pre>`)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// include language-x class as part of commonmark spec
|
// include language-x class as part of commonmark spec
|
||||||
_, err := w.WriteString("<pre><code class=\"chroma language-" + string(language) + "\">")
|
_, err := w.WriteString(`<code class="chroma language-` + string(language) + `">`)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
|
@ -38,6 +38,7 @@ func NewSanitizer() {
|
||||||
func ReplaceSanitizer() {
|
func ReplaceSanitizer() {
|
||||||
sanitizer.policy = bluemonday.UGCPolicy()
|
sanitizer.policy = bluemonday.UGCPolicy()
|
||||||
// For Chroma markdown plugin
|
// For Chroma markdown plugin
|
||||||
|
sanitizer.policy.AllowAttrs("class").Matching(regexp.MustCompile(`^is-loading$`)).OnElements("pre")
|
||||||
sanitizer.policy.AllowAttrs("class").Matching(regexp.MustCompile(`^(chroma )?language-[\w-]+$`)).OnElements("code")
|
sanitizer.policy.AllowAttrs("class").Matching(regexp.MustCompile(`^(chroma )?language-[\w-]+$`)).OnElements("code")
|
||||||
|
|
||||||
// Checkboxes
|
// Checkboxes
|
||||||
|
|
|
@ -41,9 +41,7 @@
|
||||||
data-markdown-file-exts="{{.MarkdownFileExts}}"
|
data-markdown-file-exts="{{.MarkdownFileExts}}"
|
||||||
data-line-wrap-extensions="{{.LineWrapExtensions}}">
|
data-line-wrap-extensions="{{.LineWrapExtensions}}">
|
||||||
{{.FileContent}}</textarea>
|
{{.FileContent}}</textarea>
|
||||||
<div class="editor-loading">
|
<div class="editor-loading is-loading"></div>
|
||||||
{{.i18n.Tr "loading"}}
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<div class="ui bottom attached tab segment markdown" data-tab="preview">
|
<div class="ui bottom attached tab segment markdown" data-tab="preview">
|
||||||
{{.i18n.Tr "loading"}}
|
{{.i18n.Tr "loading"}}
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import {renderMermaid} from './mermaid.js';
|
import {renderMermaid} from './mermaid.js';
|
||||||
|
|
||||||
export default async function renderMarkdownContent() {
|
export default async function renderMarkdownContent() {
|
||||||
await renderMermaid(document.querySelectorAll('.language-mermaid'));
|
await renderMermaid(document.querySelectorAll('code.language-mermaid'));
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,23 +1,56 @@
|
||||||
import {random} from '../utils.js';
|
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 markdown-block-error mono');
|
||||||
|
errorNode.textContent = err.str || err.message || String(err);
|
||||||
|
el.closest('pre').before(errorNode);
|
||||||
|
}
|
||||||
|
|
||||||
export async function renderMermaid(els) {
|
export async function renderMermaid(els) {
|
||||||
if (!els || !els.length) return;
|
if (!els || !els.length) return;
|
||||||
|
|
||||||
const {mermaidAPI} = await import(/* webpackChunkName: "mermaid" */'mermaid');
|
const mermaid = await import(/* webpackChunkName: "mermaid" */'mermaid');
|
||||||
|
|
||||||
mermaidAPI.initialize({
|
mermaid.initialize({
|
||||||
startOnLoad: false,
|
mermaid: {
|
||||||
|
startOnLoad: false,
|
||||||
|
},
|
||||||
|
flowchart: {
|
||||||
|
useMaxWidth: true,
|
||||||
|
htmlLabels: false,
|
||||||
|
},
|
||||||
theme: 'neutral',
|
theme: 'neutral',
|
||||||
securityLevel: 'strict',
|
securityLevel: 'strict',
|
||||||
});
|
});
|
||||||
|
|
||||||
for (const el of els) {
|
for (const el of els) {
|
||||||
mermaidAPI.render(`mermaid-${random(12)}`, el.textContent, (svg, bindFunctions) => {
|
if (el.textContent.length > MAX_SOURCE_CHARACTERS) {
|
||||||
const div = document.createElement('div');
|
displayError(el, new Error(`Mermaid source of ${el.textContent.length} characters exceeds the maximum allowed length of ${MAX_SOURCE_CHARACTERS}.`));
|
||||||
div.classList.add('mermaid-chart');
|
continue;
|
||||||
div.innerHTML = svg;
|
}
|
||||||
if (typeof bindFunctions === 'function') bindFunctions(div);
|
|
||||||
el.closest('pre').replaceWith(div);
|
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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -495,10 +495,20 @@
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.mermaid-chart {
|
.markdown-block-error {
|
||||||
display: flex;
|
margin-bottom: 0 !important;
|
||||||
justify-content: center;
|
border-bottom-left-radius: 0 !important;
|
||||||
align-items: center;
|
border-bottom-right-radius: 0 !important;
|
||||||
padding: 1rem;
|
box-shadow: none !important;
|
||||||
margin: 1rem 0;
|
font-size: 85% !important;
|
||||||
|
white-space: pre !important;
|
||||||
|
padding: .5rem 1rem !important;
|
||||||
|
text-align: left !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.markdown-block-error + pre {
|
||||||
|
border-top: none !important;
|
||||||
|
margin-top: 0 !important;
|
||||||
|
border-top-left-radius: 0 !important;
|
||||||
|
border-top-right-radius: 0 !important;
|
||||||
}
|
}
|
||||||
|
|
34
web_src/less/features/animations.less
Normal file
34
web_src/less/features/animations.less
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
@keyframes isloadingspin {
|
||||||
|
0% { transform: translate(-50%, -50%) rotate(0deg); }
|
||||||
|
100% { transform: translate(-50%, -50%) rotate(360deg); }
|
||||||
|
}
|
||||||
|
|
||||||
|
.is-loading {
|
||||||
|
background: transparent !important;
|
||||||
|
color: transparent !important;
|
||||||
|
border: transparent !important;
|
||||||
|
pointer-events: none !important;
|
||||||
|
position: relative !important;
|
||||||
|
overflow: hidden !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.is-loading:after {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
display: block;
|
||||||
|
width: 4rem;
|
||||||
|
height: 4rem;
|
||||||
|
left: 50%;
|
||||||
|
top: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
animation: isloadingspin 500ms infinite linear;
|
||||||
|
border-width: 4px;
|
||||||
|
border-style: solid;
|
||||||
|
border-color: #ececec #ececec #666 #666;
|
||||||
|
border-radius: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.markdown pre.is-loading,
|
||||||
|
.editor-loading.is-loading {
|
||||||
|
height: 12rem;
|
||||||
|
}
|
|
@ -1,5 +1,7 @@
|
||||||
@import "~font-awesome/css/font-awesome.css";
|
@import "~font-awesome/css/font-awesome.css";
|
||||||
@import "./vendor/gitGraph.css";
|
@import "./vendor/gitGraph.css";
|
||||||
|
@import "./features/animations.less";
|
||||||
|
@import "./markdown/mermaid.less";
|
||||||
|
|
||||||
@import "_svg";
|
@import "_svg";
|
||||||
@import "_tribute";
|
@import "_tribute";
|
||||||
|
|
12
web_src/less/markdown/mermaid.less
Normal file
12
web_src/less/markdown/mermaid.less
Normal file
|
@ -0,0 +1,12 @@
|
||||||
|
.mermaid-chart {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 1rem;
|
||||||
|
margin: 1rem 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* mermaid's errorRenderer seems to unavoidably spew stuff into <body>, hide it */
|
||||||
|
body > div[id*="mermaid-"] {
|
||||||
|
display: none !important;
|
||||||
|
}
|
|
@ -1260,7 +1260,8 @@ input {
|
||||||
border-color: #794f31;
|
border-color: #794f31;
|
||||||
}
|
}
|
||||||
|
|
||||||
.ui.red.message {
|
.ui.red.message,
|
||||||
|
.ui.error.message {
|
||||||
background-color: rgba(80, 23, 17, .6);
|
background-color: rgba(80, 23, 17, .6);
|
||||||
color: #f9cbcb;
|
color: #f9cbcb;
|
||||||
box-shadow: 0 0 0 1px rgba(121, 71, 66, .5) inset, 0 0 0 0 transparent;
|
box-shadow: 0 0 0 1px rgba(121, 71, 66, .5) inset, 0 0 0 0 transparent;
|
||||||
|
@ -1923,3 +1924,12 @@ footer .container .links > * {
|
||||||
.mermaid-chart {
|
.mermaid-chart {
|
||||||
filter: invert(84%) hue-rotate(180deg);
|
filter: invert(84%) hue-rotate(180deg);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.is-loading:after {
|
||||||
|
border-color: #4a4c58 #4a4c58 #d7d7da #d7d7da;
|
||||||
|
}
|
||||||
|
|
||||||
|
.markdown-block-error {
|
||||||
|
border: 1px solid rgba(121, 71, 66, .5) !important;
|
||||||
|
border-bottom: none !important;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue