mirror of
https://github.com/zadam/trilium.git
synced 2025-03-01 14:22:32 +01:00
295 lines
11 KiB
JavaScript
295 lines
11 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";
|
|
|
|
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" 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>
|
|
</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.$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.$listDropdown.dropdown();
|
|
|
|
this.$listDropdown.parent().on('hide.bs.dropdown', e => {
|
|
// prevent closing dropdown by clicking outside
|
|
if (e.clickEvent) {
|
|
e.preventDefault();
|
|
}
|
|
});
|
|
|
|
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('click', '.dropdown-item', e => {
|
|
e.preventDefault();
|
|
return false;
|
|
});
|
|
|
|
this.$list.on('focus', '.dropdown-item', e => {
|
|
this.$list.find('.dropdown-item').each((i, el) => {
|
|
$(el).toggleClass('active', el === e.target);
|
|
});
|
|
|
|
this.setContentPane();
|
|
});
|
|
}
|
|
|
|
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.dropdown('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);
|
|
}
|
|
|
|
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') {
|
|
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"));
|
|
}
|
|
}
|
|
}
|