trilium/src/public/javascripts/services/tab_context.js

465 lines
14 KiB
JavaScript
Raw Normal View History

2019-05-02 05:06:18 +08:00
import treeService from "./tree.js";
import protectedSessionHolder from "./protected_session_holder.js";
import server from "./server.js";
import bundleService from "./bundle.js";
import Attributes from "./attributes.js";
2019-05-02 05:06:18 +08:00
import treeUtils from "./tree_utils.js";
import utils from "./utils.js";
import NoteTypeContext from "./note_type.js";
import noteDetailService from "./note_detail.js";
import protectedSessionService from "./protected_session.js";
import optionsService from "./options.js";
2019-05-20 03:22:35 +08:00
import linkService from "./link.js";
2019-07-21 16:17:08 +08:00
import Sidebar from "./sidebar.js";
2019-05-02 05:06:18 +08:00
2019-05-09 01:55:24 +08:00
const $tabContentsContainer = $("#note-tab-container");
2019-05-03 04:24:43 +08:00
2019-05-02 04:19:29 +08:00
const componentClasses = {
2019-08-27 01:49:19 +08:00
'empty': "./note_detail_empty.js",
'text': "./note_detail_text.js",
'code': "./note_detail_code.js",
'file': "./note_detail_file.js",
'image': "./note_detail_image.js",
'search': "./note_detail_search.js",
'render': "./note_detail_render.js",
'relation-map': "./note_detail_relation_map.js",
'protected-session': "./note_detail_protected_session.js"
2019-05-02 04:19:29 +08:00
};
let showSidebarInNewTab = true;
optionsService.addLoadListener(options => {
showSidebarInNewTab = options.is('showSidebarInNewTab');
});
2019-05-09 01:55:24 +08:00
class TabContext {
2019-05-12 01:44:58 +08:00
/**
* @param {TabRow} tabRow
* @param {object} state
2019-05-12 01:44:58 +08:00
*/
constructor(tabRow, state = {}) {
2019-05-12 01:44:58 +08:00
this.tabRow = tabRow;
this.tabId = state.tabId || utils.randomString(4);
2019-05-15 04:29:47 +08:00
this.$tab = $(this.tabRow.addTab(this.tabId));
this.initialized = false;
this.state = state;
}
2019-09-05 04:13:22 +08:00
async initTabContent() {
if (this.initialized) {
return;
}
2019-05-05 16:59:34 +08:00
2019-09-05 04:13:22 +08:00
this.initialized = true;
2019-05-09 01:55:24 +08:00
this.$tabContent = $(".note-tab-content-template").clone();
this.$tabContent.removeClass('note-tab-content-template');
this.$tabContent.attr('data-tab-id', this.tabId);
this.$tabContent.hide();
2019-05-05 16:59:34 +08:00
2019-05-09 01:55:24 +08:00
$tabContentsContainer.append(this.$tabContent);
2019-05-05 16:59:34 +08:00
2019-05-09 01:55:24 +08:00
this.$noteTitle = this.$tabContent.find(".note-title");
2019-05-12 18:58:55 +08:00
this.$noteTitleRow = this.$tabContent.find(".note-title-row");
2019-05-20 03:22:35 +08:00
this.$notePathList = this.$tabContent.find(".note-path-list");
this.$notePathCount = this.$tabContent.find(".note-path-count");
2019-05-09 01:55:24 +08:00
this.$noteDetailComponents = this.$tabContent.find(".note-detail-component");
this.$childrenOverview = this.$tabContent.find(".children-overview");
this.$scriptArea = this.$tabContent.find(".note-detail-script-area");
this.$savedIndicator = this.$tabContent.find(".saved-indicator");
2019-05-04 20:34:03 +08:00
this.noteChangeDisabled = false;
2019-05-02 04:19:29 +08:00
this.isNoteChanged = false;
this.attributes = new Attributes(this);
2019-05-14 05:08:59 +08:00
if (utils.isDesktop()) {
const sidebarState = this.state.sidebar || {
visible: showSidebarInNewTab
};
this.sidebar = new Sidebar(this, sidebarState);
2019-05-14 05:08:59 +08:00
this.noteType = new NoteTypeContext(this);
}
2019-05-02 04:19:29 +08:00
this.components = {};
this.$noteTitle.on('input', () => {
if (!this.note) {
return;
}
2019-05-02 04:19:29 +08:00
this.noteChanged();
this.note.title = this.$noteTitle.val();
this.tabRow.updateTab(this.$tab[0], {title: this.note.title});
treeService.setNoteTitle(this.noteId, this.note.title);
2019-05-02 04:19:29 +08:00
this.setTitleBar();
2019-05-02 04:19:29 +08:00
});
2019-05-03 04:24:43 +08:00
2019-05-23 02:53:59 +08:00
if (utils.isDesktop()) {
// keyboard plugin is not loaded in mobile
utils.bindElShortcut(this.$noteTitle, 'return', () => {
2019-07-04 02:29:55 +08:00
this.getComponent().focus();
return false; // to not propagate the enter into the editor (causes issues with codemirror)
});
2019-05-23 02:53:59 +08:00
}
2019-05-20 03:22:35 +08:00
2019-05-09 01:55:24 +08:00
this.$protectButton = this.$tabContent.find(".protect-button");
this.$protectButton.click(protectedSessionService.protectNoteAndSendToServer);
2019-05-04 03:50:14 +08:00
2019-05-09 01:55:24 +08:00
this.$unprotectButton = this.$tabContent.find(".unprotect-button");
this.$unprotectButton.click(protectedSessionService.unprotectNoteAndSendToServer);
2019-09-05 04:13:22 +08:00
const type = this.getComponentType();
if (!(type in this.components)) {
const clazz = await import(componentClasses[type]);
this.components[type] = new clazz.default(this);
}
2019-05-03 04:24:43 +08:00
}
2019-08-27 01:49:19 +08:00
async setNote(note, notePath) {
2019-05-03 04:24:43 +08:00
this.noteId = note.noteId;
2019-05-09 02:14:41 +08:00
this.notePath = notePath;
2019-08-17 17:28:36 +08:00
/** @property {NoteFull} */
2019-05-03 04:24:43 +08:00
this.note = note;
this.tabRow.updateTab(this.$tab[0], {title: note.title});
if (!this.initialized) {
return;
}
2019-05-05 16:59:34 +08:00
// after loading new note make sure editor is scrolled to the top
this.getComponent().scrollToTop();
this.setupClasses();
2019-05-15 04:29:47 +08:00
this.setCurrentNotePathToHash();
this.noteChangeDisabled = true;
try {
this.$noteTitle.val(this.note.title);
await this.renderComponent();
} finally {
this.noteChangeDisabled = false;
}
this.setTitleBar();
this.closeAutocomplete(); // esp. on windows autocomplete is not getting closed automatically
2019-05-15 04:29:47 +08:00
setTimeout(async () => {
// we include the note into recent list only if the user stayed on the note at least 5 seconds
if (notePath && notePath === this.notePath) {
await server.post('recent-notes', {
noteId: this.noteId,
notePath: this.notePath
});
2019-05-15 04:29:47 +08:00
}
}, 5000);
2019-05-20 03:22:35 +08:00
this.showPaths();
2019-09-05 04:13:22 +08:00
if (utils.isDesktop()) {
this.attributes.refreshAttributes();
this.noteType.update();
this.showChildrenOverview();
} else {
// mobile usually doesn't need attributes so we just invalidate
this.attributes.invalidateAttributes();
}
2019-07-21 17:32:38 +08:00
if (this.sidebar) {
2019-08-17 17:28:36 +08:00
this.sidebar.noteLoaded(); // load async
2019-07-21 17:32:38 +08:00
}
2019-09-05 04:13:22 +08:00
bundleService.executeRelationBundles(this.note, 'runOnNoteView', this);
}
async show() {
if (!this.initialized) {
2019-09-05 04:13:22 +08:00
await this.initTabContent();
2019-09-05 04:13:22 +08:00
this.$tabContent.show(); // show immediately so that user can see something
if (this.note) {
await this.setNote(this.note, this.notePath);
}
}
2019-05-15 04:29:47 +08:00
this.$tabContent.show();
2019-09-05 04:13:22 +08:00
2019-05-15 04:29:47 +08:00
this.setCurrentNotePathToHash();
this.setTitleBar();
}
async renderComponent() {
for (const componentType in this.components) {
2019-09-05 04:13:22 +08:00
if (componentType !== this.getComponentType()) {
this.components[componentType].cleanup();
}
}
this.$noteDetailComponents.hide();
this.$noteTitle.show(); // this can be hidden by empty detail
this.$noteTitle.removeAttr("readonly"); // this can be set by protected session service
2019-09-05 04:13:22 +08:00
this.getComponent().show();
await this.getComponent().render();
}
setTitleBar() {
if (!this.$tabContent.is(":visible")) {
return;
}
2019-05-15 04:29:47 +08:00
document.title = "Trilium Notes";
if (this.note) {
// it helps navigating in history if note title is included in the title
document.title += " - " + this.note.title;
}
}
hide() {
if (this.initialized) {
this.$tabContent.hide();
}
2019-05-15 04:29:47 +08:00
}
setCurrentNotePathToHash() {
if (this.$tab[0] === this.tabRow.activeTabEl) {
document.location.hash = (this.notePath || "") + "-" + this.tabId;
}
}
setupClasses() {
for (const clazz of Array.from(this.$tab[0].classList)) { // create copy to safely iterate over while removing classes
if (clazz !== 'note-tab') {
this.$tab.removeClass(clazz);
}
}
2019-05-09 01:55:24 +08:00
for (const clazz of Array.from(this.$tabContent[0].classList)) { // create copy to safely iterate over while removing classes
if (clazz !== 'note-tab-content') {
2019-05-09 01:55:24 +08:00
this.$tabContent.removeClass(clazz);
}
}
this.$tab.addClass(this.note.cssClass);
this.$tab.addClass(utils.getNoteTypeClass(this.note.type));
this.$tab.addClass(utils.getMimeTypeClass(this.note.mime));
this.$tabContent.addClass(this.note.cssClass);
2019-05-09 01:55:24 +08:00
this.$tabContent.addClass(utils.getNoteTypeClass(this.note.type));
this.$tabContent.addClass(utils.getMimeTypeClass(this.note.mime));
this.$noteTitleRow.show(); // might be hidden from empty detail
this.$tabContent.toggleClass("protected", this.note.isProtected);
this.$protectButton.toggleClass("active", this.note.isProtected);
this.$protectButton.prop("disabled", this.note.isProtected);
this.$unprotectButton.toggleClass("active", !this.note.isProtected);
this.$unprotectButton.prop("disabled", !this.note.isProtected || !protectedSessionHolder.isProtectedSessionAvailable());
2019-05-02 04:19:29 +08:00
}
getComponent() {
2019-08-27 01:49:19 +08:00
const type = this.getComponentType();
return this.components[type];
}
getComponentType() {
2019-05-12 18:58:55 +08:00
let type;
2019-05-12 18:58:55 +08:00
if (this.note) {
type = this.note.type;
2019-05-12 18:58:55 +08:00
if (this.note.isProtected) {
if (protectedSessionHolder.isProtectedSessionAvailable()) {
protectedSessionHolder.touchProtectedSession();
} else {
type = 'protected-session';
// user shouldn't be able to edit note title
this.$noteTitle.prop("readonly", true);
}
}
2019-08-27 01:49:19 +08:00
} else {
2019-05-12 18:58:55 +08:00
type = 'empty';
}
2019-08-27 01:49:19 +08:00
return type;
2019-05-02 04:19:29 +08:00
}
2019-05-22 02:24:40 +08:00
async activate() {
await this.tabRow.activateTab(this.$tab[0]);
}
2019-05-02 04:19:29 +08:00
async saveNote() {
if (this.note.isProtected && !protectedSessionHolder.isProtectedSessionAvailable()) {
return;
}
this.note.title = this.$noteTitle.val();
2019-05-22 02:24:40 +08:00
this.note.content = this.getComponent().getContent();
2019-05-02 04:19:29 +08:00
// it's important to set the flag back to false immediatelly after retrieving title and content
// otherwise we might overwrite another change (especially async code)
this.isNoteChanged = false;
treeService.setNoteTitle(this.note.noteId, this.note.title);
const resp = await server.put('notes/' + this.note.noteId, this.note.dto);
this.note.dateModified = resp.dateModified;
this.note.utcDateModified = resp.utcDateModified;
2019-05-02 04:19:29 +08:00
if (this.note.isProtected) {
protectedSessionHolder.touchProtectedSession();
}
2019-05-04 03:50:14 +08:00
this.$savedIndicator.fadeIn();
2019-05-02 04:19:29 +08:00
this.$scriptArea.empty();
2019-05-02 04:19:29 +08:00
// run async
bundleService.executeRelationBundles(this.note, 'runOnNoteChange', this);
this.eventReceived('noteSaved');
2019-05-02 04:19:29 +08:00
}
async saveNoteIfChanged() {
if (this.isNoteChanged) {
await this.saveNote();
2019-05-21 04:25:04 +08:00
noteDetailService.refreshTabs(this.tabId, this.noteId);
2019-05-02 04:19:29 +08:00
}
}
noteChanged() {
2019-05-04 20:34:03 +08:00
if (this.noteChangeDisabled) {
2019-05-02 04:19:29 +08:00
return;
}
this.isNoteChanged = true;
2019-05-04 03:50:14 +08:00
this.$savedIndicator.fadeOut();
2019-05-02 04:19:29 +08:00
}
async showChildrenOverview() {
const attributes = await this.attributes.getAttributes();
2019-05-02 04:19:29 +08:00
const hideChildrenOverview = attributes.some(attr => attr.type === 'label' && attr.name === 'hideChildrenOverview')
|| this.note.type === 'relation-map'
|| this.note.type === 'image'
|| this.note.type === 'file';
if (hideChildrenOverview) {
this.$childrenOverview.hide();
return;
}
this.$childrenOverview.empty();
for (const childBranch of await this.note.getChildBranches()) {
const link = $('<a>', {
href: 'javascript:',
text: await treeUtils.getNoteTitle(childBranch.noteId, childBranch.parentNoteId)
2019-05-15 04:29:47 +08:00
}).attr('data-action', 'note').attr('data-note-path', this.notePath + '/' + childBranch.noteId);
2019-05-02 04:19:29 +08:00
const childEl = $('<div class="child-overview-item">').html(link);
this.$childrenOverview.append(childEl);
}
this.$childrenOverview.show();
}
2019-05-20 03:22:35 +08:00
async addPath(notePath, isCurrent) {
const title = await treeUtils.getNotePathTitle(notePath);
const noteLink = await linkService.createNoteLink(notePath, title);
noteLink
.addClass("no-tooltip-preview")
.addClass("dropdown-item");
if (isCurrent) {
noteLink.addClass("current");
}
this.$notePathList.append(noteLink);
}
async showPaths() {
if (this.note.noteId === 'root') {
// root doesn't have any parent, but it's still technically 1 path
this.$notePathCount.html("1 path");
this.$notePathList.empty();
await this.addPath('root', true);
}
else {
const parents = await this.note.getParentNotes();
this.$notePathCount.html(parents.length + " path" + (parents.length > 1 ? "s" : ""));
this.$notePathList.empty();
const pathSegments = this.notePath.split("/");
const activeNoteParentNoteId = pathSegments[pathSegments.length - 2]; // we know this is not root so there must be a parent
for (const parentNote of parents) {
const parentNotePath = await treeService.getSomeNotePath(parentNote);
// this is to avoid having root notes leading '/'
const notePath = parentNotePath ? (parentNotePath + '/' + this.noteId) : this.noteId;
const isCurrent = activeNoteParentNoteId === parentNote.noteId;
await this.addPath(notePath, isCurrent);
}
}
}
closeAutocomplete() {
2019-07-01 02:14:57 +08:00
if (utils.isDesktop()) {
this.$tabContent.find('.aa-input').autocomplete('close');
}
}
eventReceived(name, data) {
if (!this.initialized) {
return;
}
this.attributes.eventReceived(name, data);
if (this.sidebar) {
this.sidebar.eventReceived(name, data);
}
}
2019-08-15 16:04:03 +08:00
getTabState() {
if (!this.notePath) {
return null;
}
return {
tabId: this.tabId,
notePath: this.notePath,
active: this.tabRow.activeTabEl === this.$tab[0],
sidebar: this.sidebar && this.sidebar.getSidebarState()
}
}
stateChanged() {
noteDetailService.openTabsChanged();
}
2019-05-02 04:19:29 +08:00
}
2019-05-09 01:55:24 +08:00
export default TabContext;