mirror of
				https://github.com/zadam/trilium.git
				synced 2025-10-27 00:06:30 +01:00 
			
		
		
		
	Compare commits
	
		
			50 Commits
		
	
	
		
			main
			...
			feature/ex
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|  | 52a6f2597e | ||
|  | ba26c478d6 | ||
|  | 055fcb7b2a | ||
|  | f4468706ef | ||
|  | 212956201a | ||
|  | 1182592fc5 | ||
|  | 0c399a676a | ||
|  | 395f33cd5b | ||
|  | 21b20cf575 | ||
|  | e3dd25b591 | ||
|  | b9a4e7ab11 | ||
|  | 6ae67c410c | ||
|  | 4ef7667484 | ||
|  | 3660e2f127 | ||
|  | 357d294f2d | ||
|  | bb636128b0 | ||
|  | aa102ab393 | ||
|  | ea53665e64 | ||
|  | 9cf7fa1997 | ||
|  | fded714f18 | ||
|  | 06de06b501 | ||
|  | 9abdbbbc5b | ||
|  | 3ebfee8bd2 | ||
|  | 6d446c5b27 | ||
|  | 3a55490bbf | ||
|  | bc4643fed2 | ||
|  | a2110ca631 | ||
|  | 413137ac64 | ||
|  | 9bc966491d | ||
|  | 61dbc15fc6 | ||
|  | b475037127 | ||
|  | 35622a2122 | ||
|  | 77e4c3d0ec | ||
|  | 8523050ab2 | ||
|  | 0efdf65202 | ||
|  | acb0991d05 | ||
|  | a9f68f5487 | ||
|  | 55bb2fdb9b | ||
|  | e529633b8b | ||
|  | dfd575b6eb | ||
|  | c5196721d4 | ||
|  | 968c75b618 | ||
|  | 01beebf660 | ||
|  | d3115e834a | ||
|  | 01a552ceb5 | ||
|  | d8958adea5 | ||
|  | 4d5e866db6 | ||
|  | f189deb415 | ||
|  | 9c460dbc87 | ||
|  | 2c6ba9ba2c | 
| @@ -9,16 +9,6 @@ async function ensureJQuery() { | |||||||
|     (window as any).$ = $; |     (window as any).$ = $; | ||||||
| } | } | ||||||
|  |  | ||||||
| async function applyMath() { |  | ||||||
|     const anyMathBlock = document.querySelector("#content .math-tex"); |  | ||||||
|     if (!anyMathBlock) { |  | ||||||
|         return; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const renderMathInElement = (await import("./services/math.js")).renderMathInElement; |  | ||||||
|     renderMathInElement(document.getElementById("content")); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function formatCodeBlocks() { | async function formatCodeBlocks() { | ||||||
|     const anyCodeBlock = document.querySelector("#content pre"); |     const anyCodeBlock = document.querySelector("#content pre"); | ||||||
|     if (!anyCodeBlock) { |     if (!anyCodeBlock) { | ||||||
| @@ -31,54 +21,4 @@ async function formatCodeBlocks() { | |||||||
|  |  | ||||||
| async function setupTextNote() { | async function setupTextNote() { | ||||||
|     formatCodeBlocks(); |     formatCodeBlocks(); | ||||||
|     applyMath(); |  | ||||||
|  |  | ||||||
|     const setupMermaid = (await import("./share/mermaid.js")).default; |  | ||||||
|     setupMermaid(); |  | ||||||
| } | } | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Fetch note with given ID from backend |  | ||||||
|  * |  | ||||||
|  * @param noteId of the given note to be fetched. If false, fetches current note. |  | ||||||
|  */ |  | ||||||
| async function fetchNote(noteId: string | null = null) { |  | ||||||
|     if (!noteId) { |  | ||||||
|         noteId = document.body.getAttribute("data-note-id"); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     const resp = await fetch(`api/notes/${noteId}`); |  | ||||||
|  |  | ||||||
|     return await resp.json(); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| document.addEventListener( |  | ||||||
|     "DOMContentLoaded", |  | ||||||
|     () => { |  | ||||||
|         const noteType = determineNoteType(); |  | ||||||
|  |  | ||||||
|         if (noteType === "text") { |  | ||||||
|             setupTextNote(); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const toggleMenuButton = document.getElementById("toggleMenuButton"); |  | ||||||
|         const layout = document.getElementById("layout"); |  | ||||||
|  |  | ||||||
|         if (toggleMenuButton && layout) { |  | ||||||
|             toggleMenuButton.addEventListener("click", () => layout.classList.toggle("showMenu")); |  | ||||||
|         } |  | ||||||
|     }, |  | ||||||
|     false |  | ||||||
| ); |  | ||||||
|  |  | ||||||
| function determineNoteType() { |  | ||||||
|     const bodyClass = document.body.className; |  | ||||||
|     const match = bodyClass.match(/type-([^\s]+)/); |  | ||||||
|     return match ? match[1] : null; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| // workaround to prevent webpack from removing "fetchNote" as dead code: |  | ||||||
| // add fetchNote as property to the window object |  | ||||||
| Object.defineProperty(window, "fetchNote", { |  | ||||||
|     value: fetchNote |  | ||||||
| }); |  | ||||||
|   | |||||||
| @@ -104,7 +104,8 @@ | |||||||
|     "export_status": "Export status", |     "export_status": "Export status", | ||||||
|     "export_in_progress": "Export in progress: {{progressCount}}", |     "export_in_progress": "Export in progress: {{progressCount}}", | ||||||
|     "export_finished_successfully": "Export finished successfully.", |     "export_finished_successfully": "Export finished successfully.", | ||||||
|     "format_pdf": "PDF - for printing or sharing purposes." |     "format_pdf": "PDF - for printing or sharing purposes.", | ||||||
|  |     "share-format": "HTML for web publishing - uses the same theme that is used shared notes, but can be published as a static website." | ||||||
|   }, |   }, | ||||||
|   "help": { |   "help": { | ||||||
|     "title": "Cheatsheet", |     "title": "Cheatsheet", | ||||||
|   | |||||||
| @@ -79,7 +79,8 @@ export default function ExportDialog() { | |||||||
|                         values={[ |                         values={[ | ||||||
|                             { value: "html", label: t("export.format_html_zip") }, |                             { value: "html", label: t("export.format_html_zip") }, | ||||||
|                             { value: "markdown", label: t("export.format_markdown") }, |                             { value: "markdown", label: t("export.format_markdown") }, | ||||||
|                             { value: "opml", label: t("export.format_opml") } |                             { value: "opml", label: t("export.format_opml") }, | ||||||
|  |                             { value: "share", label: t("export.share-format") } | ||||||
|                         ]} |                         ]} | ||||||
|                     /> |                     /> | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ import { initializeTranslations } from "@triliumnext/server/src/services/i18n.js | |||||||
| import debounce from "@triliumnext/client/src/services/debounce.js"; | import debounce from "@triliumnext/client/src/services/debounce.js"; | ||||||
| import { extractZip, importData, initializeDatabase, startElectron } from "./utils.js"; | import { extractZip, importData, initializeDatabase, startElectron } from "./utils.js"; | ||||||
| import cls from "@triliumnext/server/src/services/cls.js"; | import cls from "@triliumnext/server/src/services/cls.js"; | ||||||
| import type { AdvancedExportOptions } from "@triliumnext/server/src/services/export/zip.js"; | import type { AdvancedExportOptions, ExportFormat } from "@triliumnext/server/src/services/export/zip/abstract_provider.js"; | ||||||
| import { parseNoteMetaFile } from "@triliumnext/server/src/services/in_app_help.js"; | import { parseNoteMetaFile } from "@triliumnext/server/src/services/in_app_help.js"; | ||||||
| import type NoteMeta from "@triliumnext/server/src/services/meta/note_meta.js"; | import type NoteMeta from "@triliumnext/server/src/services/meta/note_meta.js"; | ||||||
|  |  | ||||||
| @@ -75,7 +75,7 @@ async function setOptions() { | |||||||
|     optionsService.setOption("compressImages", "false"); |     optionsService.setOption("compressImages", "false"); | ||||||
| } | } | ||||||
|  |  | ||||||
| async function exportData(noteId: string, format: "html" | "markdown", outputPath: string, ignoredFiles?: Set<string>) { | async function exportData(noteId: string, format: ExportFormat, outputPath: string, ignoredFiles?: Set<string>) { | ||||||
|     const zipFilePath = "output.zip"; |     const zipFilePath = "output.zip"; | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ async function main() { | |||||||
|  |  | ||||||
|     // Copy assets |     // Copy assets | ||||||
|     build.copy("src/assets", "assets/"); |     build.copy("src/assets", "assets/"); | ||||||
|  |     build.triggerBuildAndCopyTo("packages/share-theme", "share-theme/assets/"); | ||||||
|     build.copy("/packages/share-theme/src/templates", "share-theme/templates/"); |     build.copy("/packages/share-theme/src/templates", "share-theme/templates/"); | ||||||
|  |  | ||||||
|     // Copy node modules dependencies |     // Copy node modules dependencies | ||||||
|   | |||||||
| @@ -278,6 +278,11 @@ class BBranch extends AbstractBeccaEntity<BBranch> { | |||||||
|             }); |             }); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     getParentNote() { | ||||||
|  |         return this.parentNote; | ||||||
|  |     } | ||||||
|  |  | ||||||
| } | } | ||||||
|  |  | ||||||
| export default BBranch; | export default BBranch; | ||||||
|   | |||||||
| @@ -1758,6 +1758,26 @@ class BNote extends AbstractBeccaEntity<BNote> { | |||||||
|         return childBranches; |         return childBranches; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     get encodedTitle() { | ||||||
|  |         return encodeURIComponent(this.title); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     getVisibleChildBranches() { | ||||||
|  |         return this.getChildBranches().filter((branch) => !branch.getNote().isLabelTruthy("shareHiddenFromTree")); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     getVisibleChildNotes() { | ||||||
|  |         return this.getVisibleChildBranches().map((branch) => branch.getNote()); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     hasVisibleChildren() { | ||||||
|  |         return this.getVisibleChildNotes().length > 0; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     get shareId() { | ||||||
|  |         return this.noteId; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Return an attribute by it's attributeId.  Requires the attribute cache to be available. |      * Return an attribute by it's attributeId.  Requires the attribute cache to be available. | ||||||
|      * @param attributeId - the id of the attribute owned by this note |      * @param attributeId - the id of the attribute owned by this note | ||||||
|   | |||||||
| @@ -14,6 +14,7 @@ import type { ParsedQs } from "qs"; | |||||||
| import type { NoteParams } from "../services/note-interface.js"; | import type { NoteParams } from "../services/note-interface.js"; | ||||||
| import type { SearchParams } from "../services/search/services/types.js"; | import type { SearchParams } from "../services/search/services/types.js"; | ||||||
| import type { ValidatorMap } from "./etapi-interface.js"; | import type { ValidatorMap } from "./etapi-interface.js"; | ||||||
|  | import type { ExportFormat } from "../services/export/zip/abstract_provider.js"; | ||||||
|  |  | ||||||
| function register(router: Router) { | function register(router: Router) { | ||||||
|     eu.route(router, "get", "/etapi/notes", (req, res, next) => { |     eu.route(router, "get", "/etapi/notes", (req, res, next) => { | ||||||
| @@ -149,7 +150,7 @@ function register(router: Router) { | |||||||
|         const note = eu.getAndCheckNote(req.params.noteId); |         const note = eu.getAndCheckNote(req.params.noteId); | ||||||
|         const format = req.query.format || "html"; |         const format = req.query.format || "html"; | ||||||
|  |  | ||||||
|         if (typeof format !== "string" || !["html", "markdown"].includes(format)) { |         if (typeof format !== "string" || !["html", "markdown", "share"].includes(format)) { | ||||||
|             throw new eu.EtapiError(400, "UNRECOGNIZED_EXPORT_FORMAT", `Unrecognized export format '${format}', supported values are 'html' (default) or 'markdown'.`); |             throw new eu.EtapiError(400, "UNRECOGNIZED_EXPORT_FORMAT", `Unrecognized export format '${format}', supported values are 'html' (default) or 'markdown'.`); | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -159,7 +160,7 @@ function register(router: Router) { | |||||||
|         // (e.g. branchIds are not seen in UI), that we export "note export" instead. |         // (e.g. branchIds are not seen in UI), that we export "note export" instead. | ||||||
|         const branch = note.getParentBranches()[0]; |         const branch = note.getParentBranches()[0]; | ||||||
|  |  | ||||||
|         zipExportService.exportToZip(taskContext, branch, format as "html" | "markdown", res); |         zipExportService.exportToZip(taskContext, branch, format as ExportFormat, res); | ||||||
|     }); |     }); | ||||||
|  |  | ||||||
|     eu.route(router, "post", "/etapi/notes/:noteId/import", (req, res, next) => { |     eu.route(router, "post", "/etapi/notes/:noteId/import", (req, res, next) => { | ||||||
|   | |||||||
| @@ -26,7 +26,7 @@ function exportBranch(req: Request, res: Response) { | |||||||
|     const taskContext = new TaskContext(taskId, "export", null); |     const taskContext = new TaskContext(taskId, "export", null); | ||||||
|  |  | ||||||
|     try { |     try { | ||||||
|         if (type === "subtree" && (format === "html" || format === "markdown")) { |         if (type === "subtree" && (format === "html" || format === "markdown" || format === "share")) { | ||||||
|             zipExportService.exportToZip(taskContext, branch, format, res); |             zipExportService.exportToZip(taskContext, branch, format, res); | ||||||
|         } else if (type === "single") { |         } else if (type === "single") { | ||||||
|             if (format !== "html" && format !== "markdown") { |             if (format !== "html" && format !== "markdown") { | ||||||
|   | |||||||
| @@ -44,6 +44,7 @@ async function register(app: express.Application) { | |||||||
|         app.use(`/${assetUrlFragment}/translations/`, persistentCacheStatic(path.join(publicDir, "translations"))); |         app.use(`/${assetUrlFragment}/translations/`, persistentCacheStatic(path.join(publicDir, "translations"))); | ||||||
|         app.use(`/node_modules/`, persistentCacheStatic(path.join(publicDir, "node_modules"))); |         app.use(`/node_modules/`, persistentCacheStatic(path.join(publicDir, "node_modules"))); | ||||||
|     } |     } | ||||||
|  |     app.use(`/share/assets/`, express.static(getShareThemeAssetDir())); | ||||||
|     app.use(`/${assetUrlFragment}/images`, persistentCacheStatic(path.join(resourceDir, "assets", "images"))); |     app.use(`/${assetUrlFragment}/images`, persistentCacheStatic(path.join(resourceDir, "assets", "images"))); | ||||||
|     app.use(`/${assetUrlFragment}/doc_notes`, persistentCacheStatic(path.join(resourceDir, "assets", "doc_notes"))); |     app.use(`/${assetUrlFragment}/doc_notes`, persistentCacheStatic(path.join(resourceDir, "assets", "doc_notes"))); | ||||||
|     app.use(`/assets/vX/fonts`, express.static(path.join(srcRoot, "public/fonts"))); |     app.use(`/assets/vX/fonts`, express.static(path.join(srcRoot, "public/fonts"))); | ||||||
| @@ -51,6 +52,16 @@ async function register(app: express.Application) { | |||||||
|     app.use(`/assets/vX/stylesheets`, express.static(path.join(srcRoot, "public/stylesheets"))); |     app.use(`/assets/vX/stylesheets`, express.static(path.join(srcRoot, "public/stylesheets"))); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | export function getShareThemeAssetDir() { | ||||||
|  |     if (process.env.NODE_ENV === "development") { | ||||||
|  |         const srcRoot = path.join(__dirname, "..", ".."); | ||||||
|  |         return path.join(srcRoot, "../../packages/share-theme/dist"); | ||||||
|  |     } else { | ||||||
|  |         const resourceDir = getResourceDir(); | ||||||
|  |         return path.join(resourceDir, "share-theme/assets"); | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| export default { | export default { | ||||||
|     register |     register | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -9,8 +9,9 @@ import type TaskContext from "../task_context.js"; | |||||||
| import type BBranch from "../../becca/entities/bbranch.js"; | import type BBranch from "../../becca/entities/bbranch.js"; | ||||||
| import type { Response } from "express"; | import type { Response } from "express"; | ||||||
| import type BNote from "../../becca/entities/bnote.js"; | import type BNote from "../../becca/entities/bnote.js"; | ||||||
|  | import type { ExportFormat } from "./zip/abstract_provider.js"; | ||||||
|  |  | ||||||
| function exportSingleNote(taskContext: TaskContext<"export">, branch: BBranch, format: "html" | "markdown", res: Response) { | function exportSingleNote(taskContext: TaskContext<"export">, branch: BBranch, format: ExportFormat, res: Response) { | ||||||
|     const note = branch.getNote(); |     const note = branch.getNote(); | ||||||
|  |  | ||||||
|     if (note.type === "image" || note.type === "file") { |     if (note.type === "image" || note.type === "file") { | ||||||
| @@ -33,7 +34,7 @@ function exportSingleNote(taskContext: TaskContext<"export">, branch: BBranch, f | |||||||
|     taskContext.taskSucceeded(null); |     taskContext.taskSucceeded(null); | ||||||
| } | } | ||||||
|  |  | ||||||
| export function mapByNoteType(note: BNote, content: string | Buffer<ArrayBufferLike>, format: "html" | "markdown") { | export function mapByNoteType(note: BNote, content: string | Buffer<ArrayBufferLike>, format: ExportFormat) { | ||||||
|     let payload, extension, mime; |     let payload, extension, mime; | ||||||
|  |  | ||||||
|     if (typeof content !== "string") { |     if (typeof content !== "string") { | ||||||
|   | |||||||
| @@ -1,12 +1,9 @@ | |||||||
| "use strict"; | "use strict"; | ||||||
|  |  | ||||||
| import html from "html"; |  | ||||||
| import dateUtils from "../date_utils.js"; | import dateUtils from "../date_utils.js"; | ||||||
| import path from "path"; | import path from "path"; | ||||||
| import mimeTypes from "mime-types"; |  | ||||||
| import mdService from "./markdown.js"; |  | ||||||
| import packageInfo from "../../../package.json" with { type: "json" }; | import packageInfo from "../../../package.json" with { type: "json" }; | ||||||
| import { getContentDisposition, escapeHtml, getResourceDir, isDev } from "../utils.js"; | import { getContentDisposition } from "../utils.js"; | ||||||
| import protectedSessionService from "../protected_session.js"; | import protectedSessionService from "../protected_session.js"; | ||||||
| import sanitize from "sanitize-filename"; | import sanitize from "sanitize-filename"; | ||||||
| import fs from "fs"; | import fs from "fs"; | ||||||
| @@ -18,39 +15,48 @@ import ValidationError from "../../errors/validation_error.js"; | |||||||
| import type NoteMeta from "../meta/note_meta.js"; | import type NoteMeta from "../meta/note_meta.js"; | ||||||
| import type AttachmentMeta from "../meta/attachment_meta.js"; | import type AttachmentMeta from "../meta/attachment_meta.js"; | ||||||
| import type AttributeMeta from "../meta/attribute_meta.js"; | import type AttributeMeta from "../meta/attribute_meta.js"; | ||||||
| import type BBranch from "../../becca/entities/bbranch.js"; | import BBranch from "../../becca/entities/bbranch.js"; | ||||||
| import type { Response } from "express"; | import type { Response } from "express"; | ||||||
| import type { NoteMetaFile } from "../meta/note_meta.js"; | import type { NoteMetaFile } from "../meta/note_meta.js"; | ||||||
|  | import HtmlExportProvider from "./zip/html.js"; | ||||||
|  | import { AdvancedExportOptions, type ExportFormat, ZipExportProviderData } from "./zip/abstract_provider.js"; | ||||||
|  | import MarkdownExportProvider from "./zip/markdown.js"; | ||||||
|  | import ShareThemeExportProvider from "./zip/share_theme.js"; | ||||||
|  | import type BNote from "../../becca/entities/bnote.js"; | ||||||
|  | import { NoteType } from "@triliumnext/commons"; | ||||||
|  |  | ||||||
| type RewriteLinksFn = (content: string, noteMeta: NoteMeta) => string; | async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, format: ExportFormat, res: Response | fs.WriteStream, setHeaders = true, zipExportOptions?: AdvancedExportOptions) { | ||||||
|  |     if (!["html", "markdown", "share"].includes(format)) { | ||||||
| export interface AdvancedExportOptions { |         throw new ValidationError(`Only 'html', 'markdown' and 'share' allowed as export format, '${format}' given`); | ||||||
|     /** |  | ||||||
|      * If `true`, then only the note's content will be kept. If `false` (default), then each page will have its own <html> template. |  | ||||||
|      */ |  | ||||||
|     skipHtmlTemplate?: boolean; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Provides a custom function to rewrite the links found in HTML or Markdown notes. This method is called for every note imported, if it's of the right type. |  | ||||||
|      * |  | ||||||
|      * @param originalRewriteLinks the original rewrite links function. Can be used to access the default behaviour without having to reimplement it. |  | ||||||
|      * @param getNoteTargetUrl the method to obtain a note's target URL, used internally by `originalRewriteLinks` but can be used here as well. |  | ||||||
|      * @returns a function to rewrite the links in HTML or Markdown notes. |  | ||||||
|      */ |  | ||||||
|     customRewriteLinks?: (originalRewriteLinks: RewriteLinksFn, getNoteTargetUrl: (targetNoteId: string, sourceMeta: NoteMeta) => string | null) => RewriteLinksFn; |  | ||||||
| } |  | ||||||
|  |  | ||||||
| async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, format: "html" | "markdown", res: Response | fs.WriteStream, setHeaders = true, zipExportOptions?: AdvancedExportOptions) { |  | ||||||
|     if (!["html", "markdown"].includes(format)) { |  | ||||||
|         throw new ValidationError(`Only 'html' and 'markdown' allowed as export format, '${format}' given`); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const archive = archiver("zip", { |     const archive = archiver("zip", { | ||||||
|         zlib: { level: 9 } // Sets the compression level. |         zlib: { level: 9 } // Sets the compression level. | ||||||
|     }); |     }); | ||||||
|  |     const rewriteFn = (zipExportOptions?.customRewriteLinks ? zipExportOptions?.customRewriteLinks(rewriteLinks, getNoteTargetUrl) : rewriteLinks); | ||||||
|  |     const provider = buildProvider(); | ||||||
|  |  | ||||||
|     const noteIdToMeta: Record<string, NoteMeta> = {}; |     const noteIdToMeta: Record<string, NoteMeta> = {}; | ||||||
|  |  | ||||||
|  |     function buildProvider() { | ||||||
|  |         const providerData: ZipExportProviderData = { | ||||||
|  |             getNoteTargetUrl, | ||||||
|  |             archive, | ||||||
|  |             branch, | ||||||
|  |             rewriteFn | ||||||
|  |         }; | ||||||
|  |         switch (format) { | ||||||
|  |             case "html": | ||||||
|  |                 return new HtmlExportProvider(providerData); | ||||||
|  |             case "markdown": | ||||||
|  |                 return new MarkdownExportProvider(providerData); | ||||||
|  |             case "share": | ||||||
|  |                 return new ShareThemeExportProvider(providerData); | ||||||
|  |             default: | ||||||
|  |                 throw new Error(); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     function getUniqueFilename(existingFileNames: Record<string, number>, fileName: string) { |     function getUniqueFilename(existingFileNames: Record<string, number>, fileName: string) { | ||||||
|         const lcFileName = fileName.toLowerCase(); |         const lcFileName = fileName.toLowerCase(); | ||||||
|  |  | ||||||
| @@ -72,7 +78,7 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function getDataFileName(type: string | null, mime: string, baseFileName: string, existingFileNames: Record<string, number>): string { |     function getDataFileName(type: NoteType | null, mime: string, baseFileName: string, existingFileNames: Record<string, number>): string { | ||||||
|         let fileName = baseFileName.trim(); |         let fileName = baseFileName.trim(); | ||||||
|         if (!fileName) { |         if (!fileName) { | ||||||
|             fileName = "note"; |             fileName = "note"; | ||||||
| @@ -90,36 +96,14 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         let existingExtension = path.extname(fileName).toLowerCase(); |         let existingExtension = path.extname(fileName).toLowerCase(); | ||||||
|         let newExtension; |         const newExtension = provider.mapExtension(type, mime, existingExtension, format); | ||||||
|  |  | ||||||
|         // the following two are handled specifically since we always want to have these extensions no matter the automatic detection |  | ||||||
|         // and/or existing detected extensions in the note name |  | ||||||
|         if (type === "text" && format === "markdown") { |  | ||||||
|             newExtension = "md"; |  | ||||||
|         } else if (type === "text" && format === "html") { |  | ||||||
|             newExtension = "html"; |  | ||||||
|         } else if (mime === "application/x-javascript" || mime === "text/javascript") { |  | ||||||
|             newExtension = "js"; |  | ||||||
|         } else if (type === "canvas" || mime === "application/json") { |  | ||||||
|             newExtension = "json"; |  | ||||||
|         } else if (existingExtension.length > 0) { |  | ||||||
|             // if the page already has an extension, then we'll just keep it |  | ||||||
|             newExtension = null; |  | ||||||
|         } else { |  | ||||||
|             if (mime?.toLowerCase()?.trim() === "image/jpg") { |  | ||||||
|                 newExtension = "jpg"; |  | ||||||
|             } else if (mime?.toLowerCase()?.trim() === "text/mermaid") { |  | ||||||
|                 newExtension = "txt"; |  | ||||||
|             } else { |  | ||||||
|                 newExtension = mimeTypes.extension(mime) || "dat"; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // if the note is already named with the extension (e.g. "image.jpg"), then it's silly to append the exact same extension again |         // if the note is already named with the extension (e.g. "image.jpg"), then it's silly to append the exact same extension again | ||||||
|         if (newExtension && existingExtension !== `.${newExtension.toLowerCase()}`) { |         if (newExtension && existingExtension !== `.${newExtension.toLowerCase()}`) { | ||||||
|             fileName += `.${newExtension}`; |             fileName += `.${newExtension}`; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |  | ||||||
|         return getUniqueFilename(existingFileNames, fileName); |         return getUniqueFilename(existingFileNames, fileName); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -145,7 +129,8 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|         const notePath = parentMeta.notePath.concat([note.noteId]); |         const notePath = parentMeta.notePath.concat([note.noteId]); | ||||||
|  |  | ||||||
|         if (note.noteId in noteIdToMeta) { |         if (note.noteId in noteIdToMeta) { | ||||||
|             const fileName = getUniqueFilename(existingFileNames, `${baseFileName}.clone.${format === "html" ? "html" : "md"}`); |             const extension = provider.mapExtension("text", "text/html", "", format); | ||||||
|  |             const fileName = getUniqueFilename(existingFileNames, `${baseFileName}.clone.${extension}`); | ||||||
|  |  | ||||||
|             const meta: NoteMeta = { |             const meta: NoteMeta = { | ||||||
|                 isClone: true, |                 isClone: true, | ||||||
| @@ -155,7 +140,7 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|                 prefix: branch.prefix, |                 prefix: branch.prefix, | ||||||
|                 dataFileName: fileName, |                 dataFileName: fileName, | ||||||
|                 type: "text", // export will have text description |                 type: "text", // export will have text description | ||||||
|                 format: format |                 format: (format === "markdown" ? "markdown" : "html") | ||||||
|             }; |             }; | ||||||
|             return meta; |             return meta; | ||||||
|         } |         } | ||||||
| @@ -185,7 +170,7 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|         taskContext.increaseProgressCount(); |         taskContext.increaseProgressCount(); | ||||||
|  |  | ||||||
|         if (note.type === "text") { |         if (note.type === "text") { | ||||||
|             meta.format = format; |             meta.format = (format === "markdown" ? "markdown" : "html"); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         noteIdToMeta[note.noteId] = meta as NoteMeta; |         noteIdToMeta[note.noteId] = meta as NoteMeta; | ||||||
| @@ -194,10 +179,13 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|         note.sortChildren(); |         note.sortChildren(); | ||||||
|         const childBranches = note.getChildBranches().filter((branch) => branch?.noteId !== "_hidden"); |         const childBranches = note.getChildBranches().filter((branch) => branch?.noteId !== "_hidden"); | ||||||
|  |  | ||||||
|         const available = !note.isProtected || protectedSessionService.isProtectedSessionAvailable(); |         let shouldIncludeFile = (!note.isProtected || protectedSessionService.isProtectedSessionAvailable()); | ||||||
|  |         if (format !== "share") { | ||||||
|  |             shouldIncludeFile = shouldIncludeFile && (note.getContent().length > 0 || childBranches.length === 0); | ||||||
|  |         } | ||||||
|  |  | ||||||
|         // if it's a leaf, then we'll export it even if it's empty |         // if it's a leaf, then we'll export it even if it's empty | ||||||
|         if (available && (note.getContent().length > 0 || childBranches.length === 0)) { |         if (shouldIncludeFile) { | ||||||
|             meta.dataFileName = getDataFileName(note.type, note.mime, baseFileName, existingFileNames); |             meta.dataFileName = getDataFileName(note.type, note.mime, baseFileName, existingFileNames); | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -273,8 +261,6 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|         return url; |         return url; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     const rewriteFn = (zipExportOptions?.customRewriteLinks ? zipExportOptions?.customRewriteLinks(rewriteLinks, getNoteTargetUrl) : rewriteLinks); |  | ||||||
|  |  | ||||||
|     function rewriteLinks(content: string, noteMeta: NoteMeta): string { |     function rewriteLinks(content: string, noteMeta: NoteMeta): string { | ||||||
|         content = content.replace(/src="[^"]*api\/images\/([a-zA-Z0-9_]+)\/[^"]*"/g, (match, targetNoteId) => { |         content = content.replace(/src="[^"]*api\/images\/([a-zA-Z0-9_]+)\/[^"]*"/g, (match, targetNoteId) => { | ||||||
|             const url = getNoteTargetUrl(targetNoteId, noteMeta); |             const url = getNoteTargetUrl(targetNoteId, noteMeta); | ||||||
| @@ -316,54 +302,16 @@ async function exportToZip(taskContext: TaskContext<"export">, branch: BBranch, | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function prepareContent(title: string, content: string | Buffer, noteMeta: NoteMeta): string | Buffer { |     function prepareContent(title: string, content: string | Buffer, noteMeta: NoteMeta, note?: BNote): string | Buffer { | ||||||
|         if (["html", "markdown"].includes(noteMeta?.format || "")) { |         const isText = ["html", "markdown"].includes(noteMeta?.format || ""); | ||||||
|  |         if (isText) { | ||||||
|             content = content.toString(); |             content = content.toString(); | ||||||
|             content = rewriteFn(content, noteMeta); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (noteMeta.format === "html" && typeof content === "string") { |         content = provider.prepareContent(title, content, noteMeta, note, branch); | ||||||
|             if (!content.substr(0, 100).toLowerCase().includes("<html") && !zipExportOptions?.skipHtmlTemplate) { |  | ||||||
|                 if (!noteMeta?.notePath?.length) { |  | ||||||
|                     throw new Error("Missing note path."); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 const cssUrl = `${"../".repeat(noteMeta.notePath.length - 1)}style.css`; |  | ||||||
|                 const htmlTitle = escapeHtml(title); |  | ||||||
|  |  | ||||||
|                 // <base> element will make sure external links are openable - https://github.com/zadam/trilium/issues/1289#issuecomment-704066809 |  | ||||||
|                 content = `<html> |  | ||||||
| <head> |  | ||||||
|     <meta charset="utf-8"> |  | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1"> |  | ||||||
|     <link rel="stylesheet" href="${cssUrl}"> |  | ||||||
|     <base target="_parent"> |  | ||||||
|     <title data-trilium-title>${htmlTitle}</title> |  | ||||||
| </head> |  | ||||||
| <body> |  | ||||||
|   <div class="content"> |  | ||||||
|     <h1 data-trilium-h1>${htmlTitle}</h1> |  | ||||||
|  |  | ||||||
|     <div class="ck-content">${content}</div> |  | ||||||
|   </div> |  | ||||||
| </body> |  | ||||||
| </html>`; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return content.length < 100_000 ? html.prettyPrint(content, { indent_size: 2 }) : content; |  | ||||||
|         } else if (noteMeta.format === "markdown" && typeof content === "string") { |  | ||||||
|             let markdownContent = mdService.toMarkdown(content); |  | ||||||
|  |  | ||||||
|             if (markdownContent.trim().length > 0 && !markdownContent.startsWith("# ")) { |  | ||||||
|                 markdownContent = `# ${title}\r |  | ||||||
| ${markdownContent}`; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return markdownContent; |  | ||||||
|         } else { |  | ||||||
|         return content; |         return content; | ||||||
|     } |     } | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function saveNote(noteMeta: NoteMeta, filePathPrefix: string) { |     function saveNote(noteMeta: NoteMeta, filePathPrefix: string) { | ||||||
|         log.info(`Exporting note '${noteMeta.noteId}'`); |         log.info(`Exporting note '${noteMeta.noteId}'`); | ||||||
| @@ -377,7 +325,7 @@ ${markdownContent}`; | |||||||
|  |  | ||||||
|             let content: string | Buffer = `<p>This is a clone of a note. Go to its <a href="${targetUrl}">primary location</a>.</p>`; |             let content: string | Buffer = `<p>This is a clone of a note. Go to its <a href="${targetUrl}">primary location</a>.</p>`; | ||||||
|  |  | ||||||
|             content = prepareContent(noteMeta.title, content, noteMeta); |             content = prepareContent(noteMeta.title, content, noteMeta, undefined); | ||||||
|  |  | ||||||
|             archive.append(content, { name: filePathPrefix + noteMeta.dataFileName }); |             archive.append(content, { name: filePathPrefix + noteMeta.dataFileName }); | ||||||
|  |  | ||||||
| @@ -393,7 +341,7 @@ ${markdownContent}`; | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         if (noteMeta.dataFileName) { |         if (noteMeta.dataFileName) { | ||||||
|             const content = prepareContent(noteMeta.title, note.getContent(), noteMeta); |             const content = prepareContent(noteMeta.title, note.getContent(), noteMeta, note); | ||||||
|  |  | ||||||
|             archive.append(content, { |             archive.append(content, { | ||||||
|                 name: filePathPrefix + noteMeta.dataFileName, |                 name: filePathPrefix + noteMeta.dataFileName, | ||||||
| @@ -429,99 +377,6 @@ ${markdownContent}`; | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     function saveNavigation(rootMeta: NoteMeta, navigationMeta: NoteMeta) { |  | ||||||
|         if (!navigationMeta.dataFileName) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         function saveNavigationInner(meta: NoteMeta) { |  | ||||||
|             let html = "<li>"; |  | ||||||
|  |  | ||||||
|             const escapedTitle = escapeHtml(`${meta.prefix ? `${meta.prefix} - ` : ""}${meta.title}`); |  | ||||||
|  |  | ||||||
|             if (meta.dataFileName && meta.noteId) { |  | ||||||
|                 const targetUrl = getNoteTargetUrl(meta.noteId, rootMeta); |  | ||||||
|  |  | ||||||
|                 html += `<a href="${targetUrl}" target="detail">${escapedTitle}</a>`; |  | ||||||
|             } else { |  | ||||||
|                 html += escapedTitle; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (meta.children && meta.children.length > 0) { |  | ||||||
|                 html += "<ul>"; |  | ||||||
|  |  | ||||||
|                 for (const child of meta.children) { |  | ||||||
|                     html += saveNavigationInner(child); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 html += "</ul>"; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return `${html}</li>`; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const fullHtml = `<html> |  | ||||||
| <head> |  | ||||||
|     <meta charset="utf-8"> |  | ||||||
|     <link rel="stylesheet" href="style.css"> |  | ||||||
| </head> |  | ||||||
| <body> |  | ||||||
|     <ul>${saveNavigationInner(rootMeta)}</ul> |  | ||||||
| </body> |  | ||||||
| </html>`; |  | ||||||
|         const prettyHtml = fullHtml.length < 100_000 ? html.prettyPrint(fullHtml, { indent_size: 2 }) : fullHtml; |  | ||||||
|  |  | ||||||
|         archive.append(prettyHtml, { name: navigationMeta.dataFileName }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function saveIndex(rootMeta: NoteMeta, indexMeta: NoteMeta) { |  | ||||||
|         let firstNonEmptyNote; |  | ||||||
|         let curMeta = rootMeta; |  | ||||||
|  |  | ||||||
|         if (!indexMeta.dataFileName) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         while (!firstNonEmptyNote) { |  | ||||||
|             if (curMeta.dataFileName && curMeta.noteId) { |  | ||||||
|                 firstNonEmptyNote = getNoteTargetUrl(curMeta.noteId, rootMeta); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if (curMeta.children && curMeta.children.length > 0) { |  | ||||||
|                 curMeta = curMeta.children[0]; |  | ||||||
|             } else { |  | ||||||
|                 break; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const fullHtml = `<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> |  | ||||||
| <html> |  | ||||||
| <head> |  | ||||||
|     <meta charset="utf-8"> |  | ||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1"> |  | ||||||
| </head> |  | ||||||
| <frameset cols="25%,75%"> |  | ||||||
|     <frame name="navigation" src="navigation.html"> |  | ||||||
|     <frame name="detail" src="${firstNonEmptyNote}"> |  | ||||||
| </frameset> |  | ||||||
| </html>`; |  | ||||||
|  |  | ||||||
|         archive.append(fullHtml, { name: indexMeta.dataFileName }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     function saveCss(rootMeta: NoteMeta, cssMeta: NoteMeta) { |  | ||||||
|         if (!cssMeta.dataFileName) { |  | ||||||
|             return; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const cssFile = isDev |  | ||||||
|             ? path.join(__dirname, "../../../../../node_modules/ckeditor5/dist/ckeditor5-content.css") |  | ||||||
|             : path.join(getResourceDir(), "ckeditor5-content.css"); |  | ||||||
|  |  | ||||||
|         archive.append(fs.readFileSync(cssFile, "utf-8"), { name: cssMeta.dataFileName }); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     try { |  | ||||||
|     const existingFileNames: Record<string, number> = format === "html" ? { navigation: 0, index: 1 } : {}; |     const existingFileNames: Record<string, number> = format === "html" ? { navigation: 0, index: 1 } : {}; | ||||||
|     const rootMeta = createNoteMeta(branch, { notePath: [] }, existingFileNames); |     const rootMeta = createNoteMeta(branch, { notePath: [] }, existingFileNames); | ||||||
|     if (!rootMeta) { |     if (!rootMeta) { | ||||||
| @@ -534,33 +389,9 @@ ${markdownContent}`; | |||||||
|         files: [rootMeta] |         files: [rootMeta] | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|         let navigationMeta: NoteMeta | null = null; |     provider.prepareMeta(metaFile); | ||||||
|         let indexMeta: NoteMeta | null = null; |  | ||||||
|         let cssMeta: NoteMeta | null = null; |  | ||||||
|  |  | ||||||
|         if (format === "html") { |  | ||||||
|             navigationMeta = { |  | ||||||
|                 noImport: true, |  | ||||||
|                 dataFileName: "navigation.html" |  | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             metaFile.files.push(navigationMeta); |  | ||||||
|  |  | ||||||
|             indexMeta = { |  | ||||||
|                 noImport: true, |  | ||||||
|                 dataFileName: "index.html" |  | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             metaFile.files.push(indexMeta); |  | ||||||
|  |  | ||||||
|             cssMeta = { |  | ||||||
|                 noImport: true, |  | ||||||
|                 dataFileName: "style.css" |  | ||||||
|             }; |  | ||||||
|  |  | ||||||
|             metaFile.files.push(cssMeta); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|  |     try { | ||||||
|         for (const noteMeta of Object.values(noteIdToMeta)) { |         for (const noteMeta of Object.values(noteIdToMeta)) { | ||||||
|             // filter out relations which are not inside this export |             // filter out relations which are not inside this export | ||||||
|             noteMeta.attributes = (noteMeta.attributes || []).filter((attr) => { |             noteMeta.attributes = (noteMeta.attributes || []).filter((attr) => { | ||||||
| @@ -584,34 +415,6 @@ ${markdownContent}`; | |||||||
|             } |             } | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         const metaFileJson = JSON.stringify(metaFile, null, "\t"); |  | ||||||
|  |  | ||||||
|         archive.append(metaFileJson, { name: "!!!meta.json" }); |  | ||||||
|  |  | ||||||
|         saveNote(rootMeta, ""); |  | ||||||
|  |  | ||||||
|         if (format === "html") { |  | ||||||
|             if (!navigationMeta || !indexMeta || !cssMeta) { |  | ||||||
|                 throw new Error("Missing meta."); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             saveNavigation(rootMeta, navigationMeta); |  | ||||||
|             saveIndex(rootMeta, indexMeta); |  | ||||||
|             saveCss(rootMeta, cssMeta); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         const note = branch.getNote(); |  | ||||||
|         const zipFileName = `${branch.prefix ? `${branch.prefix} - ` : ""}${note.getTitleOrProtected() || "note"}.zip`; |  | ||||||
|  |  | ||||||
|         if (setHeaders && "setHeader" in res) { |  | ||||||
|             res.setHeader("Content-Disposition", getContentDisposition(zipFileName)); |  | ||||||
|             res.setHeader("Content-Type", "application/zip"); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         archive.pipe(res); |  | ||||||
|         await archive.finalize(); |  | ||||||
|         taskContext.taskSucceeded(null); |  | ||||||
|     } catch (e: unknown) { |     } catch (e: unknown) { | ||||||
|         const message = `Export failed with error: ${e instanceof Error ? e.message : String(e)}`; |         const message = `Export failed with error: ${e instanceof Error ? e.message : String(e)}`; | ||||||
|         log.error(message); |         log.error(message); | ||||||
| @@ -623,9 +426,30 @@ ${markdownContent}`; | |||||||
|             res.status(500).send(message); |             res.status(500).send(message); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     const metaFileJson = JSON.stringify(metaFile, null, "\t"); | ||||||
|  |  | ||||||
|  |     archive.append(metaFileJson, { name: "!!!meta.json" }); | ||||||
|  |  | ||||||
|  |     saveNote(rootMeta, ""); | ||||||
|  |  | ||||||
|  |     provider.afterDone(rootMeta); | ||||||
|  |  | ||||||
|  |     const note = branch.getNote(); | ||||||
|  |     const zipFileName = `${branch.prefix ? `${branch.prefix} - ` : ""}${note.getTitleOrProtected()}.zip`; | ||||||
|  |  | ||||||
|  |     if (setHeaders && "setHeader" in res) { | ||||||
|  |         res.setHeader("Content-Disposition", getContentDisposition(zipFileName)); | ||||||
|  |         res.setHeader("Content-Type", "application/zip"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     archive.pipe(res); | ||||||
|  |     await archive.finalize(); | ||||||
|  |  | ||||||
|  |     taskContext.taskSucceeded(null); | ||||||
| } | } | ||||||
|  |  | ||||||
| async function exportToZipFile(noteId: string, format: "markdown" | "html", zipFilePath: string, zipExportOptions?: AdvancedExportOptions) { | async function exportToZipFile(noteId: string, format: ExportFormat, zipFilePath: string, zipExportOptions?: AdvancedExportOptions) { | ||||||
|     const fileOutputStream = fs.createWriteStream(zipFilePath); |     const fileOutputStream = fs.createWriteStream(zipFilePath); | ||||||
|     const taskContext = new TaskContext("no-progress-reporting", "export", null); |     const taskContext = new TaskContext("no-progress-reporting", "export", null); | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										89
									
								
								apps/server/src/services/export/zip/abstract_provider.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										89
									
								
								apps/server/src/services/export/zip/abstract_provider.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,89 @@ | |||||||
|  | import { Archiver } from "archiver"; | ||||||
|  | import type { default as NoteMeta, NoteMetaFile } from "../../meta/note_meta.js"; | ||||||
|  | import type BNote from "../../../becca/entities/bnote.js"; | ||||||
|  | import type BBranch from "../../../becca/entities/bbranch.js"; | ||||||
|  | import mimeTypes from "mime-types"; | ||||||
|  | import { NoteType } from "@triliumnext/commons"; | ||||||
|  |  | ||||||
|  | type RewriteLinksFn = (content: string, noteMeta: NoteMeta) => string; | ||||||
|  |  | ||||||
|  | export type ExportFormat = "html" | "markdown" | "share"; | ||||||
|  |  | ||||||
|  | export interface AdvancedExportOptions { | ||||||
|  |     /** | ||||||
|  |      * If `true`, then only the note's content will be kept. If `false` (default), then each page will have its own <html> template. | ||||||
|  |      */ | ||||||
|  |     skipHtmlTemplate?: boolean; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Provides a custom function to rewrite the links found in HTML or Markdown notes. This method is called for every note imported, if it's of the right type. | ||||||
|  |      * | ||||||
|  |      * @param originalRewriteLinks the original rewrite links function. Can be used to access the default behaviour without having to reimplement it. | ||||||
|  |      * @param getNoteTargetUrl the method to obtain a note's target URL, used internally by `originalRewriteLinks` but can be used here as well. | ||||||
|  |      * @returns a function to rewrite the links in HTML or Markdown notes. | ||||||
|  |      */ | ||||||
|  |     customRewriteLinks?: (originalRewriteLinks: RewriteLinksFn, getNoteTargetUrl: (targetNoteId: string, sourceMeta: NoteMeta) => string | null) => RewriteLinksFn; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export interface ZipExportProviderData { | ||||||
|  |     branch: BBranch; | ||||||
|  |     getNoteTargetUrl: (targetNoteId: string, sourceMeta: NoteMeta) => string | null; | ||||||
|  |     archive: Archiver; | ||||||
|  |     zipExportOptions?: AdvancedExportOptions; | ||||||
|  |     rewriteFn: RewriteLinksFn; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export abstract class ZipExportProvider { | ||||||
|  |     branch: BBranch; | ||||||
|  |     getNoteTargetUrl: (targetNoteId: string, sourceMeta: NoteMeta) => string | null; | ||||||
|  |     archive: Archiver; | ||||||
|  |     zipExportOptions?: AdvancedExportOptions; | ||||||
|  |     rewriteFn: RewriteLinksFn; | ||||||
|  |  | ||||||
|  |     constructor(data: ZipExportProviderData) { | ||||||
|  |         this.branch = data.branch; | ||||||
|  |         this.getNoteTargetUrl = data.getNoteTargetUrl; | ||||||
|  |         this.archive = data.archive; | ||||||
|  |         this.zipExportOptions = data.zipExportOptions; | ||||||
|  |         this.rewriteFn = data.rewriteFn; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     abstract prepareMeta(metaFile: NoteMetaFile): void; | ||||||
|  |     abstract prepareContent(title: string, content: string | Buffer, noteMeta: NoteMeta, note: BNote | undefined, branch: BBranch): string | Buffer; | ||||||
|  |     abstract afterDone(rootMeta: NoteMeta): void; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Determines the extension of the resulting file for a specific note type. | ||||||
|  |      * | ||||||
|  |      * @param type the type of the note. | ||||||
|  |      * @param mime the mime type of the note. | ||||||
|  |      * @param existingExtension the existing extension, including the leading period character. | ||||||
|  |      * @param format the format requested for export (e.g. HTML, Markdown). | ||||||
|  |      * @returns an extension *without* the leading period character, or `null` to preserve the existing extension instead. | ||||||
|  |      */ | ||||||
|  |     mapExtension(type: NoteType | null, mime: string, existingExtension: string, format: ExportFormat) { | ||||||
|  |         // the following two are handled specifically since we always want to have these extensions no matter the automatic detection | ||||||
|  |         // and/or existing detected extensions in the note name | ||||||
|  |         if (type === "text" && format === "markdown") { | ||||||
|  |             return "md"; | ||||||
|  |         } else if (type === "text" && format === "html") { | ||||||
|  |             return "html"; | ||||||
|  |         } else if (mime === "application/x-javascript" || mime === "text/javascript") { | ||||||
|  |             return "js"; | ||||||
|  |         } else if (type === "canvas" || mime === "application/json") { | ||||||
|  |             return "json"; | ||||||
|  |         } else if (existingExtension.length > 0) { | ||||||
|  |             // if the page already has an extension, then we'll just keep it | ||||||
|  |             return null; | ||||||
|  |         } else { | ||||||
|  |             if (mime?.toLowerCase()?.trim() === "image/jpg") { | ||||||
|  |                 return "jpg"; | ||||||
|  |             } else if (mime?.toLowerCase()?.trim() === "text/mermaid") { | ||||||
|  |                 return "txt"; | ||||||
|  |             } else { | ||||||
|  |                 return mimeTypes.extension(mime) || "dat"; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										176
									
								
								apps/server/src/services/export/zip/html.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										176
									
								
								apps/server/src/services/export/zip/html.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,176 @@ | |||||||
|  | import type NoteMeta from "../../meta/note_meta.js"; | ||||||
|  | import { escapeHtml, getResourceDir, isDev } from "../../utils"; | ||||||
|  | import html from "html"; | ||||||
|  | import { ZipExportProvider } from "./abstract_provider.js"; | ||||||
|  | import path from "path"; | ||||||
|  | import fs from "fs"; | ||||||
|  |  | ||||||
|  | export default class HtmlExportProvider extends ZipExportProvider { | ||||||
|  |  | ||||||
|  |     private navigationMeta: NoteMeta | null = null; | ||||||
|  |     private indexMeta: NoteMeta | null = null; | ||||||
|  |     private cssMeta: NoteMeta | null = null; | ||||||
|  |  | ||||||
|  |     prepareMeta(metaFile) { | ||||||
|  |         this.navigationMeta = { | ||||||
|  |             noImport: true, | ||||||
|  |             dataFileName: "navigation.html" | ||||||
|  |         }; | ||||||
|  |         metaFile.files.push(this.navigationMeta); | ||||||
|  |  | ||||||
|  |         this.indexMeta = { | ||||||
|  |             noImport: true, | ||||||
|  |             dataFileName: "index.html" | ||||||
|  |         }; | ||||||
|  |         metaFile.files.push(this.indexMeta); | ||||||
|  |  | ||||||
|  |         this.cssMeta = { | ||||||
|  |             noImport: true, | ||||||
|  |             dataFileName: "style.css" | ||||||
|  |         }; | ||||||
|  |         metaFile.files.push(this.cssMeta); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     prepareContent(title: string, content: string | Buffer, noteMeta: NoteMeta): string | Buffer { | ||||||
|  |         if (noteMeta.format === "html" && typeof content === "string") { | ||||||
|  |             if (!content.substr(0, 100).toLowerCase().includes("<html") && !this.zipExportOptions?.skipHtmlTemplate) { | ||||||
|  |                 if (!noteMeta?.notePath?.length) { | ||||||
|  |                     throw new Error("Missing note path."); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 const cssUrl = `${"../".repeat(noteMeta.notePath.length - 1)}style.css`; | ||||||
|  |                 const htmlTitle = escapeHtml(title); | ||||||
|  |  | ||||||
|  |                 // <base> element will make sure external links are openable - https://github.com/zadam/trilium/issues/1289#issuecomment-704066809 | ||||||
|  |                 content = `<html> | ||||||
|  | <head> | ||||||
|  |     <meta charset="utf-8"> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||||
|  |     <link rel="stylesheet" href="${cssUrl}"> | ||||||
|  |     <base target="_parent"> | ||||||
|  |     <title data-trilium-title>${htmlTitle}</title> | ||||||
|  | </head> | ||||||
|  | <body> | ||||||
|  |     <div class="content"> | ||||||
|  |     <h1 data-trilium-h1>${htmlTitle}</h1> | ||||||
|  |  | ||||||
|  |     <div class="ck-content">${content}</div> | ||||||
|  |     </div> | ||||||
|  | </body> | ||||||
|  | </html>`; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (content.length < 100_000) { | ||||||
|  |                 content = html.prettyPrint(content, { indent_size: 2 }) | ||||||
|  |             } | ||||||
|  |             content = this.rewriteFn(content as string, noteMeta); | ||||||
|  |             return content; | ||||||
|  |         } else { | ||||||
|  |             return content; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     afterDone(rootMeta: NoteMeta) { | ||||||
|  |         if (!this.navigationMeta || !this.indexMeta || !this.cssMeta) { | ||||||
|  |             throw new Error("Missing meta."); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         this.#saveNavigation(rootMeta, this.navigationMeta); | ||||||
|  |         this.#saveIndex(rootMeta, this.indexMeta); | ||||||
|  |         this.#saveCss(rootMeta, this.cssMeta); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #saveNavigationInner(rootMeta: NoteMeta, meta: NoteMeta) { | ||||||
|  |         let html = "<li>"; | ||||||
|  |  | ||||||
|  |         const escapedTitle = escapeHtml(`${meta.prefix ? `${meta.prefix} - ` : ""}${meta.title}`); | ||||||
|  |  | ||||||
|  |         if (meta.dataFileName && meta.noteId) { | ||||||
|  |             const targetUrl = this.getNoteTargetUrl(meta.noteId, rootMeta); | ||||||
|  |  | ||||||
|  |             html += `<a href="${targetUrl}" target="detail">${escapedTitle}</a>`; | ||||||
|  |         } else { | ||||||
|  |             html += escapedTitle; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if (meta.children && meta.children.length > 0) { | ||||||
|  |             html += "<ul>"; | ||||||
|  |  | ||||||
|  |             for (const child of meta.children) { | ||||||
|  |                 html += this.#saveNavigationInner(rootMeta, child); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             html += "</ul>"; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return `${html}</li>`; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #saveNavigation(rootMeta: NoteMeta, navigationMeta: NoteMeta) { | ||||||
|  |         if (!navigationMeta.dataFileName) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const fullHtml = `<html> | ||||||
|  |     <head> | ||||||
|  |         <meta charset="utf-8"> | ||||||
|  |         <link rel="stylesheet" href="style.css"> | ||||||
|  |     </head> | ||||||
|  |     <body> | ||||||
|  |         <ul>${this.#saveNavigationInner(rootMeta, rootMeta)}</ul> | ||||||
|  |     </body> | ||||||
|  |     </html>`; | ||||||
|  |         const prettyHtml = fullHtml.length < 100_000 ? html.prettyPrint(fullHtml, { indent_size: 2 }) : fullHtml; | ||||||
|  |  | ||||||
|  |         this.archive.append(prettyHtml, { name: navigationMeta.dataFileName }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #saveIndex(rootMeta: NoteMeta, indexMeta: NoteMeta) { | ||||||
|  |         let firstNonEmptyNote; | ||||||
|  |         let curMeta = rootMeta; | ||||||
|  |  | ||||||
|  |         if (!indexMeta.dataFileName) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         while (!firstNonEmptyNote) { | ||||||
|  |             if (curMeta.dataFileName && curMeta.noteId) { | ||||||
|  |                 firstNonEmptyNote = this.getNoteTargetUrl(curMeta.noteId, rootMeta); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             if (curMeta.children && curMeta.children.length > 0) { | ||||||
|  |                 curMeta = curMeta.children[0]; | ||||||
|  |             } else { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const fullHtml = `<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"> | ||||||
|  | <html> | ||||||
|  | <head> | ||||||
|  |     <meta charset="utf-8"> | ||||||
|  |     <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||||
|  | </head> | ||||||
|  | <frameset cols="25%,75%"> | ||||||
|  |     <frame name="navigation" src="navigation.html"> | ||||||
|  |     <frame name="detail" src="${firstNonEmptyNote}"> | ||||||
|  | </frameset> | ||||||
|  | </html>`; | ||||||
|  |  | ||||||
|  |         this.archive.append(fullHtml, { name: indexMeta.dataFileName }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #saveCss(rootMeta: NoteMeta, cssMeta: NoteMeta) { | ||||||
|  |         if (!cssMeta.dataFileName) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const cssFile = isDev | ||||||
|  |             ? path.join(__dirname, "../../../../../../node_modules/ckeditor5/dist/ckeditor5-content.css") | ||||||
|  |             : path.join(getResourceDir(), "ckeditor5-content.css"); | ||||||
|  |         const cssContent = fs.readFileSync(cssFile, "utf-8"); | ||||||
|  |         this.archive.append(cssContent, { name: cssMeta.dataFileName }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
							
								
								
									
										27
									
								
								apps/server/src/services/export/zip/markdown.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										27
									
								
								apps/server/src/services/export/zip/markdown.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,27 @@ | |||||||
|  | import NoteMeta from "../../meta/note_meta" | ||||||
|  | import { ZipExportProvider } from "./abstract_provider.js" | ||||||
|  | import mdService from "../markdown.js"; | ||||||
|  |  | ||||||
|  | export default class MarkdownExportProvider extends ZipExportProvider { | ||||||
|  |  | ||||||
|  |     prepareMeta() { } | ||||||
|  |  | ||||||
|  |     prepareContent(title: string, content: string | Buffer, noteMeta: NoteMeta): string | Buffer { | ||||||
|  |         if (noteMeta.format === "markdown" && typeof content === "string") { | ||||||
|  |             let markdownContent = mdService.toMarkdown(content); | ||||||
|  |  | ||||||
|  |             if (markdownContent.trim().length > 0 && !markdownContent.startsWith("# ")) { | ||||||
|  |                 markdownContent = `# ${title}\r | ||||||
|  | ${markdownContent}`; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             markdownContent = this.rewriteFn(markdownContent, noteMeta); | ||||||
|  |             return markdownContent; | ||||||
|  |         } else { | ||||||
|  |             return content; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     afterDone() { } | ||||||
|  |  | ||||||
|  | } | ||||||
							
								
								
									
										122
									
								
								apps/server/src/services/export/zip/share_theme.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										122
									
								
								apps/server/src/services/export/zip/share_theme.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,122 @@ | |||||||
|  | import { join } from "path"; | ||||||
|  | import NoteMeta, { NoteMetaFile } from "../../meta/note_meta"; | ||||||
|  | import { ExportFormat, ZipExportProvider } from "./abstract_provider.js"; | ||||||
|  | import { RESOURCE_DIR } from "../../resource_dir"; | ||||||
|  | import { getResourceDir, isDev } from "../../utils"; | ||||||
|  | import fs, { readdirSync } from "fs"; | ||||||
|  | import { renderNoteForExport } from "../../../share/content_renderer"; | ||||||
|  | import type BNote from "../../../becca/entities/bnote.js"; | ||||||
|  | import type BBranch from "../../../becca/entities/bbranch.js"; | ||||||
|  | import { getShareThemeAssetDir } from "../../../routes/assets"; | ||||||
|  |  | ||||||
|  | const shareThemeAssetDir = getShareThemeAssetDir(); | ||||||
|  |  | ||||||
|  | export default class ShareThemeExportProvider extends ZipExportProvider { | ||||||
|  |  | ||||||
|  |     private assetsMeta: NoteMeta[] = []; | ||||||
|  |     private indexMeta: NoteMeta | null = null; | ||||||
|  |  | ||||||
|  |     prepareMeta(metaFile: NoteMetaFile): void { | ||||||
|  |  | ||||||
|  |         const assets = [ | ||||||
|  |             "icon-color.svg" | ||||||
|  |         ]; | ||||||
|  |  | ||||||
|  |         for (const file of readdirSync(shareThemeAssetDir)) { | ||||||
|  |             assets.push(`assets/${file}`); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         for (const asset of assets) { | ||||||
|  |             const assetMeta = { | ||||||
|  |                 noImport: true, | ||||||
|  |                 dataFileName: asset | ||||||
|  |             }; | ||||||
|  |             this.assetsMeta.push(assetMeta); | ||||||
|  |             metaFile.files.push(assetMeta); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         this.indexMeta = { | ||||||
|  |             noImport: true, | ||||||
|  |             dataFileName: "index.html" | ||||||
|  |         }; | ||||||
|  |  | ||||||
|  |         metaFile.files.push(this.indexMeta); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     prepareContent(title: string, content: string | Buffer, noteMeta: NoteMeta, note: BNote | undefined, branch: BBranch): string | Buffer { | ||||||
|  |         if (!noteMeta?.notePath?.length) { | ||||||
|  |             throw new Error("Missing note path."); | ||||||
|  |         } | ||||||
|  |         const basePath = "../".repeat(noteMeta.notePath.length - 1); | ||||||
|  |  | ||||||
|  |         if (note) { | ||||||
|  |             content = renderNoteForExport(note, branch, basePath, noteMeta.notePath.slice(0, -1)); | ||||||
|  |             if (typeof content === "string") { | ||||||
|  |                 content = content.replace(/href="[^"]*\.\/([a-zA-Z0-9_\/]{12})[^"]*"/g, (match, id) => { | ||||||
|  |                     if (match.includes("/assets/")) return match; | ||||||
|  |                     return `href="#root/${id}"`; | ||||||
|  |                 }); | ||||||
|  |                 content = this.rewriteFn(content, noteMeta); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return content; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     afterDone(rootMeta: NoteMeta): void { | ||||||
|  |         this.#saveAssets(rootMeta, this.assetsMeta); | ||||||
|  |         this.#saveIndex(rootMeta); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     mapExtension(type: string | null, mime: string, existingExtension: string, format: ExportFormat): string | null { | ||||||
|  |         if (mime.startsWith("image/")) { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return "html"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #saveIndex(rootMeta: NoteMeta) { | ||||||
|  |         if (!this.indexMeta?.dataFileName) { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const note = this.branch.getNote(); | ||||||
|  |         const fullHtml = this.prepareContent(rootMeta.title ?? "", note.getContent(), rootMeta, note, this.branch); | ||||||
|  |         this.archive.append(fullHtml, { name: this.indexMeta.dataFileName }); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     #saveAssets(rootMeta: NoteMeta, assetsMeta: NoteMeta[]) { | ||||||
|  |         for (const assetMeta of assetsMeta) { | ||||||
|  |             if (!assetMeta.dataFileName) { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             let cssContent = getShareThemeAssets(assetMeta.dataFileName); | ||||||
|  |             this.archive.append(cssContent, { name: assetMeta.dataFileName }); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function getShareThemeAssets(nameWithExtension: string) { | ||||||
|  |     // Rename share.css to style.css. | ||||||
|  |     if (nameWithExtension === "style.css") { | ||||||
|  |         nameWithExtension = "share.css"; | ||||||
|  |     } else if (nameWithExtension === "script.js") { | ||||||
|  |         nameWithExtension = "share.js"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     let path: string | undefined; | ||||||
|  |     if (nameWithExtension === "icon-color.svg") { | ||||||
|  |         path = join(RESOURCE_DIR, "images", nameWithExtension); | ||||||
|  |     } else if (nameWithExtension.startsWith("assets")) { | ||||||
|  |         path = join(shareThemeAssetDir, nameWithExtension.replace(/^assets\//, "")); | ||||||
|  |     } else if (isDev) { | ||||||
|  |         path = join(getResourceDir(), "..", "..", "client", "dist", "src", nameWithExtension); | ||||||
|  |     } else { | ||||||
|  |         path = join(getResourceDir(), "public", "src", nameWithExtension); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return fs.readFileSync(path); | ||||||
|  | } | ||||||
| @@ -1,6 +1,7 @@ | |||||||
| import type { NoteType } from "@triliumnext/commons"; | import type { NoteType } from "@triliumnext/commons"; | ||||||
| import type AttachmentMeta from "./attachment_meta.js"; | import type AttachmentMeta from "./attachment_meta.js"; | ||||||
| import type AttributeMeta from "./attribute_meta.js"; | import type AttributeMeta from "./attribute_meta.js"; | ||||||
|  | import type { ExportFormat } from "../export/zip/abstract_provider.js"; | ||||||
|  |  | ||||||
| export interface NoteMetaFile { | export interface NoteMetaFile { | ||||||
|     formatVersion: number; |     formatVersion: number; | ||||||
| @@ -19,7 +20,7 @@ export default interface NoteMeta { | |||||||
|     type?: NoteType; |     type?: NoteType; | ||||||
|     mime?: string; |     mime?: string; | ||||||
|     /** 'html' or 'markdown', applicable to text notes only */ |     /** 'html' or 'markdown', applicable to text notes only */ | ||||||
|     format?: "html" | "markdown"; |     format?: ExportFormat; | ||||||
|     dataFileName?: string; |     dataFileName?: string; | ||||||
|     dirFileName?: string; |     dirFileName?: string; | ||||||
|     /** this file should not be imported (e.g., HTML navigation) */ |     /** this file should not be imported (e.g., HTML navigation) */ | ||||||
|   | |||||||
| @@ -1,10 +1,22 @@ | |||||||
| import { parse, HTMLElement, TextNode } from "node-html-parser"; | import { parse, HTMLElement, TextNode } from "node-html-parser"; | ||||||
| import shaca from "./shaca/shaca.js"; | import shaca from "./shaca/shaca.js"; | ||||||
| import assetPath from "../services/asset_path.js"; | import assetPath, { assetUrlFragment } from "../services/asset_path.js"; | ||||||
| import shareRoot from "./share_root.js"; | import shareRoot from "./share_root.js"; | ||||||
| import escapeHtml from "escape-html"; | import escapeHtml from "escape-html"; | ||||||
| import type SNote from "./shaca/entities/snote.js"; | import type SNote from "./shaca/entities/snote.js"; | ||||||
|  | import BNote from "../becca/entities/bnote.js"; | ||||||
|  | import type BBranch from "../becca/entities/bbranch.js"; | ||||||
| import { t } from "i18next"; | import { t } from "i18next"; | ||||||
|  | import SBranch from "./shaca/entities/sbranch.js"; | ||||||
|  | import options from "../services/options.js"; | ||||||
|  | import utils, { getResourceDir, isDev, safeExtractMessageAndStackFromError } from "../services/utils.js"; | ||||||
|  | import ejs from "ejs"; | ||||||
|  | import log from "../services/log.js"; | ||||||
|  | import { join } from "path"; | ||||||
|  | import { readFileSync } from "fs"; | ||||||
|  |  | ||||||
|  | const shareAdjustedAssetPath = isDev ? assetPath : `../${assetPath}`; | ||||||
|  | const templateCache: Map<string, string> = new Map(); | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Represents the output of the content renderer. |  * Represents the output of the content renderer. | ||||||
| @@ -16,7 +28,192 @@ export interface Result { | |||||||
|     isEmpty?: boolean; |     isEmpty?: boolean; | ||||||
| } | } | ||||||
|  |  | ||||||
| export function getContent(note: SNote) { | interface Subroot { | ||||||
|  |     note?: SNote | BNote; | ||||||
|  |     branch?: SBranch | BBranch | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function getSharedSubTreeRoot(note: SNote | BNote | undefined): Subroot { | ||||||
|  |     if (!note || note.noteId === shareRoot.SHARE_ROOT_NOTE_ID) { | ||||||
|  |         // share root itself is not shared | ||||||
|  |         return {}; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // every path leads to share root, but which one to choose? | ||||||
|  |     // for the sake of simplicity, URLs are not note paths | ||||||
|  |     const parentBranch = note.getParentBranches()[0]; | ||||||
|  |  | ||||||
|  |     if (note instanceof BNote) { | ||||||
|  |         return { | ||||||
|  |             note, | ||||||
|  |             branch: parentBranch | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     if (parentBranch.parentNoteId === shareRoot.SHARE_ROOT_NOTE_ID) { | ||||||
|  |         return { | ||||||
|  |             note, | ||||||
|  |             branch: parentBranch | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     return getSharedSubTreeRoot(parentBranch.getParentNote()); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function renderNoteForExport(note: BNote, parentBranch: BBranch, basePath: string, ancestors: string[]) { | ||||||
|  |     const subRoot: Subroot = { | ||||||
|  |         branch: parentBranch, | ||||||
|  |         note: parentBranch.getNote() | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     return renderNoteContentInternal(note, { | ||||||
|  |         subRoot, | ||||||
|  |         rootNoteId: parentBranch.noteId, | ||||||
|  |         cssToLoad: [ | ||||||
|  |             `${basePath}assets/styles.css`, | ||||||
|  |             `${basePath}assets/scripts.css`, | ||||||
|  |         ], | ||||||
|  |         jsToLoad: [ | ||||||
|  |             `${basePath}assets/scripts.js` | ||||||
|  |         ], | ||||||
|  |         logoUrl: `${basePath}icon-color.svg`, | ||||||
|  |         ancestors | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function renderNoteContent(note: SNote) { | ||||||
|  |     const subRoot = getSharedSubTreeRoot(note); | ||||||
|  |  | ||||||
|  |     const ancestors: string[] = []; | ||||||
|  |     let notePointer = note; | ||||||
|  |     while (notePointer.parents[0]?.noteId !== subRoot.note?.noteId) { | ||||||
|  |         const pointerParent = notePointer.parents[0]; | ||||||
|  |         if (!pointerParent) { | ||||||
|  |             break; | ||||||
|  |         } | ||||||
|  |         ancestors.push(pointerParent.noteId); | ||||||
|  |         notePointer = pointerParent; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Determine CSS to load. | ||||||
|  |     const cssToLoad: string[] = []; | ||||||
|  |     if (!note.isLabelTruthy("shareOmitDefaultCss")) { | ||||||
|  |         cssToLoad.push(`assets/styles.css`); | ||||||
|  |         cssToLoad.push(`assets/scripts.css`); | ||||||
|  |     } | ||||||
|  |     for (const cssRelation of note.getRelations("shareCss")) { | ||||||
|  |         cssToLoad.push(`api/notes/${cssRelation.value}/download`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Determine JS to load. | ||||||
|  |     const jsToLoad: string[] = [ | ||||||
|  |         "assets/scripts.js" | ||||||
|  |     ]; | ||||||
|  |     for (const jsRelation of note.getRelations("shareJs")) { | ||||||
|  |         jsToLoad.push(`api/notes/${jsRelation.value}/download`); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const customLogoId = note.getRelation("shareLogo")?.value; | ||||||
|  |     const logoUrl = customLogoId ? `api/images/${customLogoId}/image.png` : `../${assetUrlFragment}/images/icon-color.svg`; | ||||||
|  |  | ||||||
|  |     return renderNoteContentInternal(note, { | ||||||
|  |         subRoot, | ||||||
|  |         rootNoteId: "_share", | ||||||
|  |         cssToLoad, | ||||||
|  |         jsToLoad, | ||||||
|  |         logoUrl, | ||||||
|  |         ancestors | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | interface RenderArgs { | ||||||
|  |     subRoot: Subroot; | ||||||
|  |     rootNoteId: string; | ||||||
|  |     cssToLoad: string[]; | ||||||
|  |     jsToLoad: string[]; | ||||||
|  |     logoUrl: string; | ||||||
|  |     ancestors: string[]; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function renderNoteContentInternal(note: SNote | BNote, renderArgs: RenderArgs) { | ||||||
|  |     const { header, content, isEmpty } = getContent(note); | ||||||
|  |     const showLoginInShareTheme = options.getOption("showLoginInShareTheme"); | ||||||
|  |     const opts = { | ||||||
|  |         note, | ||||||
|  |         header, | ||||||
|  |         content, | ||||||
|  |         isEmpty, | ||||||
|  |         assetPath: shareAdjustedAssetPath, | ||||||
|  |         assetUrlFragment, | ||||||
|  |         showLoginInShareTheme, | ||||||
|  |         t, | ||||||
|  |         isDev, | ||||||
|  |         utils, | ||||||
|  |         ...renderArgs | ||||||
|  |     }; | ||||||
|  |  | ||||||
|  |     // Check if the user has their own template. | ||||||
|  |     if (note.hasRelation("shareTemplate")) { | ||||||
|  |         // Get the template note and content | ||||||
|  |         const templateId = note.getRelation("shareTemplate")?.value; | ||||||
|  |         const templateNote = templateId && shaca.getNote(templateId); | ||||||
|  |  | ||||||
|  |         // Make sure the note type is correct | ||||||
|  |         if (templateNote && templateNote.type === "code" && templateNote.mime === "application/x-ejs") { | ||||||
|  |             // EJS caches the result of this so we don't need to pre-cache | ||||||
|  |             const includer = (path: string) => { | ||||||
|  |                 const childNote = templateNote.children.find((n) => path === n.title); | ||||||
|  |                 if (!childNote) throw new Error(`Unable to find child note: ${path}.`); | ||||||
|  |                 if (childNote.type !== "code" || childNote.mime !== "application/x-ejs") throw new Error("Incorrect child note type."); | ||||||
|  |  | ||||||
|  |                 const template = childNote.getContent(); | ||||||
|  |                 if (typeof template !== "string") throw new Error("Invalid template content type."); | ||||||
|  |  | ||||||
|  |                 return { template }; | ||||||
|  |             }; | ||||||
|  |  | ||||||
|  |             // Try to render user's template, w/ fallback to default view | ||||||
|  |             try { | ||||||
|  |                 const content = templateNote.getContent(); | ||||||
|  |                 if (typeof content === "string") { | ||||||
|  |                     return ejs.render(content, opts, { includer }); | ||||||
|  |                 } | ||||||
|  |             } catch (e: unknown) { | ||||||
|  |                 const [errMessage, errStack] = safeExtractMessageAndStackFromError(e); | ||||||
|  |                 log.error(`Rendering user provided share template (${templateId}) threw exception ${errMessage} with stacktrace: ${errStack}`); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     // Render with the default view otherwise. | ||||||
|  |     const templatePath = getDefaultTemplatePath("page"); | ||||||
|  |     return ejs.render(readTemplate(templatePath), opts, { | ||||||
|  |         includer: (path) => { | ||||||
|  |             // Path is relative to apps/server/dist/assets/views | ||||||
|  |             return { template: readTemplate(getDefaultTemplatePath(path)) }; | ||||||
|  |         } | ||||||
|  |     }); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function getDefaultTemplatePath(template: string) { | ||||||
|  |     // Path is relative to apps/server/dist/assets/views | ||||||
|  |     return process.env.NODE_ENV === "development" | ||||||
|  |         ? join(__dirname, `../../../../packages/share-theme/src/templates/${template}.ejs`) | ||||||
|  |         : join(getResourceDir(), `share-theme/templates/${template}.ejs`); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | function readTemplate(path: string) { | ||||||
|  |     const cachedTemplate = templateCache.get(path); | ||||||
|  |     if (cachedTemplate) { | ||||||
|  |         return cachedTemplate; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const templateString = readFileSync(path, "utf-8"); | ||||||
|  |     templateCache.set(path, templateString); | ||||||
|  |     return templateString; | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export function getContent(note: SNote | BNote) { | ||||||
|     if (note.isProtected) { |     if (note.isProtected) { | ||||||
|         return { |         return { | ||||||
|             header: "", |             header: "", | ||||||
| @@ -65,7 +262,7 @@ function renderIndex(result: Result) { | |||||||
|     result.content += "</ul>"; |     result.content += "</ul>"; | ||||||
| } | } | ||||||
|  |  | ||||||
| function renderText(result: Result, note: SNote) { | function renderText(result: Result, note: SNote | BNote) { | ||||||
|     if (typeof result.content !== "string") return; |     if (typeof result.content !== "string") return; | ||||||
|     const document = parse(result.content || ""); |     const document = parse(result.content || ""); | ||||||
|  |  | ||||||
| @@ -174,7 +371,7 @@ export function renderCode(result: Result) { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| function renderMermaid(result: Result, note: SNote) { | function renderMermaid(result: Result, note: SNote | BNote) { | ||||||
|     if (typeof result.content !== "string") { |     if (typeof result.content !== "string") { | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
| @@ -188,11 +385,11 @@ function renderMermaid(result: Result, note: SNote) { | |||||||
| </details>`; | </details>`; | ||||||
| } | } | ||||||
|  |  | ||||||
| function renderImage(result: Result, note: SNote) { | function renderImage(result: Result, note: SNote | BNote) { | ||||||
|     result.content = `<img src="api/images/${note.noteId}/${note.encodedTitle}?${note.utcDateModified}">`; |     result.content = `<img src="api/images/${note.noteId}/${note.encodedTitle}?${note.utcDateModified}">`; | ||||||
| } | } | ||||||
|  |  | ||||||
| function renderFile(note: SNote, result: Result) { | function renderFile(note: SNote | BNote, result: Result) { | ||||||
|     if (note.mime === "application/pdf") { |     if (note.mime === "application/pdf") { | ||||||
|         result.content = `<iframe class="pdf-view" src="api/notes/${note.noteId}/view"></iframe>`; |         result.content = `<iframe class="pdf-view" src="api/notes/${note.noteId}/view"></iframe>`; | ||||||
|     } else { |     } else { | ||||||
|   | |||||||
| @@ -4,41 +4,12 @@ import type { Request, Response, Router } from "express"; | |||||||
|  |  | ||||||
| import shaca from "./shaca/shaca.js"; | import shaca from "./shaca/shaca.js"; | ||||||
| import shacaLoader from "./shaca/shaca_loader.js"; | import shacaLoader from "./shaca/shaca_loader.js"; | ||||||
| import shareRoot from "./share_root.js"; |  | ||||||
| import contentRenderer from "./content_renderer.js"; |  | ||||||
| import assetPath, { assetUrlFragment } from "../services/asset_path.js"; |  | ||||||
| import appPath from "../services/app_path.js"; |  | ||||||
| import searchService from "../services/search/services/search.js"; | import searchService from "../services/search/services/search.js"; | ||||||
| import SearchContext from "../services/search/search_context.js"; | import SearchContext from "../services/search/search_context.js"; | ||||||
| import log from "../services/log.js"; |  | ||||||
| import type SNote from "./shaca/entities/snote.js"; | import type SNote from "./shaca/entities/snote.js"; | ||||||
| import type SBranch from "./shaca/entities/sbranch.js"; |  | ||||||
| import type SAttachment from "./shaca/entities/sattachment.js"; | import type SAttachment from "./shaca/entities/sattachment.js"; | ||||||
| import utils, { isDev, safeExtractMessageAndStackFromError } from "../services/utils.js"; | import { renderNoteContent } from "./content_renderer.js"; | ||||||
| import options from "../services/options.js"; | import utils from "../services/utils.js"; | ||||||
| import { t } from "i18next"; |  | ||||||
| import ejs from "ejs"; |  | ||||||
| import { join } from "path"; |  | ||||||
|  |  | ||||||
| function getSharedSubTreeRoot(note: SNote): { note?: SNote; branch?: SBranch } { |  | ||||||
|     if (note.noteId === shareRoot.SHARE_ROOT_NOTE_ID) { |  | ||||||
|         // share root itself is not shared |  | ||||||
|         return {}; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     // every path leads to share root, but which one to choose? |  | ||||||
|     // for the sake of simplicity, URLs are not note paths |  | ||||||
|     const parentBranch = note.getParentBranches()[0]; |  | ||||||
|  |  | ||||||
|     if (parentBranch.parentNoteId === shareRoot.SHARE_ROOT_NOTE_ID) { |  | ||||||
|         return { |  | ||||||
|             note, |  | ||||||
|             branch: parentBranch |  | ||||||
|         }; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     return getSharedSubTreeRoot(parentBranch.getParentNote()); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| function addNoIndexHeader(note: SNote, res: Response) { | function addNoIndexHeader(note: SNote, res: Response) { | ||||||
|     if (note.isLabelTruthy("shareDisallowRobotIndexing")) { |     if (note.isLabelTruthy("shareDisallowRobotIndexing")) { | ||||||
| @@ -109,8 +80,7 @@ function renderImageAttachment(image: SNote, res: Response, attachmentName: stri | |||||||
|     let svgString = "<svg/>"; |     let svgString = "<svg/>"; | ||||||
|     const attachment = image.getAttachmentByTitle(attachmentName); |     const attachment = image.getAttachmentByTitle(attachmentName); | ||||||
|     if (!attachment) { |     if (!attachment) { | ||||||
|         res.status(404); |  | ||||||
|         renderDefault(res, "404"); |  | ||||||
|         return; |         return; | ||||||
|     } |     } | ||||||
|     const content = attachment.getContent(); |     const content = attachment.getContent(); | ||||||
| @@ -138,12 +108,19 @@ function renderImageAttachment(image: SNote, res: Response, attachmentName: stri | |||||||
|     res.send(svg); |     res.send(svg); | ||||||
| } | } | ||||||
|  |  | ||||||
|  | function render404(res: Response) { | ||||||
|  |     res.status(404); | ||||||
|  |     const shareThemePath = `../../share-theme/templates/404.ejs`; | ||||||
|  |     res.render(shareThemePath); | ||||||
|  | } | ||||||
|  |  | ||||||
| function register(router: Router) { | function register(router: Router) { | ||||||
|  |  | ||||||
|     function renderNote(note: SNote, req: Request, res: Response) { |     function renderNote(note: SNote, req: Request, res: Response) { | ||||||
|         if (!note) { |         if (!note) { | ||||||
|             console.log("Unable to find note ", note); |             console.log("Unable to find note ", note); | ||||||
|             res.status(404); |             res.status(404); | ||||||
|             renderDefault(res, "404"); |             render404(res); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -161,63 +138,7 @@ function register(router: Router) { | |||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         const { header, content, isEmpty } = contentRenderer.getContent(note); |         res.send(renderNoteContent(note)); | ||||||
|         const subRoot = getSharedSubTreeRoot(note); |  | ||||||
|         const showLoginInShareTheme = options.getOption("showLoginInShareTheme"); |  | ||||||
|         const opts = { |  | ||||||
|             note, |  | ||||||
|             header, |  | ||||||
|             content, |  | ||||||
|             isEmpty, |  | ||||||
|             subRoot, |  | ||||||
|             assetPath: isDev ? assetPath : `../${assetPath}`, |  | ||||||
|             assetUrlFragment, |  | ||||||
|             appPath: isDev ? appPath : `../${appPath}`, |  | ||||||
|             showLoginInShareTheme, |  | ||||||
|             t, |  | ||||||
|             isDev, |  | ||||||
|             utils |  | ||||||
|         }; |  | ||||||
|         let useDefaultView = true; |  | ||||||
|  |  | ||||||
|         // Check if the user has their own template |  | ||||||
|         if (note.hasRelation("shareTemplate")) { |  | ||||||
|             // Get the template note and content |  | ||||||
|             const templateId = note.getRelation("shareTemplate")?.value; |  | ||||||
|             const templateNote = templateId && shaca.getNote(templateId); |  | ||||||
|  |  | ||||||
|             // Make sure the note type is correct |  | ||||||
|             if (templateNote && templateNote.type === "code" && templateNote.mime === "application/x-ejs") { |  | ||||||
|                 // EJS caches the result of this so we don't need to pre-cache |  | ||||||
|                 const includer = (path: string) => { |  | ||||||
|                     const childNote = templateNote.children.find((n) => path === n.title); |  | ||||||
|                     if (!childNote) throw new Error(`Unable to find child note: ${path}.`); |  | ||||||
|                     if (childNote.type !== "code" || childNote.mime !== "application/x-ejs") throw new Error("Incorrect child note type."); |  | ||||||
|  |  | ||||||
|                     const template = childNote.getContent(); |  | ||||||
|                     if (typeof template !== "string") throw new Error("Invalid template content type."); |  | ||||||
|  |  | ||||||
|                     return { template }; |  | ||||||
|                 }; |  | ||||||
|  |  | ||||||
|                 // Try to render user's template, w/ fallback to default view |  | ||||||
|                 try { |  | ||||||
|                     const content = templateNote.getContent(); |  | ||||||
|                     if (typeof content === "string") { |  | ||||||
|                         const ejsResult = ejs.render(content, opts, { includer }); |  | ||||||
|                         res.send(ejsResult); |  | ||||||
|                         useDefaultView = false; // Rendering went okay, don't use default view |  | ||||||
|                     } |  | ||||||
|                 } catch (e: unknown) { |  | ||||||
|                     const [errMessage, errStack] = safeExtractMessageAndStackFromError(e); |  | ||||||
|                     log.error(`Rendering user provided share template (${templateId}) threw exception ${errMessage} with stacktrace: ${errStack}`); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if (useDefaultView) { |  | ||||||
|             renderDefault(res, "page", opts); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     router.get("/share/", (req, res) => { |     router.get("/share/", (req, res) => { | ||||||
| @@ -401,14 +322,6 @@ function register(router: Router) { | |||||||
|     }); |     }); | ||||||
| } | } | ||||||
|  |  | ||||||
| function renderDefault(res: Response<any, Record<string, any>>, template: "page" | "404", opts: any = {}) { |  | ||||||
|     // Path is relative to apps/server/dist/assets/views |  | ||||||
|     const shareThemePath = process.env.NODE_ENV === "development" |  | ||||||
|         ? join(__dirname, `../../../../packages/share-theme/src/templates/${template}.ejs`) |  | ||||||
|         : `../../share-theme/templates/${template}.ejs`; |  | ||||||
|     res.render(shareThemePath, opts); |  | ||||||
| } |  | ||||||
|  |  | ||||||
| export default { | export default { | ||||||
|     register |     register | ||||||
| }; | }; | ||||||
|   | |||||||
| @@ -21,6 +21,11 @@ | |||||||
|     "Zerebos <me@zerebos.com>" |     "Zerebos <me@zerebos.com>" | ||||||
|   ], |   ], | ||||||
|   "license": "Apache-2.0", |   "license": "Apache-2.0", | ||||||
|  |   "dependencies": { | ||||||
|  |     "katex": "0.16.25", | ||||||
|  |     "mermaid": "11.12.0", | ||||||
|  |     "boxicons": "2.1.4" | ||||||
|  |   }, | ||||||
|   "devDependencies": { |   "devDependencies": { | ||||||
|     "@digitak/esrun": "3.2.26", |     "@digitak/esrun": "3.2.26", | ||||||
|     "@types/swagger-ui": "5.21.1", |     "@types/swagger-ui": "5.21.1", | ||||||
|   | |||||||
| @@ -1,4 +1,3 @@ | |||||||
| import fs from "node:fs"; |  | ||||||
| import path from "node:path"; | import path from "node:path"; | ||||||
| // import {fileURLToPath} from "node:url"; | // import {fileURLToPath} from "node:url"; | ||||||
|  |  | ||||||
| @@ -51,15 +50,18 @@ async function runBuild() { | |||||||
|     await esbuild.build({ |     await esbuild.build({ | ||||||
|         entryPoints: entryPoints, |         entryPoints: entryPoints, | ||||||
|         bundle: true, |         bundle: true, | ||||||
|  |         splitting: true, | ||||||
|         outdir: path.join(rootDir, "dist"), |         outdir: path.join(rootDir, "dist"), | ||||||
|         format: "cjs", |         format: "esm", | ||||||
|         target: ["chrome96"], |         target: ["chrome96"], | ||||||
|         loader: { |         loader: { | ||||||
|             ".png": "dataurl", |             ".png": "dataurl", | ||||||
|             ".gif": "dataurl", |             ".gif": "dataurl", | ||||||
|             ".woff": "dataurl", |             ".woff": "file", | ||||||
|             ".woff2": "dataurl", |             ".woff2": "file", | ||||||
|             ".ttf": "dataurl", |             ".ttf": "file", | ||||||
|  |             ".eot": "empty", | ||||||
|  |             ".svg": "empty", | ||||||
|             ".html": "text", |             ".html": "text", | ||||||
|             ".css": "css" |             ".css": "css" | ||||||
|         }, |         }, | ||||||
|   | |||||||
| @@ -3,6 +3,10 @@ import setupExpanders from "./modules/expanders"; | |||||||
| import setupMobileMenu from "./modules/mobile"; | import setupMobileMenu from "./modules/mobile"; | ||||||
| import setupSearch from "./modules/search"; | import setupSearch from "./modules/search"; | ||||||
| import setupThemeSelector from "./modules/theme"; | import setupThemeSelector from "./modules/theme"; | ||||||
|  | import setupMermaid from "./modules/mermaid"; | ||||||
|  | import setupMath from "./modules/math"; | ||||||
|  | import api from "./modules/api"; | ||||||
|  | import "boxicons/css/boxicons.min.css"; | ||||||
|  |  | ||||||
| function $try<T extends (...a: unknown[]) => unknown>(func: T, ...args: Parameters<T>) { | function $try<T extends (...a: unknown[]) => unknown>(func: T, ...args: Parameters<T>) { | ||||||
|     try { |     try { | ||||||
| @@ -13,8 +17,39 @@ function $try<T extends (...a: unknown[]) => unknown>(func: T, ...args: Paramete | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | Object.assign(window, api); | ||||||
| $try(setupThemeSelector); | $try(setupThemeSelector); | ||||||
| $try(setupToC); | $try(setupToC); | ||||||
| $try(setupExpanders); | $try(setupExpanders); | ||||||
| $try(setupMobileMenu); | $try(setupMobileMenu); | ||||||
| $try(setupSearch); | $try(setupSearch); | ||||||
|  |  | ||||||
|  | function setupTextNote() { | ||||||
|  |     $try(setupMermaid); | ||||||
|  |     $try(setupMath); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | document.addEventListener( | ||||||
|  |     "DOMContentLoaded", | ||||||
|  |     () => { | ||||||
|  |         const noteType = determineNoteType(); | ||||||
|  |  | ||||||
|  |         if (noteType === "text") { | ||||||
|  |             setupTextNote(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         const toggleMenuButton = document.getElementById("toggleMenuButton"); | ||||||
|  |         const layout = document.getElementById("layout"); | ||||||
|  |  | ||||||
|  |         if (toggleMenuButton && layout) { | ||||||
|  |             toggleMenuButton.addEventListener("click", () => layout.classList.toggle("showMenu")); | ||||||
|  |         } | ||||||
|  |     }, | ||||||
|  |     false | ||||||
|  | ); | ||||||
|  |  | ||||||
|  | function determineNoteType() { | ||||||
|  |     const bodyClass = document.body.className; | ||||||
|  |     const match = bodyClass.match(/type-([^\s]+)/); | ||||||
|  |     return match ? match[1] : null; | ||||||
|  | } | ||||||
|   | |||||||
							
								
								
									
										18
									
								
								packages/share-theme/src/scripts/modules/api.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								packages/share-theme/src/scripts/modules/api.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | /** | ||||||
|  |  * Fetch note with given ID from backend | ||||||
|  |  * | ||||||
|  |  * @param noteId of the given note to be fetched. If false, fetches current note. | ||||||
|  |  */ | ||||||
|  | async function fetchNote(noteId: string | null = null) { | ||||||
|  |     if (!noteId) { | ||||||
|  |         noteId = document.body.getAttribute("data-note-id"); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const resp = await fetch(`api/notes/${noteId}`); | ||||||
|  |  | ||||||
|  |     return await resp.json(); | ||||||
|  | } | ||||||
|  |  | ||||||
|  | export default { | ||||||
|  |     fetchNote | ||||||
|  | }; | ||||||
							
								
								
									
										16
									
								
								packages/share-theme/src/scripts/modules/math.ts
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										16
									
								
								packages/share-theme/src/scripts/modules/math.ts
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,16 @@ | |||||||
|  | import "katex/dist/katex.min.css"; | ||||||
|  |  | ||||||
|  | export default async function setupMath() { | ||||||
|  |     const anyMathBlock = document.querySelector("#content .math-tex"); | ||||||
|  |     if (!anyMathBlock) { | ||||||
|  |         return; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     const renderMathInElement = (await import("katex/contrib/auto-render")).default; | ||||||
|  |     await import("katex/contrib/mhchem"); | ||||||
|  |  | ||||||
|  |     const contentEl = document.getElementById("content"); | ||||||
|  |     if (!contentEl) return; | ||||||
|  |     renderMathInElement(contentEl); | ||||||
|  |     document.body.classList.add("math-loaded"); | ||||||
|  | } | ||||||
| @@ -1,7 +1,12 @@ | |||||||
| import mermaid from "mermaid"; | export default async function setupMermaid() { | ||||||
|  |     const mermaidEls = document.querySelectorAll("#content pre code.language-mermaid"); | ||||||
|  |     if (mermaidEls.length === 0) { | ||||||
|  |         return; | ||||||
|  |     } | ||||||
| 
 | 
 | ||||||
| export default function setupMermaid() { |     const mermaid = (await import("mermaid")).default; | ||||||
|     for (const codeBlock of document.querySelectorAll("#content pre code.language-mermaid")) { | 
 | ||||||
|  |     for (const codeBlock of mermaidEls) { | ||||||
|         const parentPre = codeBlock.parentElement; |         const parentPre = codeBlock.parentElement; | ||||||
|         if (!parentPre) { |         if (!parentPre) { | ||||||
|             continue; |             continue; | ||||||
| @@ -47,3 +47,7 @@ | |||||||
| #content img { | #content img { | ||||||
|     max-width: 100%; |     max-width: 100%; | ||||||
| } | } | ||||||
|  |  | ||||||
|  | body:not(.math-loaded) .math-tex { | ||||||
|  |     visibility: hidden; | ||||||
|  | } | ||||||
| @@ -30,17 +30,11 @@ | |||||||
|     <meta name="viewport" content="width=device-width, initial-scale=1"> |     <meta name="viewport" content="width=device-width, initial-scale=1"> | ||||||
|  |  | ||||||
|     <link rel="shortcut icon" href="<% if (note.hasRelation("shareFavicon")) { %>api/notes/<%= note.getRelation("shareFavicon").value %>/download<% } else { %>../favicon.ico<% } %>"> |     <link rel="shortcut icon" href="<% if (note.hasRelation("shareFavicon")) { %>api/notes/<%= note.getRelation("shareFavicon").value %>/download<% } else { %>../favicon.ico<% } %>"> | ||||||
|     <script src="../<%= appPath %>/share.js" type="module"></script> |     <% for (const url of cssToLoad) { %> | ||||||
|     <% if (!isDev && !note.isLabelTruthy("shareOmitDefaultCss")) { %> |         <link href="<%= url %>" rel="stylesheet"> | ||||||
|         <link href="<%= assetPath %>/src/share.css" rel="stylesheet"> |  | ||||||
|         <link href="<%= assetPath %>/src/boxicons.css" rel="stylesheet"> |  | ||||||
|     <% } %> |     <% } %> | ||||||
|  |     <% for (const url of jsToLoad) { %> | ||||||
|     <% for (const cssRelation of note.getRelations("shareCss")) { %> |         <script type="module" src="<%= url %>"></script> | ||||||
|         <link href="api/notes/<%= cssRelation.value %>/download" rel="stylesheet"> |  | ||||||
|     <% } %> |  | ||||||
|     <% for (const jsRelation of note.getRelations("shareJs")) { %> |  | ||||||
|         <script type="module" src="api/notes/<%= jsRelation.value %>/download"></script> |  | ||||||
|     <% } %> |     <% } %> | ||||||
|     <% if (note.hasLabel("shareDisallowRobotIndexing")) { %> |     <% if (note.hasLabel("shareDisallowRobotIndexing")) { %> | ||||||
|         <meta name="robots" content="noindex,follow" /> |         <meta name="robots" content="noindex,follow" /> | ||||||
| @@ -80,8 +74,6 @@ | |||||||
|     <%- renderSnippets("head:end") %> |     <%- renderSnippets("head:end") %> | ||||||
| </head> | </head> | ||||||
| <% | <% | ||||||
| const customLogoId = subRoot.note.getRelation("shareLogo")?.value; |  | ||||||
| const logoUrl = customLogoId ? `api/images/${customLogoId}/image.png` : `../${assetUrlFragment}/images/icon-color.svg`; |  | ||||||
| const logoWidth = subRoot.note.getLabelValue("shareLogoWidth") ?? 53; | const logoWidth = subRoot.note.getLabelValue("shareLogoWidth") ?? 53; | ||||||
| const logoHeight = subRoot.note.getLabelValue("shareLogoHeight") ?? 40; | const logoHeight = subRoot.note.getLabelValue("shareLogoHeight") ?? 40; | ||||||
| const mobileLogoHeight = logoHeight && logoWidth ? 32 / (logoWidth / logoHeight) : ""; | const mobileLogoHeight = logoHeight && logoWidth ? 32 / (logoWidth / logoHeight) : ""; | ||||||
| @@ -131,16 +123,7 @@ content = content.replaceAll(headingRe, (...match) => { | |||||||
|             </div> |             </div> | ||||||
|         <% if (hasTree) { %> |         <% if (hasTree) { %> | ||||||
|             <nav id="menu"> |             <nav id="menu"> | ||||||
|                 <% |                 <%- include("tree_item", {note: subRoot.note, activeNote: note, subRoot: subRoot, ancestors}) %> | ||||||
|                 const ancestors = []; |  | ||||||
|                 let notePointer = note; |  | ||||||
|                 while (notePointer.parents[0].noteId !== "_share") { |  | ||||||
|                     const pointerParent = notePointer.parents[0]; |  | ||||||
|                     ancestors.push(pointerParent.noteId); |  | ||||||
|                     notePointer = pointerParent; |  | ||||||
|                 } |  | ||||||
|                 %> |  | ||||||
|                 <%- include("tree_item", {note: subRoot.note, activeNote: note, subRoot: subRoot, ancestors: ancestors}) %> |  | ||||||
|             </nav> |             </nav> | ||||||
|         <% } %>    |         <% } %>    | ||||||
|         </div> |         </div> | ||||||
|   | |||||||
| @@ -1,7 +1,16 @@ | |||||||
| <% | <% | ||||||
| const linkClass = `type-${note.type}` + (activeNote.noteId === note.noteId ? " active" : ""); | const linkClass = `type-${note.type}` + (activeNote.noteId === note.noteId ? " active" : ""); | ||||||
| const isExternalLink = note.hasLabel("shareExternal"); | const isExternalLink = note.hasLabel("shareExternal"); | ||||||
| const linkHref = isExternalLink ? note.getLabelValue("shareExternal") : `./${note.shareId}`; | let linkHref; | ||||||
|  |  | ||||||
|  | if (isExternalLink) { | ||||||
|  |     linkHref = note.getLabelValue("shareExternal"); | ||||||
|  | } else if (note.shareId) { | ||||||
|  |     linkHref = `./${note.shareId}`; | ||||||
|  | } else { | ||||||
|  |     linkHref = `#${note.getBestNotePath().join("/")}`; | ||||||
|  | } | ||||||
|  |  | ||||||
| const target = isExternalLink ? ` target="_blank" rel="noopener noreferrer"` : ""; | const target = isExternalLink ? ` target="_blank" rel="noopener noreferrer"` : ""; | ||||||
| %> | %> | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										5
									
								
								packages/share-theme/src/types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								packages/share-theme/src/types.d.ts
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,5 @@ | |||||||
|  | declare module "katex/contrib/auto-render" { | ||||||
|  |     export default function renderMathInElement(elem: HTMLElement, options?: {}) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | declare module "katex/contrib/mhchem" {} | ||||||
							
								
								
									
										14
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							
							
						
						
									
										14
									
								
								pnpm-lock.yaml
									
									
									
										generated
									
									
									
								
							| @@ -1328,6 +1328,16 @@ importers: | |||||||
|         version: 1.2.0 |         version: 1.2.0 | ||||||
|  |  | ||||||
|   packages/share-theme: |   packages/share-theme: | ||||||
|  |     dependencies: | ||||||
|  |       boxicons: | ||||||
|  |         specifier: 2.1.4 | ||||||
|  |         version: 2.1.4 | ||||||
|  |       katex: | ||||||
|  |         specifier: 0.16.25 | ||||||
|  |         version: 0.16.25 | ||||||
|  |       mermaid: | ||||||
|  |         specifier: 11.12.0 | ||||||
|  |         version: 11.12.0 | ||||||
|     devDependencies: |     devDependencies: | ||||||
|       '@digitak/esrun': |       '@digitak/esrun': | ||||||
|         specifier: 3.2.26 |         specifier: 3.2.26 | ||||||
| @@ -15335,6 +15345,8 @@ snapshots: | |||||||
|       '@ckeditor/ckeditor5-utils': 47.1.0 |       '@ckeditor/ckeditor5-utils': 47.1.0 | ||||||
|       ckeditor5: 47.1.0(patch_hash=8331a09d41443b39ea1c784daaccfeb0da4f9065ed556e7de92e9c77edd9eb41) |       ckeditor5: 47.1.0(patch_hash=8331a09d41443b39ea1c784daaccfeb0da4f9065ed556e7de92e9c77edd9eb41) | ||||||
|       es-toolkit: 1.39.5 |       es-toolkit: 1.39.5 | ||||||
|  |     transitivePeerDependencies: | ||||||
|  |       - supports-color | ||||||
|  |  | ||||||
|   '@ckeditor/ckeditor5-editor-multi-root@47.1.0': |   '@ckeditor/ckeditor5-editor-multi-root@47.1.0': | ||||||
|     dependencies: |     dependencies: | ||||||
| @@ -15831,6 +15843,8 @@ snapshots: | |||||||
|       '@ckeditor/ckeditor5-ui': 47.1.0 |       '@ckeditor/ckeditor5-ui': 47.1.0 | ||||||
|       '@ckeditor/ckeditor5-utils': 47.1.0 |       '@ckeditor/ckeditor5-utils': 47.1.0 | ||||||
|       ckeditor5: 47.1.0(patch_hash=8331a09d41443b39ea1c784daaccfeb0da4f9065ed556e7de92e9c77edd9eb41) |       ckeditor5: 47.1.0(patch_hash=8331a09d41443b39ea1c784daaccfeb0da4f9065ed556e7de92e9c77edd9eb41) | ||||||
|  |     transitivePeerDependencies: | ||||||
|  |       - supports-color | ||||||
|  |  | ||||||
|   '@ckeditor/ckeditor5-restricted-editing@47.1.0': |   '@ckeditor/ckeditor5-restricted-editing@47.1.0': | ||||||
|     dependencies: |     dependencies: | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user