mirror of
				https://github.com/zadam/trilium.git
				synced 2025-10-31 10:26:08 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			381 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			381 lines
		
	
	
		
			15 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import { t } from "../../services/i18n.js";
 | |
| import utils from "../../services/utils.js";
 | |
| import server from "../../services/server.js";
 | |
| import toastService from "../../services/toast.js";
 | |
| import appContext from "../../components/app_context.js";
 | |
| import openService from "../../services/open.js";
 | |
| import protectedSessionHolder from "../../services/protected_session_holder.js";
 | |
| import BasicWidget from "../basic_widget.js";
 | |
| import dialogService, { openDialog } from "../../services/dialog.js";
 | |
| import options from "../../services/options.js";
 | |
| import type FNote from "../../entities/fnote.js";
 | |
| import type { NoteType } from "../../entities/fnote.js";
 | |
| import { Dropdown, Modal } from "bootstrap";
 | |
| import { renderMathInElement } from "../../services/math.js";
 | |
| 
 | |
| const TPL = /*html*/`
 | |
| <div class="revisions-dialog modal fade mx-auto" tabindex="-1" role="dialog">
 | |
|     <style>
 | |
|         .revisions-dialog .revision-content-wrapper {
 | |
|             flex-grow: 1;
 | |
|             margin-left: 20px;
 | |
|             display: flex;
 | |
|             flex-direction: column;
 | |
|             min-width: 0;
 | |
|         }
 | |
| 
 | |
|         .revisions-dialog .revision-content {
 | |
|             overflow: auto;
 | |
|             word-break: break-word;
 | |
|         }
 | |
| 
 | |
|         .revisions-dialog .revision-content img {
 | |
|             max-width: 100%;
 | |
|             object-fit: contain;
 | |
|         }
 | |
| 
 | |
|         .revisions-dialog .revision-content pre {
 | |
|             max-width: 100%;
 | |
|             word-break: break-all;
 | |
|             white-space: pre-wrap;
 | |
|         }
 | |
|     </style>
 | |
| 
 | |
|     <div class="modal-dialog modal-xl" role="document">
 | |
|         <div class="modal-content">
 | |
|             <div class="modal-header">
 | |
|                 <h5 class="modal-title flex-grow-1">${t("revisions.note_revisions")}</h5>
 | |
|                 <button class="revisions-erase-all-revisions-button btn btn-sm"
 | |
|                         title="${t("revisions.delete_all_revisions")}"
 | |
|                         style="padding: 0 10px 0 10px;" type="button">${t("revisions.delete_all_button")}</button>
 | |
|                 <button class="help-button" type="button" data-help-page="note-revisions.html" title="${t("revisions.help_title")}">?</button>
 | |
|                 <button type="button" class="btn-close" data-bs-dismiss="modal" aria-label="${t("revisions.close")}"></button>
 | |
|             </div>
 | |
|             <div class="modal-body" style="display: flex; height: 80vh;">
 | |
|                 <div class="dropdown">
 | |
|                     <button class="revision-list-dropdown" type="button" style="display: none;"
 | |
|                             data-bs-toggle="dropdown" data-bs-display="static">
 | |
|                     </button>
 | |
| 
 | |
|                     <div class="revision-list dropdown-menu static" style="position: static; height: 100%; overflow: auto;"></div>
 | |
|                 </div>
 | |
| 
 | |
|                 <div class="revision-content-wrapper">
 | |
|                     <div style="flex-grow: 0; display: flex; justify-content: space-between;">
 | |
|                         <h3 class="revision-title" style="margin: 3px; flex-grow: 100;"></h3>
 | |
| 
 | |
|                         <div class="revision-title-buttons"></div>
 | |
|                     </div>
 | |
| 
 | |
|                     <div class="revision-content use-tn-links"></div>
 | |
|                 </div>
 | |
|             </div>
 | |
|             <div class="modal-footer py-0">
 | |
|                 <span class="revisions-snapshot-interval flex-grow-1 my-0 py-0"></span>
 | |
|                 <span class="maximum-revisions-for-current-note flex-grow-1 my-0 py-0"></span>
 | |
|                 <button class="revision-settings-button icon-action bx bx-cog my-0 py-0" title="${t("revisions.settings")}"></button>
 | |
|             </div>
 | |
|         </div>
 | |
|     </div>
 | |
| </div>`;
 | |
| 
 | |
| interface RevisionItem {
 | |
|     noteId: string;
 | |
|     revisionId: string;
 | |
|     dateLastEdited: string;
 | |
|     contentLength: number;
 | |
|     type: NoteType;
 | |
|     title: string;
 | |
|     isProtected: boolean;
 | |
|     mime: string;
 | |
| }
 | |
