trilium/src/public/javascripts/note_editor.js

400 lines
12 KiB
JavaScript
Raw Normal View History

"use strict";
2017-11-05 05:54:27 +08:00
const noteEditor = (function() {
2018-02-15 12:31:20 +08:00
const $noteTitle = $("#note-title");
const $noteDetail = $('#note-detail');
const $noteDetailCode = $('#note-detail-code');
2018-03-11 22:49:22 +08:00
const $noteDetailSearch = $('#note-detail-search');
2018-02-15 12:31:20 +08:00
const $noteDetailRender = $('#note-detail-render');
const $noteDetailAttachment = $('#note-detail-attachment');
const $protectButton = $("#protect-button");
const $unprotectButton = $("#unprotect-button");
const $noteDetailWrapper = $("#note-detail-wrapper");
const $noteIdDisplay = $("#note-id-display");
const $attributeList = $("#attribute-list");
const $attributeListInner = $("#attribute-list-inner");
const $attachmentFileName = $("#attachment-filename");
const $attachmentFileType = $("#attachment-filetype");
const $attachmentFileSize = $("#attachment-filesize");
const $attachmentDownload = $("#attachment-download");
const $attachmentOpen = $("#attachment-open");
2018-03-11 22:49:22 +08:00
const $searchString = $("#search-string");
const $executeScriptButton = $("#execute-script-button");
2017-12-02 23:37:12 +08:00
let editor = null;
2018-01-21 10:56:03 +08:00
let codeEditor = null;
2017-11-05 05:54:27 +08:00
let currentNote = null;
let noteChangeDisabled = false;
let isNoteChanged = false;
function getCurrentNote() {
return currentNote;
}
function getCurrentNoteId() {
2018-01-29 08:30:14 +08:00
return currentNote ? currentNote.detail.noteId : null;
2017-11-05 05:54:27 +08:00
}
function noteChanged() {
if (noteChangeDisabled) {
return;
}
isNoteChanged = true;
}
async function reload() {
// no saving here
await loadNoteToEditor(getCurrentNoteId());
}
async function switchToNote(noteId) {
if (getCurrentNoteId() !== noteId) {
await saveNoteIfChanged();
await loadNoteToEditor(noteId);
}
}
2017-11-05 05:54:27 +08:00
async function saveNoteIfChanged() {
if (!isNoteChanged) {
return;
}
const note = noteEditor.getCurrentNote();
updateNoteFromInputs(note);
await saveNoteToServer(note);
2018-01-29 08:30:14 +08:00
if (note.detail.isProtected) {
protected_session.touchProtectedSession();
}
2017-11-05 05:54:27 +08:00
}
function updateNoteFromInputs(note) {
if (note.detail.type === 'text') {
let content = editor.getData();
// if content is only tags/whitespace (typically <p>&nbsp;</p>), then just make it empty
// this is important when setting new note to code
if (jQuery(content).text().trim() === '' && !content.includes("<img")) {
content = '';
}
note.detail.content = content;
}
else if (note.detail.type === 'code') {
2018-01-29 08:30:14 +08:00
note.detail.content = codeEditor.getValue();
}
2018-03-11 22:49:22 +08:00
else if (note.detail.type === 'search') {
note.detail.content = JSON.stringify({
searchString: $searchString.val()
});
}
else if (note.detail.type === 'render' || note.detail.type === 'file') {
2018-01-24 12:41:22 +08:00
// nothing
}
else {
throwError("Unrecognized type: " + note.detail.type);
}
2017-11-05 05:54:27 +08:00
2018-02-15 12:31:20 +08:00
const title = $noteTitle.val();
2017-11-05 05:54:27 +08:00
2018-01-29 08:30:14 +08:00
note.detail.title = title;
2018-03-25 09:39:15 +08:00
treeService.setNoteTitle(note.detail.noteId, title);
2017-11-05 05:54:27 +08:00
}
async function saveNoteToServer(note) {
2018-01-29 08:30:14 +08:00
await server.put('notes/' + note.detail.noteId, note);
2017-11-05 05:54:27 +08:00
isNoteChanged = false;
2017-11-09 11:33:08 +08:00
showMessage("Saved!");
2017-11-05 05:54:27 +08:00
}
function setNoteBackgroundIfProtected(note) {
2018-01-29 08:30:14 +08:00
const isProtected = !!note.detail.isProtected;
2017-11-05 05:54:27 +08:00
2018-02-15 12:31:20 +08:00
$noteDetailWrapper.toggleClass("protected", isProtected);
$protectButton.toggle(!isProtected);
$unprotectButton.toggle(isProtected);
2017-11-05 05:54:27 +08:00
}
let isNewNoteCreated = false;
function newNoteCreated() {
isNewNoteCreated = true;
}
async function setContent(content) {
if (currentNote.detail.type === 'text') {
if (!editor) {
await requireLibrary(CKEDITOR);
editor = await BalloonEditor.create($noteDetail[0], {});
editor.document.on('change', noteChanged);
}
// temporary workaround for https://github.com/ckeditor/ckeditor5-enter/issues/49
editor.setData(content ? content : "<p></p>");
2018-02-15 12:31:20 +08:00
$noteDetail.show();
}
else if (currentNote.detail.type === 'code') {
if (!codeEditor) {
await requireLibrary(CODE_MIRROR);
CodeMirror.keyMap.default["Shift-Tab"] = "indentLess";
CodeMirror.keyMap.default["Tab"] = "indentMore";
CodeMirror.modeURL = 'libraries/codemirror/mode/%N/%N.js';
codeEditor = CodeMirror($("#note-detail-code")[0], {
value: "",
viewportMargin: Infinity,
indentUnit: 4,
matchBrackets: true,
matchTags: { bothTags: true },
highlightSelectionMatches: { showToken: /\w/, annotateScrollbar: false },
lint: true,
gutters: ["CodeMirror-lint-markers"],
lineNumbers: true
});
codeEditor.on('change', noteChanged);
}
2018-02-15 12:31:20 +08:00
$noteDetailCode.show();
// this needs to happen after the element is shown, otherwise the editor won't be refresheds
codeEditor.setValue(content);
const info = CodeMirror.findModeByMIME(currentNote.detail.mime);
if (info) {
codeEditor.setOption("mode", info.mime);
CodeMirror.autoLoadMode(codeEditor, info.mode);
}
codeEditor.refresh();
}
2018-03-11 22:49:22 +08:00
else if (currentNote.detail.type === 'search') {
$noteDetailSearch.show();
try {
const json = JSON.parse(content);
$searchString.val(json.searchString);
}
catch (e) {
console.log(e);
$searchString.val('');
}
$searchString.on('input', noteChanged);
}
}
2017-11-05 05:54:27 +08:00
async function loadNoteToEditor(noteId) {
currentNote = await loadNote(noteId);
2017-11-05 05:54:27 +08:00
if (isNewNoteCreated) {
isNewNoteCreated = false;
2017-11-05 05:54:27 +08:00
2018-02-15 12:31:20 +08:00
$noteTitle.focus().select();
2017-11-05 05:54:27 +08:00
}
2018-02-15 12:31:20 +08:00
$noteIdDisplay.html(noteId);
2018-01-29 08:30:14 +08:00
await protected_session.ensureProtectedSession(currentNote.detail.isProtected, false);
2017-11-05 05:54:27 +08:00
2018-01-29 08:30:14 +08:00
if (currentNote.detail.isProtected) {
protected_session.touchProtectedSession();
}
// this might be important if we focused on protected note when not in protected note and we got a dialog
// to login, but we chose instead to come to another node - at that point the dialog is still visible and this will close it.
protected_session.ensureDialogIsClosed();
2018-02-15 12:31:20 +08:00
$noteDetailWrapper.show();
2017-11-05 05:54:27 +08:00
noteChangeDisabled = true;
2018-02-15 12:31:20 +08:00
$noteTitle.val(currentNote.detail.title);
noteType.setNoteType(currentNote.detail.type);
noteType.setNoteMime(currentNote.detail.mime);
2018-02-15 12:31:20 +08:00
$noteDetail.hide();
2018-03-11 22:49:22 +08:00
$noteDetailSearch.hide();
2018-02-15 12:31:20 +08:00
$noteDetailCode.hide();
$noteDetailRender.html('').hide();
$noteDetailAttachment.hide();
if (currentNote.detail.type === 'render') {
2018-02-15 12:31:20 +08:00
$noteDetailRender.show();
2018-01-24 12:41:22 +08:00
2018-03-07 13:17:18 +08:00
const bundle = await server.get('script/bundle/' + getCurrentNoteId());
2018-01-24 12:41:22 +08:00
2018-03-07 13:17:18 +08:00
$noteDetailRender.html(bundle.html);
executeBundle(bundle);
2018-02-15 12:31:20 +08:00
}
else if (currentNote.detail.type === 'file') {
$noteDetailAttachment.show();
$attachmentFileName.text(currentNote.attributes.original_file_name);
$attachmentFileSize.text(currentNote.attributes.file_size + " bytes");
$attachmentFileType.text(currentNote.detail.mime);
2018-01-24 12:41:22 +08:00
}
2018-01-21 10:56:03 +08:00
else {
await setContent(currentNote.detail.content);
2018-01-21 10:56:03 +08:00
}
2017-11-05 05:54:27 +08:00
noteChangeDisabled = false;
setNoteBackgroundIfProtected(currentNote);
2018-03-25 09:39:15 +08:00
treeService.setBranchBackgroundBasedOnProtectedStatus(noteId);
// after loading new note make sure editor is scrolled to the top
2018-02-15 12:31:20 +08:00
$noteDetailWrapper.scrollTop(0);
2018-02-05 09:23:30 +08:00
loadAttributeList();
}
async function loadAttributeList() {
const noteId = getCurrentNoteId();
const attributes = await server.get('notes/' + noteId + '/attributes');
2018-02-15 12:31:20 +08:00
$attributeListInner.html('');
2018-02-05 09:23:30 +08:00
if (attributes.length > 0) {
for (const attr of attributes) {
2018-02-15 12:31:20 +08:00
$attributeListInner.append(formatAttribute(attr) + " ");
2018-02-05 09:23:30 +08:00
}
2018-02-15 12:31:20 +08:00
$attributeList.show();
2018-02-05 09:23:30 +08:00
}
else {
2018-02-15 12:31:20 +08:00
$attributeList.hide();
2018-02-05 09:23:30 +08:00
}
2017-11-05 05:54:27 +08:00
}
async function loadNote(noteId) {
return await server.get('notes/' + noteId);
2017-11-05 05:54:27 +08:00
}
2017-12-03 02:54:16 +08:00
function getEditor() {
return editor;
}
function focus() {
const note = getCurrentNote();
if (note.detail.type === 'text') {
2018-02-15 12:31:20 +08:00
$noteDetail.focus();
}
else if (note.detail.type === 'code') {
codeEditor.focus();
}
2018-03-11 22:49:22 +08:00
else if (note.detail.type === 'render' || note.detail.type === 'file' || note.detail.type === 'search') {
2018-01-24 12:41:22 +08:00
// do nothing
}
else {
throwError('Unrecognized type: ' + note.detail.type);
}
}
function getCurrentNoteType() {
const currentNote = getCurrentNote();
return currentNote ? currentNote.detail.type : null;
}
async function executeCurrentNote() {
if (getCurrentNoteType() === 'code') {
// make sure note is saved so we load latest changes
await saveNoteIfChanged();
if (currentNote.detail.mime.endsWith("env=frontend")) {
const bundle = await server.get('script/bundle/' + getCurrentNoteId());
executeBundle(bundle);
}
if (currentNote.detail.mime.endsWith("env=backend")) {
await server.post('script/run/' + getCurrentNoteId());
}
showMessage("Note executed");
}
}
2018-02-25 23:55:21 +08:00
$attachmentDownload.click(() => download(getAttachmentUrl()));
$attachmentOpen.click(() => {
if (isElectron()) {
const open = require("open");
open(getAttachmentUrl());
}
else {
window.location.href = getAttachmentUrl();
}
});
function getAttachmentUrl() {
// electron needs absolute URL so we extract current host, port, protocol
2018-02-25 23:55:21 +08:00
return getHost() + "/api/attachments/download/" + getCurrentNoteId()
+ "?protectedSessionId=" + encodeURIComponent(protected_session.getProtectedSessionId());
}
2017-11-05 05:54:27 +08:00
$(document).ready(() => {
2018-02-15 12:31:20 +08:00
$noteTitle.on('input', () => {
noteChanged();
2018-02-15 12:31:20 +08:00
const title = $noteTitle.val();
2018-03-25 09:39:15 +08:00
treeService.setNoteTitle(getCurrentNoteId(), title);
});
2017-11-05 05:54:27 +08:00
// so that tab jumps from note title (which has tabindex 1)
2018-02-15 12:31:20 +08:00
$noteDetail.attr("tabindex", 2);
2017-11-05 05:54:27 +08:00
});
$(document).bind('keydown', "ctrl+return", executeCurrentNote);
$executeScriptButton.click(executeCurrentNote());
2017-11-05 05:54:27 +08:00
setInterval(saveNoteIfChanged, 5000);
return {
reload,
switchToNote,
2017-11-05 05:54:27 +08:00
saveNoteIfChanged,
updateNoteFromInputs,
saveNoteToServer,
setNoteBackgroundIfProtected,
2017-11-05 05:54:27 +08:00
loadNote,
getCurrentNote,
getCurrentNoteType,
2017-11-05 05:54:27 +08:00
getCurrentNoteId,
2017-12-03 02:54:16 +08:00
newNoteCreated,
getEditor,
focus,
2018-02-05 09:23:30 +08:00
executeCurrentNote,
loadAttributeList,
setContent
2017-11-05 05:54:27 +08:00
};
})();