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: "
Protected note cannot be displayed
",
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 = `${t("content_renderer.note-cannot-be-displayed")}
`;
}
return result;
}
function renderIndex(result: Result) {
result.content += '';
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 += `${childNote.escapedTitle} `;
}
result.content += " ";
}
function renderText(result: Result, note: SNote) {
const document = new JSDOM(result.content || "").window.document;
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(``)) {
result.header += `
`;
}
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 = `
Chart source
${escapeHtml(result.content)}
`;
}
function renderImage(result: Result, note: SNote) {
result.content = ` `;
}
function renderFile(note: SNote, result: Result) {
if (note.mime === "application/pdf") {
result.content = ``;
} else {
result.content = `Download file `;
}
}
export default {
getContent
};