Notes/src/public/app/dialogs/note_revisions.js

242 lines
8.5 KiB
JavaScript
Raw Normal View History

import utils from '../services/utils.js';
import server from '../services/server.js';
2019-11-09 15:21:14 +01:00
import toastService from "../services/toast.js";
2020-01-20 22:35:52 +01:00
import appContext from "../services/app_context.js";
import libraryLoader from "../services/library_loader.js";
import openService from "../services/open.js";
const $dialog = $("#note-revisions-dialog");
const $list = $("#note-revision-list");
const $listDropdown = $("#note-revision-list-dropdown");
const $content = $("#note-revision-content");
const $title = $("#note-revision-title");
2019-11-09 08:53:13 +01:00
const $titleButtons = $("#note-revision-title-buttons");
2019-11-09 15:21:14 +01:00
const $eraseAllRevisionsButton = $("#note-revisions-erase-all-revisions-button");
$listDropdown.dropdown();
$listDropdown.parent().on('hide.bs.dropdown', e => {
// prevent closing dropdown by clicking outside
if (e.clickEvent) {
e.preventDefault();
}
});
let revisionItems = [];
let note;
let noteRevisionId;
2019-09-02 19:56:52 +02:00
export async function showCurrentNoteRevisions() {
2021-05-22 12:35:41 +02:00
await showNoteRevisionsDialog(appContext.tabManager.getActiveContextNoteId());
}
export async function showNoteRevisionsDialog(noteId, noteRevisionId) {
utils.openDialog($dialog);
2019-11-09 15:21:14 +01:00
await loadNoteRevisions(noteId, noteRevisionId);
}
async function loadNoteRevisions(noteId, noteRevId) {
$list.empty();
$content.empty();
$titleButtons.empty();
2021-05-22 12:35:41 +02:00
note = appContext.tabManager.getActiveContextNote();
2019-11-01 19:21:48 +01:00
revisionItems = await server.get(`notes/${noteId}/revisions`);
for (const item of revisionItems) {
$list.append(
$('<a class="dropdown-item" tabindex="0">')
.text(item.dateLastEdited.substr(0, 16) + ` (${item.contentLength} bytes)`)
.attr('data-note-revision-id', item.noteRevisionId)
.attr('title', 'This revision was last edited on ' + item.dateLastEdited)
);
}
$listDropdown.dropdown('show');
noteRevisionId = noteRevId;
if (revisionItems.length > 0) {
if (!noteRevisionId) {
noteRevisionId = revisionItems[0].noteRevisionId;
}
2019-11-09 15:21:14 +01:00
} else {
$title.text("No revisions for this note yet...");
noteRevisionId = null;
}
$eraseAllRevisionsButton.toggle(revisionItems.length > 0);
}
$dialog.on('shown.bs.modal', () => {
$list.find(`[data-note-revision-id="${noteRevisionId}"]`)
.trigger('focus');
});
async function setContentPane() {
const noteRevisionId = $list.find(".active").attr('data-note-revision-id');
const revisionItem = revisionItems.find(r => r.noteRevisionId === noteRevisionId);
2019-11-09 15:21:14 +01:00
$titleButtons.empty();
$content.empty();
$title.html(revisionItem.title);
2020-05-07 23:34:13 +02:00
const $restoreRevisionButton = $('<button class="btn btn-sm" type="button">Restore this revision</button>');
$restoreRevisionButton.on('click', async () => {
const confirmDialog = await import('../dialogs/confirm.js');
const text = 'Do you want to restore this revision? This will overwrite current title/content of the note with this revision.';
if (await confirmDialog.confirm(text)) {
await server.put(`notes/${revisionItem.noteId}/restore-revision/${revisionItem.noteRevisionId}`);
$dialog.modal('hide');
toastService.showMessage('Note revision has been restored.');
}
});
2019-11-09 16:51:51 +01:00
const $eraseRevisionButton = $('<button class="btn btn-sm" type="button">Delete this revision</button>');
2019-11-09 15:21:14 +01:00
$eraseRevisionButton.on('click', async () => {
const confirmDialog = await import('../dialogs/confirm.js');
2019-11-09 16:51:51 +01:00
const text = 'Do you want to delete this revision? This action will delete revision title and content, but still preserve revision metadata.';
2019-11-09 15:21:14 +01:00
if (await confirmDialog.confirm(text)) {
await server.remove(`notes/${revisionItem.noteId}/revisions/${revisionItem.noteRevisionId}`);
loadNoteRevisions(revisionItem.noteId);
2019-11-09 16:51:51 +01:00
toastService.showMessage('Note revision has been deleted.');
2019-11-09 15:21:14 +01:00
}
});
$titleButtons
2020-05-07 23:34:13 +02:00
.append($restoreRevisionButton)
.append(' &nbsp; ')
2019-11-09 15:21:14 +01:00
.append($eraseRevisionButton)
.append(' &nbsp; ');
2019-11-09 08:53:13 +01:00
const $downloadButton = $('<button class="btn btn-sm btn-primary" type="button">Download</button>');
$downloadButton.on('click', () => openService.downloadNoteRevision(revisionItem.noteId, revisionItem.noteRevisionId));
2019-11-09 08:53:13 +01:00
2019-11-09 15:21:14 +01:00
$titleButtons.append($downloadButton);
2019-11-09 08:53:13 +01:00
2019-11-01 19:21:48 +01:00
const fullNoteRevision = await server.get(`notes/${revisionItem.noteId}/revisions/${revisionItem.noteRevisionId}`);
2019-11-09 13:01:05 +01:00
if (revisionItem.type === 'text') {
2019-11-01 19:21:48 +01:00
$content.html(fullNoteRevision.content);
if ($content.find('span.math-tex').length > 0) {
await libraryLoader.requireLibrary(libraryLoader.KATEX);
renderMathInElement($content[0], {trust: true});
}
}
2019-11-09 13:01:05 +01:00
else if (revisionItem.type === 'code') {
2019-11-01 19:21:48 +01:00
$content.html($("<pre>").text(fullNoteRevision.content));
}
2019-11-09 13:01:05 +01:00
else if (revisionItem.type === 'image') {
2019-11-08 23:09:57 +01:00
$content.html($("<img>")
2019-11-09 08:53:13 +01:00
// reason why we put this inline as base64 is that we do not want to let user to copy this
// as a URL to be used in a note. Instead if they copy and paste it into a note, it will be a uploaded as a new note
2019-11-08 23:09:57 +01:00
.attr("src", `data:${note.mime};base64,` + fullNoteRevision.content)
.css("max-width", "100%")
.css("max-height", "100%"));
2019-11-08 23:09:57 +01:00
}
2019-11-09 13:01:05 +01:00
else if (revisionItem.type === 'file') {
const $table = $("<table cellpadding='10'>")
.append($("<tr>").append(
$("<th>").text("MIME: "),
$("<td>").text(revisionItem.mime)
))
.append($("<tr>").append(
$("<th>").text("File size:"),
$("<td>").text(revisionItem.contentLength + " bytes")
));
if (fullNoteRevision.content) {
$table.append($("<tr>").append(
$('<td colspan="2">').append(
$('<div style="font-weight: bold;">').text("Preview:"),
2019-11-09 13:01:05 +01:00
$('<pre class="file-preview-content"></pre>')
.text(fullNoteRevision.content)
)
));
}
$content.html($table);
2019-11-09 11:58:52 +01:00
}
2022-04-09 15:51:37 +02:00
else if (revisionItem.type === 'canvas-note') {
/**
* can the revisions called without being on the note type before?
* if so: load excalidraw
*/
2022-04-10 13:55:39 +02:00
// FIXME: Does it make sense to use EXCALIDRAW_UTILS that are 1.5 MB
// whereas excalidraw (650kB) +react(12kB)+reactdom(118kB)
/**
* FIXME: We load a font called Virgil.wof2, which originates from excalidraw.com
* REMOVE external dependency!!!! This is defined in the svg in defs.style
*/
2022-04-09 15:51:37 +02:00
await libraryLoader.requireLibrary(libraryLoader.EXCALIDRAW_UTILS);
const {exportToSvg} = window.ExcalidrawUtils
const content = fullNoteRevision.content;
try {
const data = JSON.parse(content)
const excData = {
type: "excalidraw",
version: 2,
source: "trilium",
elements: data.elements,
appState: data.appState,
files: data.files,
}
const svg = await exportToSvg(excData);
2022-04-10 13:55:39 +02:00
console.log("canvas-note revision", data, svg);
$content
.html(
$('<div>')
.html(svg)
);
2022-04-09 15:51:37 +02:00
} catch(err) {
console.error("error parsing fullNoteRevision.content as JSON", fullNoteRevision.content, err);
2022-04-10 13:55:39 +02:00
$content.html($("<div>").text("Error parsing content. Please check console.error() for more details."));
2022-04-10 16:03:39 +02:00
}
2022-04-09 15:51:37 +02:00
}
else {
$content.text("Preview isn't available for this note type.");
}
}
2019-11-09 15:21:14 +01:00
$eraseAllRevisionsButton.on('click', async () => {
const confirmDialog = await import('../dialogs/confirm.js');
2019-11-09 16:51:51 +01:00
const text = 'Do you want to delete all revisions of this note? This action will erase revision title and content, but still preserve revision metadata.';
2019-11-09 15:21:14 +01:00
if (await confirmDialog.confirm(text)) {
await server.remove(`notes/${note.noteId}/revisions`);
$dialog.modal('hide');
2019-11-09 16:51:51 +01:00
toastService.showMessage('Note revisions has been deleted.');
2019-11-09 15:21:14 +01:00
}
});
$list.on('click', '.dropdown-item', e => {
e.preventDefault();
return false;
});
$list.on('focus', '.dropdown-item', e => {
$list.find('.dropdown-item').each((i, el) => {
$(el).toggleClass('active', el === e.target);
});
setContentPane();
});