mirror of
				https://github.com/zadam/trilium.git
				synced 2025-11-04 04:16:17 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			328 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			328 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
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 libraryLoader from "../../services/library_loader.js";
 | 
						|
import openService from "../../services/open.js";
 | 
						|
import protectedSessionHolder from "../../services/protected_session_holder.js";
 | 
						|
import BasicWidget from "../basic_widget.js";
 | 
						|
import dialogService from "../../services/dialog.js";
 | 
						|
import options from "../../services/options.js";
 | 
						|
 | 
						|
const TPL = `
 | 
						|
<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="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"></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>`;
 | 
						|
 | 
						|
export default class RevisionsDialog extends BasicWidget {
 | 
						|
    constructor() {
 | 
						|
        super();
 | 
						|
 | 
						|
        this.revisionItems = [];
 | 
						|
        this.note = null;
 | 
						|
        this.revisionId = null;
 | 
						|
    }
 | 
						|
 | 
						|
    doRender() {
 | 
						|
        this.$widget = $(TPL);
 | 
						|
        this.modal = bootstrap.Modal.getOrCreateInstance(this.$widget);
 | 
						|
 | 
						|
        this.$list = this.$widget.find(".revision-list");
 | 
						|
        this.$listDropdown = this.$widget.find(".revision-list-dropdown");
 | 
						|
        this.listDropdown = bootstrap.Dropdown.getOrCreateInstance(this.$listDropdown);
 | 
						|
        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 => {
 | 
						|
            // Prevent closing dropdown by pressing ESC and clicking outside
 | 
						|
            e.preventDefault();
 | 
						|
        });
 | 
						|
 | 
						|
        document.addEventListener('keydown', e => {
 | 
						|
            // Close the revision dialog when revision element is focused and ESC is pressed
 | 
						|
            if (e.key === 'Escape' ||
 | 
						|
                e.target.classList.contains(['dropdown-item', 'active'])) {
 | 
						|
                this.modal.hide();
 | 
						|
            }
 | 
						|
        }, true)
 | 
						|
 | 
						|
        this.$widget.on('shown.bs.modal', () => {
 | 
						|
            this.$list.find(`[data-revision-id="${this.revisionId}"]`)
 | 
						|
                .trigger('focus');
 | 
						|
        });
 | 
						|
 | 
						|
        this.$eraseAllRevisionsButton.on('click', async () => {
 | 
						|
            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() }) {
 | 
						|
        utils.openDialog(this.$widget);
 | 
						|
 | 
						|
        await this.loadRevisions(noteId);
 | 
						|
    }
 | 
						|
 | 
						|
    async loadRevisions(noteId) {
 | 
						|
        this.$list.empty();
 | 
						|
        this.$content.empty();
 | 
						|
        this.$titleButtons.empty();
 | 
						|
 | 
						|
        this.note = appContext.tabManager.getActiveContextNote();
 | 
						|
        this.revisionItems = await server.get(`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 = parseInt(this.note.getLabelValue("versioningLimit") ?? "");
 | 
						|
        if (!Number.isInteger(revisionsNumberLimit)) {
 | 
						|
            revisionsNumberLimit = parseInt(options.getInt('revisionSnapshotNumberLimit'));
 | 
						|
        }
 | 
						|
        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);
 | 
						|
 | 
						|
        this.$title.html(revisionItem.title);
 | 
						|
 | 
						|
        this.renderContentButtons(revisionItem);
 | 
						|
 | 
						|
        await this.renderContent(revisionItem);
 | 
						|
    }
 | 
						|
 | 
						|
    renderContentButtons(revisionItem) {
 | 
						|
        this.$titleButtons.empty();
 | 
						|
 | 
						|
        const $restoreRevisionButton = $(`<button class="btn btn-sm" type="button">${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">${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">${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) {
 | 
						|
        this.$content.empty();
 | 
						|
 | 
						|
        const fullRevision = await server.get(`revisions/${revisionItem.revisionId}`);
 | 
						|
 | 
						|
        if (revisionItem.type === 'text') {
 | 
						|
            this.$content.html(fullRevision.content);
 | 
						|
 | 
						|
            if (this.$content.find('span.math-tex').length > 0) {
 | 
						|
                await libraryLoader.requireLibrary(libraryLoader.KATEX);
 | 
						|
 | 
						|
                renderMathInElement(this.$content[0], { trust: true });
 | 
						|
            }
 | 
						|
        } else if (revisionItem.type === 'code') {
 | 
						|
            this.$content.html($("<pre>").text(fullRevision.content));
 | 
						|
        } 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%"));
 | 
						|
            } 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%"));
 | 
						|
            }
 | 
						|
        } 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);
 | 
						|
        } 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%"));
 | 
						|
        } 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%"));
 | 
						|
 | 
						|
            this.$content.append($("<pre>").text(fullRevision.content));
 | 
						|
        } else {
 | 
						|
            this.$content.text(t("revisions.preview_not_available"));
 | 
						|
        }
 | 
						|
    }
 | 
						|
}
 |