mirror of
				https://github.com/zadam/trilium.git
				synced 2025-11-04 13:39:01 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			205 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			205 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
import { JSDOM } from "jsdom";
 | 
						|
import shaca from "./shaca/shaca.js";
 | 
						|
import assetPath from "../services/asset_path.js";
 | 
						|
import shareRoot from "./share_root.js";
 | 
						|
import escapeHtml from "escape-html";
 | 
						|
import type SNote from "./shaca/entities/snote.js";
 | 
						|
import { t } from "i18next";
 | 
						|
 | 
						|
/**
 | 
						|
 * Represents the output of the content renderer.
 | 
						|
 */
 | 
						|
export interface Result {
 | 
						|
    header: string;
 | 
						|
    content: string | Buffer | undefined;
 | 
						|
    /** Set to `true` if the provided content should be rendered as empty. */
 | 
						|
    isEmpty?: boolean;
 | 
						|
}
 | 
						|
 | 
						|
function getContent(note: SNote) {
 | 
						|
    if (note.isProtected) {
 | 
						|
        return {
 | 
						|
            header: "",
 | 
						|
            content: "<p>Protected note cannot be displayed</p>",
 | 
						|
            isEmpty: false
 | 
						|
        };
 | 
						|
    }
 | 
						|
 | 
						|
    const result: Result = {
 | 
						|
        content: note.getContent(),
 | 
						|
        header: "",
 | 
						|
        isEmpty: false
 | 
						|
    };
 | 
						|
 | 
						|
    if (note.type === "text") {
 | 
						|
        renderText(result, note);
 | 
						|
    } else if (note.type === "code") {
 | 
						|
        renderCode(result);
 | 
						|
    } else if (note.type === "mermaid") {
 | 
						|
        renderMermaid(result, note);
 | 
						|
    } else if (["image", "canvas", "mindMap"].includes(note.type)) {
 | 
						|
        renderImage(result, note);
 | 
						|
    } else if (note.type === "file") {
 | 
						|
        renderFile(note, result);
 | 
						|
    } else if (note.type === "book") {
 | 
						|
        result.isEmpty = true;
 | 
						|
    } else {
 | 
						|
        result.content = `<p>${t("content_renderer.note-cannot-be-displayed")}</p>`;
 | 
						|
    }
 | 
						|
 | 
						|
    return result;
 | 
						|
}
 | 
						|
 | 
						|
function renderIndex(result: Result) {
 | 
						|
    result.content += '<ul id="index">';
 | 
						|
 | 
						|
    const rootNote = shaca.getNote(shareRoot.SHARE_ROOT_NOTE_ID);
 | 
						|
 | 
						|
    for (const childNote of rootNote.getChildNotes()) {
 | 
						|
        const isExternalLink = childNote.hasLabel("shareExternalLink");
 | 
						|
        const href = isExternalLink ? childNote.getLabelValue("shareExternalLink") : `./${childNote.shareId}`;
 | 
						|
        const target = isExternalLink ? `target="_blank" rel="noopener noreferrer"` : "";
 | 
						|
        result.content += `<li><a class="${childNote.type}" href="${href}" ${target}>${childNote.escapedTitle}</a></li>`;
 | 
						|
    }
 | 
						|
 | 
						|
    result.content += "</ul>";
 | 
						|
}
 | 
						|
 | 
						|