| 
 | |
| interface FullRevision {
 | |
|     content: string;
 | |
|     mime: string;
 | |
| }
 | |
| 
 | |
| export default class RevisionsDialog extends BasicWidget {
 | |
| 
 | |
|     private revisionItems: RevisionItem[];
 | |
|     private note: FNote | null;
 | |
|     private revisionId: string | null;
 | |
| 
 | |
|     private modal!: Modal;
 | |
|     private listDropdown!: Dropdown;
 | |
| 
 | |
|     private $list!: JQuery<HTMLElement>;
 | |
|     private $listDropdown!: JQuery<HTMLElement>;
 | |
|     private $content!: JQuery<HTMLElement>;
 | |
|     private $title!: JQuery<HTMLElement>;
 | |
|     private $titleButtons!: JQuery<HTMLElement>;
 | |
|     private $eraseAllRevisionsButton!: JQuery<HTMLElement>;
 | |
|     private $maximumRevisions!: JQuery<HTMLElement>;
 | |
|     private $snapshotInterval!: JQuery<HTMLElement>;
 | |
|     private $revisionSettingsButton!: JQuery<HTMLElement>;
 | |
| 
 | |
|     constructor() {
 | |
|         super();
 | |
| 
 | |
|         this.revisionItems = [];
 | |
|         this.note = null;
 | |
|         this.revisionId = null;
 | |
|     }
 | |
| 
 | |
|     doRender() {
 | |
|         this.$widget = $(TPL);
 | |
| 
 | |
|         this.modal = Modal.getOrCreateInstance(this.$widget[0]);
 | |
| 
 | |
|         this.$list = this.$widget.find(".revision-list");
 | |
|         this.$listDropdown = this.$widget.find(".revision-list-dropdown");
 | |
| 
 | |
|         this.listDropdown = Dropdown.getOrCreateInstance(this.$listDropdown[0], { autoClose: false });
 | |
|         this.$content = this.$widget.find(".revision-content");
 | |
|         this.$title = this.$widget.find(".revision-title");
 | |
|         this.$titleButtons = this.$widget.find(".revision-title-buttons");
 | |
|         this.$eraseAllRevisionsButton = this.$widget.find(".revisions-erase-all-revisions-button");
 | |
|         this.$snapshotInterval = this.$widget.find(".revisions-snapshot-interval");
 | |
|         this.$maximumRevisions = this.$widget.find(".maximum-revisions-for-current-note");
 | |
|         this.$revisionSettingsButton = this.$widget.find(".revision-settings-button");
 | |
|         this.listDropdown.show();
 | |
| 
 | |
|         this.$listDropdown.parent().on("hide.bs.dropdown", (e) => {
 | |
|             this.modal.hide();
 | |
|         });
 | |
| 
 | |
|         this.$widget.on("shown.bs.modal", () => {
 | |
|             this.$list.find(`[data-revision-id="${this.revisionId}"]`).trigger("focus");
 | |
|         });
 | |
| 
 | |
|         this.$eraseAllRevisionsButton.on("click", async () => {
 | |
|             if (!this.note) {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             const text = t("revisions.confirm_delete_all");
 | |
| 
 | |
|             if (await dialogService.confirm(text)) {
 | |
|                 await server.remove(`notes/${this.note.noteId}/revisions`);
 | |
| 
 | |
|                 this.modal.hide();
 | |
| 
 | |
|                 toastService.showMessage(t("revisions.revisions_deleted"));
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         this.$list.on("focus", ".dropdown-item", (e) => {
 | |
|             this.$list.find(".dropdown-item").each((i, el) => {
 | |
|                 $(el).toggleClass("active", el === e.target);
 | |
|             });
 | |
| 
 | |
|             this.setContentPane();
 | |
|         });
 | |
| 
 | |
|         this.$revisionSettingsButton.on("click", async () => {
 | |
|             appContext.tabManager.openContextWithNote("_optionsOther", { activate: true });
 | |
|         });
 | |
|     }
 | |
| 
 | |
|     async showRevisionsEvent({ noteId = appContext.tabManager.getActiveContextNoteId() }) {
 | |
|         if (!noteId) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         openDialog(this.$widget);
 | |
| 
 | |
|         await this.loadRevisions(noteId);
 | |
|     }
 | |
| 
 | |
|     async loadRevisions(noteId: string) {
 | |
|         this.$title.empty();
 | |
|         this.$list.empty();
 | |
|         this.$content.empty();
 | |
|         this.$titleButtons.empty();
 | |
| 
 | |
|         this.note = appContext.tabManager.getActiveContextNote();
 | |
|         this.revisionItems = await server.get<RevisionItem[]>(`notes/${noteId}/revisions`);
 | |
| 
 | |
|         for (const item of this.revisionItems) {
 | |
|             this.$list.append(
 | |
|                 $('<a class="dropdown-item" tabindex="0">')
 | |
|                     .text(`${item.dateLastEdited.substr(0, 16)} (${utils.formatSize(item.contentLength)})`)
 | |
|                     .attr("data-revision-id", item.revisionId)
 | |
|                     .attr("title", t("revisions.revision_last_edited", { date: item.dateLastEdited }))
 | |
|             );
 | |
|         }
 | |
| 
 | |
|         this.listDropdown.show();
 | |
| 
 | |
|         if (this.revisionItems.length > 0) {
 | |
|             if (!this.revisionId) {
 | |
|                 this.revisionId = this.revisionItems[0].revisionId;
 | |
|             }
 | |
|         } else {
 | |
|             this.$title.text(t("revisions.no_revisions"));
 | |
|             this.revisionId = null;
 | |
|         }
 | |
| 
 | |
|         this.$eraseAllRevisionsButton.toggle(this.revisionItems.length > 0);
 | |
| 
 | |
|         // Show the footer of the revisions dialog
 | |
|         this.$snapshotInterval.text(t("revisions.snapshot_interval", { seconds: options.getInt("revisionSnapshotTimeInterval") }));
 | |
|         let revisionsNumberLimit: number | string = parseInt(this.note?.getLabelValue("versioningLimit") ?? "");
 | |
|         if (!Number.isInteger(revisionsNumberLimit)) {
 | |
|             revisionsNumberLimit = options.getInt("revisionSnapshotNumberLimit") ?? 0;
 | |
|         }
 | |
|         if (revisionsNumberLimit === -1) {
 | |
|             revisionsNumberLimit = "∞";
 | |
|         }
 | |
|         this.$maximumRevisions.text(t("revisions.maximum_revisions", { number: revisionsNumberLimit }));
 | |
|     }
 | |
| 
 | |
|     async setContentPane() {
 | |
|         const revisionId = this.$list.find(".active").attr("data-revision-id");
 | |
| 
 | |
|         const revisionItem = this.revisionItems.find((r) => r.revisionId === revisionId);
 | |
|         if (!revisionItem) {
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         this.$title.html(revisionItem.title);
 | |
| 
 | |
|         this.renderContentButtons(revisionItem);
 | |
| 
 | |
|         await this.renderContent(revisionItem);
 | |
|     }
 | |
| 
 | |
|     renderContentButtons(revisionItem: RevisionItem) {
 | |
|         this.$titleButtons.empty();
 | |
| 
 | |
|         const $restoreRevisionButton = $(`
 | |
|             <button class="btn btn-sm" type="button">
 | |
|                 <span class="bx bx-history"></span>
 | |
|                 ${t("revisions.restore_button")}
 | |
|             </button>
 | |
|         `);
 | |
| 
 | |
|         $restoreRevisionButton.on("click", async () => {
 | |
|             const text = t("revisions.confirm_restore");
 | |
| 
 | |
|             if (await dialogService.confirm(text)) {
 | |
|                 await server.post(`revisions/${revisionItem.revisionId}/restore`);
 | |
| 
 | |
|                 this.modal.hide();
 | |
| 
 | |
|                 toastService.showMessage(t("revisions.revision_restored"));
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         const $eraseRevisionButton = $(`
 | |
|             <button class="btn btn-sm" type="button">
 | |
|                 <span class="bx bx-trash"></span>
 | |
|                 ${t("revisions.delete_button")}
 | |
|             </button>
 | |
|         `);
 | |
| 
 | |
|         $eraseRevisionButton.on("click", async () => {
 | |
|             const text = t("revisions.confirm_delete");
 | |
| 
 | |
|             if (await dialogService.confirm(text)) {
 | |
|                 await server.remove(`revisions/${revisionItem.revisionId}`);
 | |
| 
 | |
|                 this.loadRevisions(revisionItem.noteId);
 | |
| 
 | |
|                 toastService.showMessage(t("revisions.revision_deleted"));
 | |
|             }
 | |
|         });
 | |
| 
 | |
|         if (!revisionItem.isProtected || protectedSessionHolder.isProtectedSessionAvailable()) {
 | |
|             this.$titleButtons.append($restoreRevisionButton).append("   ");
 | |
|         }
 | |
| 
 | |
|         this.$titleButtons.append($eraseRevisionButton).append("   ");
 | |
| 
 | |
|         const $downloadButton = $(`
 | |
|             <button class="btn btn-sm btn-primary" type="button">
 | |
|                 <span class="bx bx-download"></span>
 | |
|                 ${t("revisions.download_button")}
 | |
|             </button>
 | |
|         `);
 | |
| 
 | |
|         $downloadButton.on("click", () => openService.downloadRevision(revisionItem.noteId, revisionItem.revisionId));
 | |
| 
 | |
|         if (!revisionItem.isProtected || protectedSessionHolder.isProtectedSessionAvailable()) {
 | |
|             this.$titleButtons.append($downloadButton);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     async renderContent(revisionItem: RevisionItem) {
 | |
|         this.$content.empty();
 | |
| 
 | |
|         const fullRevision = await server.get<FullRevision>(`revisions/${revisionItem.revisionId}`);
 | |
| 
 | |
|         if (revisionItem.type === "text") {
 | |
|             this.$content.html(`<div class="ck-content">${fullRevision.content}</div>`);
 | |
| 
 | |
|             if (this.$content.find("span.math-tex").length > 0) {
 | |
|                 renderMathInElement(this.$content[0], { trust: true });
 | |
|             }
 | |
|         } else if (revisionItem.type === "code") {
 | |
|             this.$content.html($("<pre>")
 | |
|                 .text(fullRevision.content).prop("outerHTML"));
 | |
|         } else if (revisionItem.type === "image") {
 | |
|             if (fullRevision.mime === "image/svg+xml") {
 | |
|                 let encodedSVG = encodeURIComponent(fullRevision.content); //Base64 of other format images may be embedded in svg
 | |
|                 this.$content.html($("<img>")
 | |
|                     .attr("src", `data:${fullRevision.mime};utf8,${encodedSVG}`)
 | |
|                     .css("max-width", "100%")
 | |
|                     .css("max-height", "100%").prop("outerHTML"));
 | |
|             } else {
 | |
|                 this.$content.html(
 | |
|                     $("<img>")
 | |
|                         // the reason why we put this inline as base64 is that we do not want to let user copy this
 | |
|                         // as a URL to be used in a note. Instead, if they copy and paste it into a note, it will be uploaded as a new note
 | |
|                         .attr("src", `data:${fullRevision.mime};base64,${fullRevision.content}`)
 | |
|                         .css("max-width", "100%")
 | |
|                         .css("max-height", "100%")
 | |
|                         .prop("outerHTML")
 | |
|                 );
 | |
|             }
 | |
|         } else if (revisionItem.type === "file") {
 | |
|             const $table = $("<table cellpadding='10'>")
 | |
|                 .append($("<tr>")
 | |
|                     .append(
 | |
|                         $("<th>").text(t("revisions.mime")),
 | |
|                         $("<td>").text(revisionItem.mime)))
 | |
|                     .append($("<tr>").append($("<th>").text(t("revisions.file_size")), $("<td>").text(utils.formatSize(revisionItem.contentLength))));
 | |
| 
 | |
|             if (fullRevision.content) {
 | |
|                 $table.append(
 | |
|                     $("<tr>").append(
 | |
|                         $('<td colspan="2">').append($('<div style="font-weight: bold;">').text(t("revisions.preview")), $('<pre class="file-preview-content"></pre>').text(fullRevision.content))
 | |
|                     )
 | |
|                 );
 | |
|             }
 | |
| 
 | |
|             this.$content.html($table.prop("outerHTML"));
 | |
|         } else if (["canvas", "mindMap"].includes(revisionItem.type)) {
 | |
|             const encodedTitle = encodeURIComponent(revisionItem.title);
 | |
| 
 | |
|             this.$content.html(
 | |
|                 $("<img>")
 | |
|                     .attr("src", `api/revisions/${revisionItem.revisionId}/image/${encodedTitle}?${Math.random()}`)
 | |
|                     .css("max-width", "100%")
 | |
|                     .prop("outerHTML"));
 | |
|         } else if (revisionItem.type === "mermaid") {
 | |
|             const encodedTitle = encodeURIComponent(revisionItem.title);
 | |
| 
 | |
|             this.$content.html(
 | |
|                 $("<img>")
 | |
|                     .attr("src", `api/revisions/${revisionItem.revisionId}/image/${encodedTitle}?${Math.random()}`)
 | |
|                     .css("max-width", "100%")
 | |
|                     .prop("outerHTML"));
 | |
| 
 | |
|             this.$content.append($("<pre>").text(fullRevision.content));
 | |
|         } else {
 | |
|             this.$content.text(t("revisions.preview_not_available"));
 | |
|         }
 | |
|     }
 | |
| }
 |