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

150 lines
4.3 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 utils from "./utils.js";
import optionsService from "./options.js";
2020-01-12 19:30:30 +08:00
import appContext from "./app_context.js";
2020-01-16 04:36:01 +08:00
import treeUtils from "./tree_utils.js";
import noteDetailService from "./note_detail.js";
import Component from "../widgets/component.js";
2019-05-02 05:06:18 +08:00
let showSidebarInNewTab = true;
optionsService.addLoadListener(options => {
showSidebarInNewTab = options.is('showSidebarInNewTab');
});
2020-01-16 04:36:01 +08:00
class TabContext extends Component {
2019-05-12 01:44:58 +08:00
/**
2020-01-16 04:36:01 +08:00
* @param {AppContext} appContext
2020-01-13 02:05:09 +08:00
* @param {TabRowWidget} tabRow
* @param {object} state
2019-05-12 01:44:58 +08:00
*/
2020-01-16 04:36:01 +08:00
constructor(appContext, tabRow, state = {}) {
super(appContext);
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.state = state;
2019-09-05 04:13:22 +08:00
2020-01-16 04:36:01 +08:00
this.attributes = new Attributes(this.appContext, this);
this.children.push(this.attributes);
2019-05-03 04:24:43 +08:00
}
2020-01-16 04:36:01 +08:00
async setNote(notePath) {
this.notePath = notePath;
2020-01-16 04:36:01 +08:00
const noteId = treeUtils.getNoteIdFromNotePath(notePath);
/** @property {NoteFull} */
this.note = await noteDetailService.loadNote(noteId);
this.tabRow.updateTab(this.$tab[0], {title: this.note.title});
this.setupClasses();
2020-01-19 01:01:16 +08:00
//this.cleanup(); // 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.note.noteId,
notePath: this.notePath
});
2019-05-15 04:29:47 +08:00
}
}, 5000);
2019-09-05 04:13:22 +08:00
bundleService.executeRelationBundles(this.note, 'runOnNoteView', this);
2020-01-19 01:01:16 +08:00
this.trigger('tabNoteSwitched', {tabId: this.tabId});
}
async show() {
}
2019-05-15 04:29:47 +08:00
hide() {
2020-01-17 05:44:36 +08:00
// FIXME
2019-05-15 04:29:47 +08:00
}
isActive() {
2020-01-16 05:27:52 +08:00
return this.tabId === this.tabRow.activeTabId;
}
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);
}
}
this.$tab.addClass(this.note.cssClass);
this.$tab.addClass(utils.getNoteTypeClass(this.note.type));
this.$tab.addClass(utils.getMimeTypeClass(this.note.mime));
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() {
2020-01-14 04:48:44 +08:00
return; // FIXME
2019-05-02 04:19:29 +08:00
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();
}
2020-01-13 02:05:09 +08:00
// FIXME trigger "noteSaved" event so that title indicator is triggered
this.eventReceived('noteSaved');
2019-05-02 04:19:29 +08:00
// run async
bundleService.executeRelationBundles(this.note, 'runOnNoteChange', this);
2019-05-02 04:19:29 +08:00
}
async saveNoteIfChanged() {
if (this.isNoteChanged) {
await this.saveNote();
2019-05-21 04:25:04 +08:00
2020-01-12 19:30:30 +08:00
appContext.refreshTabs(this.tabId, this.note.noteId);
2019-05-02 04:19:29 +08:00
}
}
2019-08-15 16:04:03 +08:00
getTabState() {
if (!this.notePath) {
return null;
}
return {
tabId: this.tabId,
notePath: this.notePath,
2020-01-16 05:27:52 +08:00
active: this.tabRow.activeTabId === this.tabId
2019-08-15 16:04:03 +08:00
}
}
stateChanged() {
appContext.openTabsChanged();
2019-08-15 16:04:03 +08:00
}
2019-05-02 04:19:29 +08:00
}
2019-05-09 01:55:24 +08:00
export default TabContext;