mirror of
https://github.com/zadam/trilium.git
synced 2025-10-30 01:36:24 +01:00
Compare commits
17 Commits
v0.46.3-be
...
v0.46.5
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
fdce218e88 | ||
|
|
6c8d20288d | ||
|
|
88d04772c4 | ||
|
|
9fd26a9b9f | ||
|
|
98f02c3c9a | ||
|
|
584fea1992 | ||
|
|
af50a1ec52 | ||
|
|
4e76d1fa85 | ||
|
|
03a11e6f77 | ||
|
|
e1a16b4a9f | ||
|
|
12b468d3dc | ||
|
|
6f901e6852 | ||
|
|
09e9ac4d00 | ||
|
|
a33ac65fdf | ||
|
|
f8fb071a6f | ||
|
|
a654078e56 | ||
|
|
fba68681aa |
2
package-lock.json
generated
2
package-lock.json
generated
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "trilium",
|
"name": "trilium",
|
||||||
"version": "0.46.2-beta",
|
"version": "0.46.4-beta",
|
||||||
"lockfileVersion": 1,
|
"lockfileVersion": 1,
|
||||||
"requires": true,
|
"requires": true,
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
"name": "trilium",
|
"name": "trilium",
|
||||||
"productName": "Trilium Notes",
|
"productName": "Trilium Notes",
|
||||||
"description": "Trilium Notes",
|
"description": "Trilium Notes",
|
||||||
"version": "0.46.3-beta",
|
"version": "0.46.5",
|
||||||
"license": "AGPL-3.0-only",
|
"license": "AGPL-3.0-only",
|
||||||
"main": "electron.js",
|
"main": "electron.js",
|
||||||
"bin": {
|
"bin": {
|
||||||
|
|||||||
@@ -28,6 +28,16 @@ const TPL = `
|
|||||||
</select>
|
</select>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<div class="form-group row">
|
||||||
|
<div class="col-4">
|
||||||
|
<label for="heading-style">Heading style</label>
|
||||||
|
<select class="form-control" id="heading-style">
|
||||||
|
<option value="plain">Plain</option>
|
||||||
|
<option value="markdown">Markdown-style</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<p>Zooming can be controlled with CTRL+- and CTRL+= shortcuts as well.</p>
|
<p>Zooming can be controlled with CTRL+- and CTRL+= shortcuts as well.</p>
|
||||||
|
|
||||||
@@ -78,6 +88,7 @@ export default class ApperanceOptions {
|
|||||||
this.$themeSelect = $("#theme-select");
|
this.$themeSelect = $("#theme-select");
|
||||||
this.$zoomFactorSelect = $("#zoom-factor-select");
|
this.$zoomFactorSelect = $("#zoom-factor-select");
|
||||||
this.$nativeTitleBarSelect = $("#native-title-bar-select");
|
this.$nativeTitleBarSelect = $("#native-title-bar-select");
|
||||||
|
this.$headingStyle = $("#heading-style");
|
||||||
this.$mainFontSize = $("#main-font-size");
|
this.$mainFontSize = $("#main-font-size");
|
||||||
this.$treeFontSize = $("#tree-font-size");
|
this.$treeFontSize = $("#tree-font-size");
|
||||||
this.$detailFontSize = $("#detail-font-size");
|
this.$detailFontSize = $("#detail-font-size");
|
||||||
@@ -86,11 +97,7 @@ export default class ApperanceOptions {
|
|||||||
this.$themeSelect.on('change', () => {
|
this.$themeSelect.on('change', () => {
|
||||||
const newTheme = this.$themeSelect.val();
|
const newTheme = this.$themeSelect.val();
|
||||||
|
|
||||||
for (const clazz of Array.from(this.$body[0].classList)) { // create copy to safely iterate over while removing classes
|
this.toggleBodyClass("theme-", newTheme);
|
||||||
if (clazz.startsWith("theme-")) {
|
|
||||||
this.$body.removeClass(clazz);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const noteId = $(this).find(":selected").attr("data-note-id");
|
const noteId = $(this).find(":selected").attr("data-note-id");
|
||||||
|
|
||||||
@@ -100,8 +107,6 @@ export default class ApperanceOptions {
|
|||||||
libraryLoader.requireCss(`api/notes/download/${noteId}`);
|
libraryLoader.requireCss(`api/notes/download/${noteId}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.$body.addClass("theme-" + newTheme);
|
|
||||||
|
|
||||||
server.put('options/theme/' + newTheme);
|
server.put('options/theme/' + newTheme);
|
||||||
});
|
});
|
||||||
|
|
||||||
@@ -113,6 +118,14 @@ export default class ApperanceOptions {
|
|||||||
server.put('options/nativeTitleBarVisible/' + nativeTitleBarVisible);
|
server.put('options/nativeTitleBarVisible/' + nativeTitleBarVisible);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
this.$headingStyle.on('change', () => {
|
||||||
|
const newHeadingStyle = this.$headingStyle.val();
|
||||||
|
|
||||||
|
this.toggleBodyClass("heading-style-", newHeadingStyle);
|
||||||
|
|
||||||
|
server.put('options/headingStyle/' + newHeadingStyle);
|
||||||
|
});
|
||||||
|
|
||||||
this.$mainFontSize.on('change', async () => {
|
this.$mainFontSize.on('change', async () => {
|
||||||
await server.put('options/mainFontSize/' + this.$mainFontSize.val());
|
await server.put('options/mainFontSize/' + this.$mainFontSize.val());
|
||||||
|
|
||||||
@@ -132,6 +145,16 @@ export default class ApperanceOptions {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
toggleBodyClass(prefix, value) {
|
||||||
|
for (const clazz of Array.from(this.$body[0].classList)) { // create copy to safely iterate over while removing classes
|
||||||
|
if (clazz.startsWith(prefix)) {
|
||||||
|
this.$body.removeClass(clazz);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.$body.addClass(prefix + value);
|
||||||
|
}
|
||||||
|
|
||||||
async optionsLoaded(options) {
|
async optionsLoaded(options) {
|
||||||
const themes = [
|
const themes = [
|
||||||
{ val: 'white', title: 'White' },
|
{ val: 'white', title: 'White' },
|
||||||
@@ -159,6 +182,8 @@ export default class ApperanceOptions {
|
|||||||
|
|
||||||
this.$nativeTitleBarSelect.val(options.nativeTitleBarVisible === 'true' ? 'show' : 'hide');
|
this.$nativeTitleBarSelect.val(options.nativeTitleBarVisible === 'true' ? 'show' : 'hide');
|
||||||
|
|
||||||
|
this.$headingStyle.val(options.headingStyle);
|
||||||
|
|
||||||
this.$mainFontSize.val(options.mainFontSize);
|
this.$mainFontSize.val(options.mainFontSize);
|
||||||
this.$treeFontSize.val(options.treeFontSize);
|
this.$treeFontSize.val(options.treeFontSize);
|
||||||
this.$detailFontSize.val(options.detailFontSize);
|
this.$detailFontSize.val(options.detailFontSize);
|
||||||
|
|||||||
@@ -254,22 +254,39 @@ class NoteShort {
|
|||||||
return noteAttributeCache.attributes[this.noteId];
|
return noteAttributeCache.attributes[this.noteId];
|
||||||
}
|
}
|
||||||
|
|
||||||
getAllNotePaths() {
|
getAllNotePaths(encounteredNoteIds = null) {
|
||||||
if (this.noteId === 'root') {
|
if (this.noteId === 'root') {
|
||||||
return [['root']];
|
return [['root']];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (!encounteredNoteIds) {
|
||||||
|
encounteredNoteIds = new Set();
|
||||||
|
}
|
||||||
|
|
||||||
|
encounteredNoteIds.add(this.noteId);
|
||||||
|
|
||||||
const parentNotes = this.getParentNotes();
|
const parentNotes = this.getParentNotes();
|
||||||
let paths;
|
let paths;
|
||||||
|
|
||||||
if (parentNotes.length === 1) { // optimization for the most common case
|
if (parentNotes.length === 1) { // optimization for the most common case
|
||||||
paths = parentNotes[0].getAllNotePaths();
|
if (encounteredNoteIds.has(parentNotes[0].noteId)) {
|
||||||
|
return [];
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
paths = parentNotes[0].getAllNotePaths(encounteredNoteIds);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
paths = [];
|
paths = [];
|
||||||
|
|
||||||
for (const parentNote of parentNotes) {
|
for (const parentNote of parentNotes) {
|
||||||
paths.push(...parentNote.getAllNotePaths());
|
if (encounteredNoteIds.has(parentNote.noteId)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const newSet = new Set(encounteredNoteIds);
|
||||||
|
|
||||||
|
paths.push(...parentNote.getAllNotePaths(newSet));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -79,6 +79,15 @@ async function renderAttributes(attributes, renderIsInheritable) {
|
|||||||
return $container;
|
return $container;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
const HIDDEN_ATTRIBUTES = [
|
||||||
|
'originalFileName',
|
||||||
|
'template',
|
||||||
|
'cssClass',
|
||||||
|
'iconClass',
|
||||||
|
'pageSize',
|
||||||
|
'viewType'
|
||||||
|
];
|
||||||
|
|
||||||
async function renderNormalAttributes(note) {
|
async function renderNormalAttributes(note) {
|
||||||
const promotedDefinitionAttributes = note.getPromotedDefinitionAttributes();
|
const promotedDefinitionAttributes = note.getPromotedDefinitionAttributes();
|
||||||
let attrs = note.getAttributes();
|
let attrs = note.getAttributes();
|
||||||
@@ -90,6 +99,7 @@ async function renderNormalAttributes(note) {
|
|||||||
attrs = attrs.filter(
|
attrs = attrs.filter(
|
||||||
attr => !attr.isDefinition()
|
attr => !attr.isDefinition()
|
||||||
&& !attr.isAutoLink
|
&& !attr.isAutoLink
|
||||||
|
&& !HIDDEN_ATTRIBUTES.includes(attr.name)
|
||||||
&& attr.noteId === note.noteId
|
&& attr.noteId === note.noteId
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -74,7 +74,11 @@ export default class Entrypoints extends Component {
|
|||||||
|
|
||||||
await ws.waitForMaxKnownEntityChangeId();
|
await ws.waitForMaxKnownEntityChangeId();
|
||||||
|
|
||||||
await appContext.tabManager.openTabWithNote(note.noteId, true);
|
const hoistedNoteId = appContext.tabManager.getActiveTabContext()
|
||||||
|
? appContext.tabManager.getActiveTabContext().hoistedNoteId
|
||||||
|
: 'root';
|
||||||
|
|
||||||
|
await appContext.tabManager.openTabWithNote(note.noteId, true, null, hoistedNoteId);
|
||||||
|
|
||||||
appContext.triggerEvent('focusAndSelectTitle');
|
appContext.triggerEvent('focusAndSelectTitle');
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,9 +26,7 @@ function isHoistedNode(node) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async function checkNoteAccess(notePath, tabContext) {
|
async function checkNoteAccess(notePath, tabContext) {
|
||||||
// notePath argument can contain only noteId which is not good when hoisted since
|
const resolvedNotePath = await treeService.resolveNotePath(notePath, tabContext.hoistedNoteId);
|
||||||
// then we need to check the whole note path
|
|
||||||
const resolvedNotePath = await treeService.resolveNotePath(notePath);
|
|
||||||
|
|
||||||
if (!resolvedNotePath) {
|
if (!resolvedNotePath) {
|
||||||
console.log("Cannot activate " + notePath);
|
console.log("Cannot activate " + notePath);
|
||||||
@@ -37,7 +35,7 @@ async function checkNoteAccess(notePath, tabContext) {
|
|||||||
|
|
||||||
const hoistedNoteId = tabContext.hoistedNoteId;
|
const hoistedNoteId = tabContext.hoistedNoteId;
|
||||||
|
|
||||||
if (hoistedNoteId !== 'root' && !resolvedNotePath.includes(hoistedNoteId)) {
|
if (!resolvedNotePath.includes(hoistedNoteId)) {
|
||||||
const confirmDialog = await import('../dialogs/confirm.js');
|
const confirmDialog = await import('../dialogs/confirm.js');
|
||||||
|
|
||||||
if (!await confirmDialog.confirm("Requested note is outside of hoisted note subtree and you must unhoist to access the note. Do you want to proceed with unhoisting?")) {
|
if (!await confirmDialog.confirm("Requested note is outside of hoisted note subtree and you must unhoist to access the note. Do you want to proceed with unhoisting?")) {
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ async function resolveNotePathToSegments(notePath, hoistedNoteId = 'root', logEr
|
|||||||
path.push('root');
|
path.push('root');
|
||||||
}
|
}
|
||||||
|
|
||||||
const effectivePath = [];
|
const effectivePathSegments = [];
|
||||||
let childNoteId = null;
|
let childNoteId = null;
|
||||||
let i = 0;
|
let i = 0;
|
||||||
|
|
||||||
@@ -81,7 +81,7 @@ async function resolveNotePathToSegments(notePath, hoistedNoteId = 'root', logEr
|
|||||||
const pathToRoot = someNotePath.split("/").reverse().slice(1);
|
const pathToRoot = someNotePath.split("/").reverse().slice(1);
|
||||||
|
|
||||||
for (const noteId of pathToRoot) {
|
for (const noteId of pathToRoot) {
|
||||||
effectivePath.push(noteId);
|
effectivePathSegments.push(noteId);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -89,11 +89,23 @@ async function resolveNotePathToSegments(notePath, hoistedNoteId = 'root', logEr
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
effectivePath.push(parentNoteId);
|
effectivePathSegments.push(parentNoteId);
|
||||||
childNoteId = parentNoteId;
|
childNoteId = parentNoteId;
|
||||||
}
|
}
|
||||||
|
|
||||||
return effectivePath.reverse();
|
effectivePathSegments.reverse();
|
||||||
|
|
||||||
|
if (effectivePathSegments.includes(hoistedNoteId)) {
|
||||||
|
return effectivePathSegments;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
const note = await treeCache.getNote(getNoteIdFromNotePath(notePath));
|
||||||
|
|
||||||
|
const someNotePathSegments = getSomeNotePathSegments(note, hoistedNoteId);
|
||||||
|
|
||||||
|
// if there isn't actually any note path with hoisted note then return the original resolved note path
|
||||||
|
return someNotePathSegments.includes(hoistedNoteId) ? someNotePathSegments : effectivePathSegments;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function getSomeNotePathSegments(note, hoistedNotePath = 'root') {
|
function getSomeNotePathSegments(note, hoistedNotePath = 'root') {
|
||||||
|
|||||||
@@ -194,9 +194,18 @@ const ATTR_HELP = {
|
|||||||
"appTheme": "marks CSS notes which are full Trilium themes and are thus available in Trilium options.",
|
"appTheme": "marks CSS notes which are full Trilium themes and are thus available in Trilium options.",
|
||||||
"cssClass": "value of this label is then added as CSS class to the node representing given note in the tree. This can be useful for advanced theming. Can be used in template notes.",
|
"cssClass": "value of this label is then added as CSS class to the node representing given note in the tree. This can be useful for advanced theming. Can be used in template notes.",
|
||||||
"iconClass": "value of this label is added as a CSS class to the icon on the tree which can help visually distinguish the notes in the tree. Example might be bx bx-home - icons are taken from boxicons. Can be used in template notes.",
|
"iconClass": "value of this label is added as a CSS class to the icon on the tree which can help visually distinguish the notes in the tree. Example might be bx bx-home - icons are taken from boxicons. Can be used in template notes.",
|
||||||
"bookZoomLevel": 'applies only to book note and sets the "zoom level" (how many notes fit on 1 row)',
|
"pageSize": "number of items per page in note listing",
|
||||||
"customRequestHandler": 'see <a href="javascript:" data-help-page="Custom request handler">Custom request handler</a>',
|
"customRequestHandler": 'see <a href="javascript:" data-help-page="Custom request handler">Custom request handler</a>',
|
||||||
"customResourceProvider": 'see <a href="javascript:" data-help-page="Custom request handler">Custom request handler</a>'
|
"customResourceProvider": 'see <a href="javascript:" data-help-page="Custom request handler">Custom request handler</a>',
|
||||||
|
"widget": "marks this note as a custom widget which will be added to the Trilium component tree",
|
||||||
|
"workspace": "marks this note as a workspace which allows easy hoisting",
|
||||||
|
"workspaceIconClass": "defines box icon CSS class which will be used in tab when hoisted to this note",
|
||||||
|
"workspaceTabBackgroundColor": "CSS color used in the note tab when hoisted to this note",
|
||||||
|
"searchHome": "new search notes will be created as children of this note",
|
||||||
|
"hoistedSearchHome": "new search notes will be created as children of this note when hoisted to some ancestor of this note",
|
||||||
|
"inbox": "default inbox location for new notes",
|
||||||
|
"hoistedInbox": "default inbox location for new notes when hoisted to some ancestor of this note",
|
||||||
|
"sqlConsoleHome": "default location of SQL console notes",
|
||||||
},
|
},
|
||||||
"relation": {
|
"relation": {
|
||||||
"runOnNoteCreation": "executes when note is created on backend",
|
"runOnNoteCreation": "executes when note is created on backend",
|
||||||
|
|||||||
@@ -278,7 +278,7 @@ export default class NoteDetailWidget extends TabAwareWidget {
|
|||||||
|
|
||||||
const label = attrs.find(attr =>
|
const label = attrs.find(attr =>
|
||||||
attr.type === 'label'
|
attr.type === 'label'
|
||||||
&& ['readOnly', 'autoReadOnlyDisabled', 'cssClass', 'bookZoomLevel', 'displayRelations'].includes(attr.name)
|
&& ['readOnly', 'autoReadOnlyDisabled', 'cssClass', 'displayRelations'].includes(attr.name)
|
||||||
&& attr.isAffecting(this.note));
|
&& attr.isAffecting(this.note));
|
||||||
|
|
||||||
const relation = attrs.find(attr =>
|
const relation = attrs.find(attr =>
|
||||||
|
|||||||
@@ -88,7 +88,7 @@ export default class NotePathsWidget extends TabAwareWidget {
|
|||||||
.find('a')
|
.find('a')
|
||||||
.addClass("no-tooltip-preview");
|
.addClass("no-tooltip-preview");
|
||||||
|
|
||||||
const comments = [];
|
const icons = [];
|
||||||
|
|
||||||
if (this.notePath === notePath) {
|
if (this.notePath === notePath) {
|
||||||
$noteLink.addClass("path-current");
|
$noteLink.addClass("path-current");
|
||||||
@@ -98,23 +98,23 @@ export default class NotePathsWidget extends TabAwareWidget {
|
|||||||
$noteLink.addClass("path-in-hoisted-subtree");
|
$noteLink.addClass("path-in-hoisted-subtree");
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
comments.push("outside of hoisting");
|
icons.push(`<span class="bx bx-trending-up" title="This path is outside of hoisted note and you would have to unhoist."></span>`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (notePathRecord.isArchived) {
|
if (notePathRecord.isArchived) {
|
||||||
$noteLink.addClass("path-archived");
|
$noteLink.addClass("path-archived");
|
||||||
|
|
||||||
comments.push("archived");
|
icons.push(`<span class="bx bx-archive" title="Archived"></span>`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (notePathRecord.isSearch) {
|
if (notePathRecord.isSearch) {
|
||||||
$noteLink.addClass("path-search");
|
$noteLink.addClass("path-search");
|
||||||
|
|
||||||
comments.push("search");
|
icons.push(`<span class="bx bx-search" title="Search"></span>`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (comments.length > 0) {
|
if (icons.length > 0) {
|
||||||
$noteLink.append(` (${comments.join(', ')})`);
|
$noteLink.append(` ${icons.join(' ')}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
this.$notePathList.append($noteLink);
|
this.$notePathList.append($noteLink);
|
||||||
|
|||||||
@@ -994,8 +994,6 @@ export default class NoteTreeWidget extends TabAwareWidget {
|
|||||||
const nextNode = activeNode ? (activeNode.getNextSibling() || activeNode.getPrevSibling() || activeNode.getParent()) : null;
|
const nextNode = activeNode ? (activeNode.getNextSibling() || activeNode.getPrevSibling() || activeNode.getParent()) : null;
|
||||||
const activeNotePath = activeNode ? treeService.getNotePath(activeNode) : null;
|
const activeNotePath = activeNode ? treeService.getNotePath(activeNode) : null;
|
||||||
|
|
||||||
console.log(activeNotePath, activeNodeFocused);
|
|
||||||
|
|
||||||
const nextNotePath = nextNode ? treeService.getNotePath(nextNode) : null;
|
const nextNotePath = nextNode ? treeService.getNotePath(nextNode) : null;
|
||||||
const activeNoteId = activeNode ? activeNode.data.noteId : null;
|
const activeNoteId = activeNode ? activeNode.data.noteId : null;
|
||||||
|
|
||||||
@@ -1117,6 +1115,10 @@ export default class NoteTreeWidget extends TabAwareWidget {
|
|||||||
|
|
||||||
if (node) {
|
if (node) {
|
||||||
node.setActive(true, {noEvents: true, noFocus: !activeNodeFocused});
|
node.setActive(true, {noEvents: true, noFocus: !activeNodeFocused});
|
||||||
|
|
||||||
|
if (activeNodeFocused) {
|
||||||
|
node.setFocus(true);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// this is used when original note has been deleted and we want to move the focus to the note above/below
|
// this is used when original note has been deleted and we want to move the focus to the note above/below
|
||||||
|
|||||||
@@ -49,15 +49,16 @@ const TPL = `
|
|||||||
}
|
}
|
||||||
|
|
||||||
.note-detail-editable-text h2 { font-size: 1.8em; }
|
.note-detail-editable-text h2 { font-size: 1.8em; }
|
||||||
.note-detail-editable-text h2::before { content: "##\\2004"; color: var(--muted-text-color); }
|
|
||||||
.note-detail-editable-text h3 { font-size: 1.6em; }
|
.note-detail-editable-text h3 { font-size: 1.6em; }
|
||||||
.note-detail-editable-text h3::before { content: "###\\2004"; color: var(--muted-text-color); }
|
|
||||||
.note-detail-editable-text h4 { font-size: 1.4em; }
|
.note-detail-editable-text h4 { font-size: 1.4em; }
|
||||||
.note-detail-editable-text h4:not(.include-note-title)::before { content: "####\\2004"; color: var(--muted-text-color); }
|
|
||||||
.note-detail-editable-text h5 { font-size: 1.2em; }
|
.note-detail-editable-text h5 { font-size: 1.2em; }
|
||||||
.note-detail-editable-text h5::before { content: "#####\\2004"; color: var(--muted-text-color); }
|
|
||||||
.note-detail-editable-text h6 { font-size: 1.1em; }
|
.note-detail-editable-text h6 { font-size: 1.1em; }
|
||||||
.note-detail-editable-text h6::before { content: "######\\2004"; color: var(--muted-text-color); }
|
|
||||||
|
body.heading-style-markdown .note-detail-editable-text h2::before { content: "##\\2004"; color: var(--muted-text-color); }
|
||||||
|
body.heading-style-markdown .note-detail-editable-text h3::before { content: "###\\2004"; color: var(--muted-text-color); }
|
||||||
|
body.heading-style-markdown .note-detail-editable-text h4:not(.include-note-title)::before { content: "####\\2004"; color: var(--muted-text-color); }
|
||||||
|
body.heading-style-markdown .note-detail-editable-text h5::before { content: "#####\\2004"; color: var(--muted-text-color); }
|
||||||
|
body.heading-style-markdown .note-detail-editable-text h6::before { content: "######\\2004"; color: var(--muted-text-color); }
|
||||||
|
|
||||||
.note-detail-editable-text-editor {
|
.note-detail-editable-text-editor {
|
||||||
padding-top: 10px;
|
padding-top: 10px;
|
||||||
|
|||||||
@@ -14,16 +14,11 @@ const TPL = `
|
|||||||
.note-detail-readonly-text h5 { font-size: 1.2em; }
|
.note-detail-readonly-text h5 { font-size: 1.2em; }
|
||||||
.note-detail-readonly-text h6 { font-size: 1.1em; }
|
.note-detail-readonly-text h6 { font-size: 1.1em; }
|
||||||
|
|
||||||
.note-detail-readonly-text h2 { font-size: 1.8em; }
|
body.heading-style-markdown .note-detail-readonly-text h2::before { content: "##\\2004"; color: var(--muted-text-color); }
|
||||||
.note-detail-readonly-text h2::before { content: "##\\2004"; color: var(--muted-text-color); }
|
body.heading-style-markdown .note-detail-readonly-text h3::before { content: "###\\2004"; color: var(--muted-text-color); }
|
||||||
.note-detail-readonly-text h3 { font-size: 1.6em; }
|
body.heading-style-markdown .note-detail-readonly-text h4:not(.include-note-title)::before { content: "####\\2004"; color: var(--muted-text-color); }
|
||||||
.note-detail-readonly-text h3::before { content: "###\\2004"; color: var(--muted-text-color); }
|
body.heading-style-markdown .note-detail-readonly-text h5::before { content: "#####\\2004"; color: var(--muted-text-color); }
|
||||||
.note-detail-readonly-text h4 { font-size: 1.4em; }
|
body.heading-style-markdown .note-detail-readonly-text h6::before { content: "######\\2004"; color: var(--muted-text-color); }
|
||||||
.note-detail-readonly-text h4:not(.include-note-title)::before { content: "####\\2004"; color: var(--muted-text-color); }
|
|
||||||
.note-detail-readonly-text h5 { font-size: 1.2em; }
|
|
||||||
.note-detail-readonly-text h5::before { content: "#####\\2004"; color: var(--muted-text-color); }
|
|
||||||
.note-detail-readonly-text h6 { font-size: 1.1em; }
|
|
||||||
.note-detail-readonly-text h6::before { content: "######\\2004"; color: var(--muted-text-color); }
|
|
||||||
|
|
||||||
.note-detail-readonly-text {
|
.note-detail-readonly-text {
|
||||||
padding-left: 22px;
|
padding-left: 22px;
|
||||||
|
|||||||
@@ -106,7 +106,7 @@ function processContent(images, note, content) {
|
|||||||
for (const {src, dataUrl, imageId} of images) {
|
for (const {src, dataUrl, imageId} of images) {
|
||||||
const filename = path.basename(src);
|
const filename = path.basename(src);
|
||||||
|
|
||||||
if (!dataUrl.startsWith("data:image")) {
|
if (!dataUrl || !dataUrl.startsWith("data:image")) {
|
||||||
log.info("Image could not be recognized as data URL:", dataUrl.substr(0, Math.min(100, dataUrl.length)));
|
log.info("Image could not be recognized as data URL:", dataUrl.substr(0, Math.min(100, dataUrl.length)));
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -9,8 +9,27 @@ const cls = require('../../services/cls');
|
|||||||
const repository = require('../../services/repository');
|
const repository = require('../../services/repository');
|
||||||
|
|
||||||
function getInboxNote(req) {
|
function getInboxNote(req) {
|
||||||
return attributeService.getNoteWithLabel('inbox')
|
const hoistedNote = getHoistedNote();
|
||||||
|| dateNoteService.getDateNote(req.params.date);
|
|
||||||
|
let inbox;
|
||||||
|
|
||||||
|
if (hoistedNote) {
|
||||||
|
([inbox] = hoistedNote.getDescendantNotesWithLabel('hoistedInbox'));
|
||||||
|
|
||||||
|
if (!inbox) {
|
||||||
|
([inbox] = hoistedNote.getDescendantNotesWithLabel('inbox'));
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!inbox) {
|
||||||
|
inbox = hoistedNote;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
inbox = attributeService.getNoteWithLabel('inbox')
|
||||||
|
|| dateNoteService.getDateNote(req.params.date);
|
||||||
|
}
|
||||||
|
|
||||||
|
return inbox;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getDateNote(req) {
|
function getDateNote(req) {
|
||||||
@@ -66,27 +85,18 @@ function createSearchNote(req) {
|
|||||||
const searchString = params.searchString || "";
|
const searchString = params.searchString || "";
|
||||||
let ancestorNoteId = params.ancestorNoteId;
|
let ancestorNoteId = params.ancestorNoteId;
|
||||||
|
|
||||||
const hoistedNote = cls.getHoistedNoteId() && cls.getHoistedNoteId() !== 'root'
|
const hoistedNote = getHoistedNote();
|
||||||
? repository.getNote(cls.getHoistedNoteId())
|
|
||||||
: null;
|
|
||||||
|
|
||||||
let searchHome;
|
let searchHome;
|
||||||
|
|
||||||
if (hoistedNote) {
|
if (hoistedNote) {
|
||||||
([searchHome] = hoistedNote.getDescendantNotesWithLabel('hoistedSearchHome'));
|
([searchHome] = hoistedNote.getDescendantNotesWithLabel('hoistedSearchHome'));
|
||||||
}
|
|
||||||
|
|
||||||
if (!searchHome) {
|
if (!searchHome) {
|
||||||
const today = dateUtils.localNowDate();
|
([searchHome] = hoistedNote.getDescendantNotesWithLabel('searchHome'));
|
||||||
|
}
|
||||||
|
|
||||||
searchHome = attributeService.getNoteWithLabel('searchHome')
|
if (!searchHome) {
|
||||||
|| dateNoteService.getDateNote(today);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (hoistedNote) {
|
|
||||||
|
|
||||||
if (!hoistedNote.getDescendantNoteIds().includes(searchHome.noteId)) {
|
|
||||||
// otherwise the note would be saved outside of the hoisted context which is weird
|
|
||||||
searchHome = hoistedNote;
|
searchHome = hoistedNote;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -94,6 +104,12 @@ function createSearchNote(req) {
|
|||||||
ancestorNoteId = hoistedNote.noteId;
|
ancestorNoteId = hoistedNote.noteId;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
else {
|
||||||
|
const today = dateUtils.localNowDate();
|
||||||
|
|
||||||
|
searchHome = attributeService.getNoteWithLabel('searchHome')
|
||||||
|
|| dateNoteService.getDateNote(today);
|
||||||
|
}
|
||||||
|
|
||||||
const {note} = noteService.createNewNote({
|
const {note} = noteService.createNewNote({
|
||||||
parentNoteId: searchHome.noteId,
|
parentNoteId: searchHome.noteId,
|
||||||
@@ -112,6 +128,12 @@ function createSearchNote(req) {
|
|||||||
return note;
|
return note;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function getHoistedNote() {
|
||||||
|
return cls.getHoistedNoteId() && cls.getHoistedNoteId() !== 'root'
|
||||||
|
? repository.getNote(cls.getHoistedNoteId())
|
||||||
|
: null;
|
||||||
|
}
|
||||||
|
|
||||||
module.exports = {
|
module.exports = {
|
||||||
getInboxNote,
|
getInboxNote,
|
||||||
getDateNote,
|
getDateNote,
|
||||||
|
|||||||
@@ -40,7 +40,8 @@ const ALLOWED_OPTIONS = new Set([
|
|||||||
'nativeTitleBarVisible',
|
'nativeTitleBarVisible',
|
||||||
'attributeListExpanded',
|
'attributeListExpanded',
|
||||||
'promotedAttributesExpanded',
|
'promotedAttributesExpanded',
|
||||||
'similarNotesExpanded'
|
'similarNotesExpanded',
|
||||||
|
'headingStyle'
|
||||||
]);
|
]);
|
||||||
|
|
||||||
function getOptions() {
|
function getOptions() {
|
||||||
|
|||||||
@@ -19,6 +19,7 @@ function index(req, res) {
|
|||||||
res.render(view, {
|
res.render(view, {
|
||||||
csrfToken: csrfToken,
|
csrfToken: csrfToken,
|
||||||
theme: options.theme,
|
theme: options.theme,
|
||||||
|
headingStyle: options.headingStyle,
|
||||||
mainFontSize: parseInt(options.mainFontSize),
|
mainFontSize: parseInt(options.mainFontSize),
|
||||||
treeFontSize: parseInt(options.treeFontSize),
|
treeFontSize: parseInt(options.treeFontSize),
|
||||||
detailFontSize: parseInt(options.detailFontSize),
|
detailFontSize: parseInt(options.detailFontSize),
|
||||||
|
|||||||
@@ -27,7 +27,6 @@ const BUILTIN_ATTRIBUTES = [
|
|||||||
{ type: 'label', name: 'run', isDangerous: true },
|
{ type: 'label', name: 'run', isDangerous: true },
|
||||||
{ type: 'label', name: 'customRequestHandler', isDangerous: true },
|
{ type: 'label', name: 'customRequestHandler', isDangerous: true },
|
||||||
{ type: 'label', name: 'customResourceProvider', isDangerous: true },
|
{ type: 'label', name: 'customResourceProvider', isDangerous: true },
|
||||||
{ type: 'label', name: 'bookZoomLevel', isDangerous: false },
|
|
||||||
{ type: 'label', name: 'widget', isDangerous: true },
|
{ type: 'label', name: 'widget', isDangerous: true },
|
||||||
{ type: 'label', name: 'noteInfoWidgetDisabled' },
|
{ type: 'label', name: 'noteInfoWidgetDisabled' },
|
||||||
{ type: 'label', name: 'linkMapWidgetDisabled' },
|
{ type: 'label', name: 'linkMapWidgetDisabled' },
|
||||||
@@ -38,9 +37,12 @@ const BUILTIN_ATTRIBUTES = [
|
|||||||
{ type: 'label', name: 'workspaceIconClass' },
|
{ type: 'label', name: 'workspaceIconClass' },
|
||||||
{ type: 'label', name: 'workspaceTabBackgroundColor' },
|
{ type: 'label', name: 'workspaceTabBackgroundColor' },
|
||||||
{ type: 'label', name: 'searchHome' },
|
{ type: 'label', name: 'searchHome' },
|
||||||
|
{ type: 'label', name: 'hoistedInbox' },
|
||||||
{ type: 'label', name: 'hoistedSearchHome' },
|
{ type: 'label', name: 'hoistedSearchHome' },
|
||||||
{ type: 'label', name: 'sqlConsoleHome' },
|
{ type: 'label', name: 'sqlConsoleHome' },
|
||||||
{ type: 'label', name: 'datePattern' },
|
{ type: 'label', name: 'datePattern' },
|
||||||
|
{ type: 'label', name: 'pageSize' },
|
||||||
|
{ type: 'label', name: 'viewType' },
|
||||||
|
|
||||||
// relation names
|
// relation names
|
||||||
{ type: 'relation', name: 'runOnNoteCreation', isDangerous: true },
|
{ type: 'relation', name: 'runOnNoteCreation', isDangerous: true },
|
||||||
|
|||||||
@@ -1 +1 @@
|
|||||||
module.exports = { buildDate:"2021-03-08T23:11:11+01:00", buildRevision: "f27370d44f08afaa22d4cd86cba489584f9c878b" };
|
module.exports = { buildDate:"2021-03-14T22:56:27+01:00", buildRevision: "6c8d20288df302f3a415bd1bdcace98bf29d4bf6" };
|
||||||
|
|||||||
@@ -44,10 +44,14 @@ function isEntityEventsDisabled() {
|
|||||||
return !!namespace.get('disableEntityEvents');
|
return !!namespace.get('disableEntityEvents');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function clearEntityChanges() {
|
||||||
|
namespace.set('entityChanges', []);
|
||||||
|
}
|
||||||
|
|
||||||
function getAndClearEntityChanges() {
|
function getAndClearEntityChanges() {
|
||||||
const entityChanges = namespace.get('entityChanges') || [];
|
const entityChanges = namespace.get('entityChanges') || [];
|
||||||
|
|
||||||
namespace.set('entityChanges', []);
|
clearEntityChanges();
|
||||||
|
|
||||||
return entityChanges;
|
return entityChanges;
|
||||||
}
|
}
|
||||||
@@ -92,6 +96,7 @@ module.exports = {
|
|||||||
disableEntityEvents,
|
disableEntityEvents,
|
||||||
isEntityEventsDisabled,
|
isEntityEventsDisabled,
|
||||||
reset,
|
reset,
|
||||||
|
clearEntityChanges,
|
||||||
getAndClearEntityChanges,
|
getAndClearEntityChanges,
|
||||||
addEntityChange,
|
addEntityChange,
|
||||||
getEntityFromCache,
|
getEntityFromCache,
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ eventService.subscribe(eventService.ENTITY_CREATED, ({ entityName, entity }) =>
|
|||||||
|
|
||||||
const content = note.getContent();
|
const content = note.getContent();
|
||||||
|
|
||||||
if (!["text", "code"].includes(note.type)
|
if (["text", "code"].includes(note.type)
|
||||||
// if the note has already content we're not going to overwrite it with template's one
|
// if the note has already content we're not going to overwrite it with template's one
|
||||||
&& (!content || content.trim().length === 0)
|
&& (!content || content.trim().length === 0)
|
||||||
&& templateNote.isStringNote()) {
|
&& templateNote.isStringNote()) {
|
||||||
|
|||||||
@@ -23,7 +23,6 @@ const IGNORED_ATTR_NAMES = [
|
|||||||
"archived",
|
"archived",
|
||||||
"hidepromotedattributes",
|
"hidepromotedattributes",
|
||||||
"keyboardshortcut",
|
"keyboardshortcut",
|
||||||
"bookzoomlevel",
|
|
||||||
"noteinfowidgetdisabled",
|
"noteinfowidgetdisabled",
|
||||||
"linkmapwidgetdisabled",
|
"linkmapwidgetdisabled",
|
||||||
"noterevisionswidgetdisabled",
|
"noterevisionswidgetdisabled",
|
||||||
|
|||||||
@@ -84,7 +84,8 @@ const defaultOptions = [
|
|||||||
{ name: 'attributeListExpanded', value: 'false', isSynced: false },
|
{ name: 'attributeListExpanded', value: 'false', isSynced: false },
|
||||||
{ name: 'promotedAttributesExpanded', value: 'true', isSynced: true },
|
{ name: 'promotedAttributesExpanded', value: 'true', isSynced: true },
|
||||||
{ name: 'similarNotesExpanded', value: 'true', isSynced: true },
|
{ name: 'similarNotesExpanded', value: 'true', isSynced: true },
|
||||||
{ name: 'debugModeEnabled', value: 'false', isSynced: false }
|
{ name: 'debugModeEnabled', value: 'false', isSynced: false },
|
||||||
|
{ name: 'headingStyle', value: 'markdown', isSynced: true },
|
||||||
];
|
];
|
||||||
|
|
||||||
function initStartupOptions() {
|
function initStartupOptions() {
|
||||||
|
|||||||
@@ -84,7 +84,15 @@ function exec(opts) {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
request.end(opts.body);
|
let payload;
|
||||||
|
|
||||||
|
if (opts.body) {
|
||||||
|
payload = typeof opts.body === 'object'
|
||||||
|
? JSON.stringify(opts.body)
|
||||||
|
: opts.body;
|
||||||
|
}
|
||||||
|
|
||||||
|
request.end(payload);
|
||||||
}
|
}
|
||||||
catch (e) {
|
catch (e) {
|
||||||
reject(generateError(opts, e.message));
|
reject(generateError(opts, e.message));
|
||||||
|
|||||||
@@ -3,6 +3,7 @@
|
|||||||
const log = require('./log');
|
const log = require('./log');
|
||||||
const Database = require('better-sqlite3');
|
const Database = require('better-sqlite3');
|
||||||
const dataDir = require('./data_dir');
|
const dataDir = require('./data_dir');
|
||||||
|
const cls = require('./cls');
|
||||||
|
|
||||||
const dbConnection = new Database(dataDir.DOCUMENT_PATH);
|
const dbConnection = new Database(dataDir.DOCUMENT_PATH);
|
||||||
dbConnection.pragma('journal_mode = WAL');
|
dbConnection.pragma('journal_mode = WAL');
|
||||||
@@ -229,13 +230,20 @@ function wrap(query, func) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
function transactional(func) {
|
function transactional(func) {
|
||||||
const ret = dbConnection.transaction(func).deferred();
|
try {
|
||||||
|
const ret = dbConnection.transaction(func).deferred();
|
||||||
|
|
||||||
if (!dbConnection.inTransaction) { // i.e. transaction was really committed (and not just savepoint released)
|
if (!dbConnection.inTransaction) { // i.e. transaction was really committed (and not just savepoint released)
|
||||||
require('./ws.js').sendTransactionSyncsToAllClients();
|
require('./ws.js').sendTransactionSyncsToAllClients();
|
||||||
|
}
|
||||||
|
|
||||||
|
return ret;
|
||||||
}
|
}
|
||||||
|
catch (e) {
|
||||||
|
cls.clearEntityChanges();
|
||||||
|
|
||||||
return ret;
|
throw e;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function fillNoteIdList(noteIds, truncate = true) {
|
function fillNoteIdList(noteIds, truncate = true) {
|
||||||
|
|||||||
@@ -106,8 +106,6 @@ function sendPing(client, entityChanges = []) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const stats = require('./sync').stats;
|
|
||||||
|
|
||||||
sendMessage(client, {
|
sendMessage(client, {
|
||||||
type: 'sync',
|
type: 'sync',
|
||||||
data: entityChanges
|
data: entityChanges
|
||||||
@@ -118,9 +116,7 @@ function sendTransactionSyncsToAllClients() {
|
|||||||
if (webSocketServer) {
|
if (webSocketServer) {
|
||||||
const entityChanges = cls.getAndClearEntityChanges();
|
const entityChanges = cls.getAndClearEntityChanges();
|
||||||
|
|
||||||
webSocketServer.clients.forEach(function each(client) {
|
webSocketServer.clients.forEach(client => sendPing(client, entityChanges));
|
||||||
sendPing(client, entityChanges);
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@
|
|||||||
<link rel="shortcut icon" href="favicon.ico">
|
<link rel="shortcut icon" href="favicon.ico">
|
||||||
<title>Trilium Notes</title>
|
<title>Trilium Notes</title>
|
||||||
</head>
|
</head>
|
||||||
<body class="desktop theme-<%= theme %>" style="--main-font-size: <%= mainFontSize %>%; --tree-font-size: <%= treeFontSize %>%; --detail-font-size: <%= detailFontSize %>%;">
|
<body class="desktop theme-<%= theme %> heading-style-<%= headingStyle %>" style="--main-font-size: <%= mainFontSize %>%; --tree-font-size: <%= treeFontSize %>%; --detail-font-size: <%= detailFontSize %>%;">
|
||||||
<noscript>Trilium requires JavaScript to be enabled.</noscript>
|
<noscript>Trilium requires JavaScript to be enabled.</noscript>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
|||||||
@@ -95,7 +95,7 @@
|
|||||||
}
|
}
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body class="mobile theme-<%= theme %>">
|
<body class="mobile theme-<%= theme %> heading-style-<%= headingStyle %>">
|
||||||
<noscript>Trilium requires JavaScript to be enabled.</noscript>
|
<noscript>Trilium requires JavaScript to be enabled.</noscript>
|
||||||
|
|
||||||
<div id="toast-container" class="d-flex flex-column justify-content-center align-items-center"></div>
|
<div id="toast-container" class="d-flex flex-column justify-content-center align-items-center"></div>
|
||||||
|
|||||||
Reference in New Issue
Block a user