function renderText(result: Result, note: SNote) {
 | 
						|
    const document = new JSDOM(result.content || "").window.document;
 | 
						|
 | 
						|
    // Process include notes.
 | 
						|
    for (const includeNoteEl of document.querySelectorAll("section.include-note")) {
 | 
						|
        const noteId = includeNoteEl.getAttribute("data-note-id");
 | 
						|
        if (!noteId) continue;
 | 
						|
 | 
						|
        const note = shaca.getNote(noteId);
 | 
						|
        if (!note) continue;
 | 
						|
 | 
						|
        const includedResult = getContent(note);
 | 
						|
        if (typeof includedResult.content !== "string") continue;
 | 
						|
 | 
						|
        const includedDocument = new JSDOM(includedResult.content).window.document;
 | 
						|
        includeNoteEl.replaceWith(includedDocument.body);
 | 
						|
    }
 | 
						|
 | 
						|
    result.isEmpty = document.body.textContent?.trim().length === 0 && document.querySelectorAll("img").length === 0;
 | 
						|
 | 
						|
    if (!result.isEmpty) {
 | 
						|
        for (const linkEl of document.querySelectorAll("a")) {
 | 
						|
            const href = linkEl.getAttribute("href");
 | 
						|
 | 
						|
            // Preserve footnotes.
 | 
						|
            if (href?.startsWith("#fn")) {
 | 
						|
                continue;
 | 
						|
            }
 | 
						|
 | 
						|
            if (href?.startsWith("#")) {
 | 
						|
                handleAttachmentLink(linkEl, href);
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        result.content = document.body.innerHTML;
 | 
						|
 | 
						|
        if (result.content.includes(`<span class="math-tex">`)) {
 | 
						|
            result.header += `
 | 
						|
<script src="../${assetPath}/node_modules/katex/dist/katex.min.js"></script>
 | 
						|
<link rel="stylesheet" href="../${assetPath}/node_modules/katex/dist/katex.min.css">
 | 
						|
<script src="../${assetPath}/node_modules/katex/dist/contrib/auto-render.min.js"></script>
 | 
						|
<script src="../${assetPath}/node_modules/katex/dist/contrib/mhchem.min.js"></script>
 | 
						|
<script>
 | 
						|
document.addEventListener("DOMContentLoaded", function() {
 | 
						|
    renderMathInElement(document.getElementById('content'));
 | 
						|
});
 | 
						|
</script>`;
 | 
						|
        }
 | 
						|
 | 
						|
        if (note.hasLabel("shareIndex")) {
 | 
						|
            renderIndex(result);
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function handleAttachmentLink(linkEl: HTMLAnchorElement, href: string) {
 | 
						|
    const linkRegExp = /attachmentId=([a-zA-Z0-9_]+)/g;
 | 
						|
    let attachmentMatch;
 | 
						|
    if ((attachmentMatch = linkRegExp.exec(href))) {
 | 
						|
        const attachmentId = attachmentMatch[1];
 | 
						|
        const attachment = shaca.getAttachment(attachmentId);
 | 
						|
 | 
						|
        if (attachment) {
 | 
						|
            linkEl.setAttribute("href", `api/attachments/${attachmentId}/download`);
 | 
						|
            linkEl.classList.add(`attachment-link`);
 | 
						|
            linkEl.classList.add(`role-${attachment.role}`);
 | 
						|
            linkEl.innerText = attachment.title;
 | 
						|
        } else {
 | 
						|
            linkEl.removeAttribute("href");
 | 
						|
        }
 | 
						|
    } else {
 | 
						|
        const [notePath] = href.split("?");
 | 
						|
        const notePathSegments = notePath.split("/");
 | 
						|
        const noteId = notePathSegments[notePathSegments.length - 1];
 | 
						|
        const linkedNote = shaca.getNote(noteId);
 | 
						|
        if (linkedNote) {
 | 
						|
            const isExternalLink = linkedNote.hasLabel("shareExternalLink");
 | 
						|
            const href = isExternalLink ? linkedNote.getLabelValue("shareExternalLink") : `./${linkedNote.shareId}`;
 | 
						|
            if (href) {
 | 
						|
                linkEl.setAttribute("href", href);
 | 
						|
            }
 | 
						|
            if (isExternalLink) {
 | 
						|
                linkEl.setAttribute("target", "_blank");
 | 
						|
                linkEl.setAttribute("rel", "noopener noreferrer");
 | 
						|
            }
 | 
						|
            linkEl.classList.add(`type-${linkedNote.type}`);
 | 
						|
        } else {
 | 
						|
            linkEl.removeAttribute("href");
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * Renders a code note.
 | 
						|
 */
 | 
						|
export function renderCode(result: Result) {
 | 
						|
    if (typeof result.content !== "string" || !result.content?.trim()) {
 | 
						|
        result.isEmpty = true;
 | 
						|
    } else {
 | 
						|
        const document = new JSDOM().window.document;
 | 
						|
 | 
						|
        const preEl = document.createElement("pre");
 | 
						|
        preEl.appendChild(document.createTextNode(result.content));
 | 
						|
 | 
						|
        result.content = preEl.outerHTML;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
function renderMermaid(result: Result, note: SNote) {
 | 
						|
    if (typeof result.content !== "string") {
 | 
						|
        return;
 | 
						|
    }
 | 
						|
 | 
						|
    result.content = `
 | 
						|
<img src="api/images/${note.noteId}/${note.encodedTitle}?${note.utcDateModified}">
 | 
						|
<hr>
 | 
						|
<details>
 | 
						|
    <summary>Chart source</summary>
 | 
						|
    <pre>${escapeHtml(result.content)}</pre>
 | 
						|
</details>`;
 | 
						|
}
 | 
						|
 | 
						|
function renderImage(result: Result, note: SNote) {
 | 
						|
    result.content = `<img src="api/images/${note.noteId}/${note.encodedTitle}?${note.utcDateModified}">`;
 | 
						|
}
 | 
						|
 | 
						|
function renderFile(note: SNote, result: Result) {
 | 
						|
    if (note.mime === "application/pdf") {
 | 
						|
        result.content = `<iframe class="pdf-view" src="api/notes/${note.noteId}/view"></iframe>`;
 | 
						|
    } else {
 | 
						|
        result.content = `<button type="button" onclick="location.href='api/notes/${note.noteId}/download'">Download file</button>`;
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
export default {
 | 
						|
    getContent
 | 
						|
};
 |