mirror of
https://github.com/zadam/trilium.git
synced 2024-11-15 12:17:01 +08:00
502 lines
No EOL
21 KiB
Text
502 lines
No EOL
21 KiB
Text
<!DOCTYPE html>
|
|
<html lang="en">
|
|
<head>
|
|
<meta charset="utf-8">
|
|
<title>Trilium Notes</title>
|
|
</head>
|
|
<body>
|
|
<div id="container" style="display:none;">
|
|
<div id="header" class="hide-toggle">
|
|
<div id="header-title">
|
|
<img src="images/app-icons/png/24x24.png">
|
|
|
|
Trilium Notes
|
|
</div>
|
|
|
|
<div style="flex-grow: 100;">
|
|
<button class="btn btn-xs" onclick="jumpToNote.showDialog();" title="CTRL+J">Jump to note</button>
|
|
<button class="btn btn-xs" onclick="recentNotes.showDialog();" title="CTRL+E">Recent notes</button>
|
|
<button class="btn btn-xs" onclick="recentChanges.showDialog();">Recent changes</button>
|
|
</div>
|
|
|
|
<div id="plugin-buttons">
|
|
</div>
|
|
|
|
<div>
|
|
<button class="btn btn-xs" onclick="syncNow();" title="Number of outstanding changes to be pushed to server">
|
|
<span class="ui-icon ui-icon-refresh"></span>
|
|
|
|
Sync now (<span id="changes-to-push-count">0</span>)
|
|
</button>
|
|
|
|
<button class="btn btn-xs" onclick="settings.showDialog();">
|
|
<span class="ui-icon ui-icon-gear"></span> Settings</button>
|
|
|
|
<form action="logout" id="logout-button" method="POST" style="display: inline;">
|
|
<input type="submit" class="btn btn-xs" value="Logout">
|
|
</form>
|
|
</div>
|
|
</div>
|
|
|
|
<div class="hide-toggle" style="grid-area: tree-actions;">
|
|
<div style="display: flex; justify-content: space-around; padding: 10px 0 10px 0; margin: 0 20px 0 20px; border: 1px solid #ccc;">
|
|
<a onclick="noteTree.createNewTopLevelNote()" title="Create new top level note" class="icon-action">
|
|
<img src="images/icons/file-plus.png" alt="Create new top level note"/>
|
|
</a>
|
|
|
|
<a onclick="noteTree.collapseTree()" title="Collapse note tree" class="icon-action">
|
|
<img src="images/icons/list.png" alt="Collapse note tree"/>
|
|
</a>
|
|
|
|
<a onclick="noteTree.scrollToCurrentNote()" title="Scroll to current note. Shortcut CTRL+." class="icon-action">
|
|
<img src="images/icons/crosshair.png" alt="Scroll to current note"/>
|
|
</a>
|
|
|
|
<a onclick="searchTree.toggleSearch()" title="Search in notes" class="icon-action">
|
|
<img src="images/icons/search.png" alt="Search in notes"/>
|
|
</a>
|
|
</div>
|
|
</div>
|
|
|
|
<div id="search-box" class="hide-toggle" style="grid-area: search; display: none; padding: 10px; margin-top: 10px;">
|
|
<div style="display: flex; align-items: center;">
|
|
<label>Search:</label>
|
|
<input name="search-text" style="flex-grow: 100; margin-left: 5px; margin-right: 5px;" autocomplete="off">
|
|
<button id="reset-search-button" class="btn btn-sm" title="Reset search">×</button>
|
|
</div>
|
|
</div>
|
|
|
|
<div id="tree" class="hide-toggle" style="grid-area: tree; overflow: auto;">
|
|
</div>
|
|
|
|
<div id="parent-list" class="hide-toggle">
|
|
<p><strong>Note locations:</strong></p>
|
|
|
|
<ul id="parent-list-inner"></ul>
|
|
</div>
|
|
|
|
<div class="hide-toggle" style="grid-area: title;">
|
|
<div style="display: flex; align-items: center;">
|
|
<a onclick="protected_session.protectNoteAndSendToServer()"
|
|
title="Protect the note so that password will be required to view the note"
|
|
class="icon-action"
|
|
id="protect-button"
|
|
style="display: none;">
|
|
<img src="images/icons/lock.png" alt="Protect note"/>
|
|
</a>
|
|
|
|
<a onclick="protected_session.unprotectNoteAndSendToServer()"
|
|
title="Unprotect note so that password will not be required to access this note in the future"
|
|
class="icon-action"
|
|
id="unprotect-button"
|
|
style="display: none;">
|
|
<img src="images/icons/unlock.png" alt="Unprotect note"/>
|
|
</a>
|
|
|
|
|
|
|
|
<input autocomplete="off" value="" id="note-title" style="font-size: x-large; border: 0; flex-grow: 100;" tabindex="1">
|
|
|
|
<span id="note-id-display" title="Note ID"></span>
|
|
|
|
<button class="btn btn-sm"
|
|
style="display: none; margin-right: 10px"
|
|
id="execute-script-button"
|
|
onclick="noteEditor.executeCurrentNote()">Execute <kbd>Ctrl+Enter</kbd></button>
|
|
|
|
<div class="dropdown" id="note-type">
|
|
<button id="dLabel" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" class="btn btn-sm">
|
|
Type: <span data-bind="text: typeString()"></span>
|
|
<span class="caret"></span>
|
|
</button>
|
|
<ul id="note-type-dropdown" class="dropdown-menu dropdown-menu-right" aria-labelledby="dLabel">
|
|
<li data-bind="click: selectText, css: { selected: type() == 'text' }"><span class="check">✓</span> <strong>Text</strong></li>
|
|
<li role="separator" class="divider"></li>
|
|
<li data-bind="click: selectRender, css: { selected: type() == 'render' && mime() == '' }"><span class="check">✓</span> <strong>Render HTML note</strong></li>
|
|
<li role="separator" class="divider"></li>
|
|
<li data-bind="click: selectCode, css: { selected: type() == 'code' && mime() == '' }"><span class="check">✓</span> <strong>Code</strong></li>
|
|
<!-- ko foreach: codeMimeTypes -->
|
|
<li data-bind="click: $parent.selectCodeMime, css: { selected: $parent.type() == 'code' && $parent.mime() == mime }"><span class="check">✓</span> <span data-bind="text: title"></span></li>
|
|
<!-- /ko -->
|
|
</ul>
|
|
</div>
|
|
|
|
<div class="dropdown" style="margin-left: 10px; margin-right: 10px;">
|
|
<button id="dLabel" type="button" data-toggle="dropdown" aria-haspopup="true" aria-expanded="false" class="btn btn-sm">
|
|
Note actions
|
|
<span class="caret"></span>
|
|
</button>
|
|
<ul class="dropdown-menu dropdown-menu-right" aria-labelledby="dLabel">
|
|
<li><a onclick="noteHistory.showCurrentNoteHistory();"><kbd>Alt+H</kbd> History</a></li>
|
|
<li><a onclick="attributesDialog.showDialog();"><kbd>Alt+A</kbd> Attributes</a></li>
|
|
<li><a onclick="noteSource.showDialog();"><kbd>Ctrl+U</kbd> HTML source</a></li>
|
|
</ul>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div style="position: relative; overflow: auto; grid-area: note-content; padding-left: 10px; padding-top: 10px;" id="note-detail-wrapper">
|
|
<div id="note-detail"></div>
|
|
|
|
<div id="note-detail-code"></div>
|
|
|
|
<div id="note-detail-render"></div>
|
|
</div>
|
|
|
|
<div id="attribute-list">
|
|
<button class="btn btn-sm" onclick="attributesDialog.showDialog();">Attributes:</button>
|
|
|
|
<span id="attribute-list-inner"></span>
|
|
</div>
|
|
</div>
|
|
|
|
<div id="recent-notes-dialog" title="Recent notes" style="display: none;">
|
|
<input id="recent-notes-search-input" class="form-control"/>
|
|
</div>
|
|
|
|
<div id="add-link-dialog" title="Add link" style="display: none;">
|
|
<form id="add-link-form">
|
|
<div class="radio">
|
|
<label title="Add HTML link to the selected note at cursor in current note">
|
|
<input type="radio" name="add-link-type" value="html"/>
|
|
add normal HTML link</label>
|
|
|
|
<label title="Add selected note as a child of current note">
|
|
<input type="radio" name="add-link-type" value="selected-to-current"/>
|
|
add selected note to current note</label>
|
|
|
|
<label title="Add current note as a child of the selected note">
|
|
<input type="radio" name="add-link-type" value="current-to-selected"/>
|
|
add current note to selected note</label>
|
|
</div>
|
|
|
|
<div class="form-group">
|
|
<label for="note-autocomplete">Note</label>
|
|
<input id="note-autocomplete" style="width: 100%;">
|
|
</div>
|
|
|
|
<div class="form-group" id="add-link-title-form-group">
|
|
<label for="link-title">Link title</label>
|
|
<input id="link-title" style="width: 100%;">
|
|
</div>
|
|
|
|
<div class="form-group" id="add-link-prefix-form-group" title="Cloned note will be shown in note tree with given prefix">
|
|
<label for="clone-prefix">Prefix (optional)</label>
|
|
<input id="clone-prefix" style="width: 100%;">
|
|
</div>
|
|
|
|
<button class="btn btn-sm">Add link</button>
|
|
</form>
|
|
</div>
|
|
|
|
<div id="jump-to-note-dialog" title="Jump to note" style="display: none;">
|
|
<form id="jump-to-note-form">
|
|
<div class="form-group">
|
|
<label for="jump-to-note-autocomplete">Note</label>
|
|
<input id="jump-to-note-autocomplete" style="width: 100%;">
|
|
</div>
|
|
|
|
<button name="action" value="jump" class="btn btn-sm">Jump <kbd>enter</kbd></button>
|
|
</form>
|
|
</div>
|
|
|
|
<div id="protected-session-password-dialog" title="Protected session" style="display: none;">
|
|
<form id="protected-session-password-form">
|
|
<div class="form-group">
|
|
<label for="protected-session-password">To proceed with requested action you need to start protected session by entering password:</label>
|
|
<input id="protected-session-password" class="form-control" type="password">
|
|
</div>
|
|
|
|
<button class="btn btn-sm">Start protected session <kbd>enter</kbd></button>
|
|
</form>
|
|
</div>
|
|
|
|
<div id="settings-dialog" title="Settings" style="display: none;">
|
|
<div id="settings-tabs">
|
|
<ul>
|
|
<li><a href="#change-password">Change password</a></li>
|
|
<li><a href="#protected-session-timeout">Protected session</a></li>
|
|
<li><a href="#history-snapshot-time-interval">History snapshots</a></li>
|
|
<li><a href="#advanced">Advanced</a></li>
|
|
<li><a href="#about">About Trilium</a></li>
|
|
</ul>
|
|
<div id="change-password">
|
|
<form id="change-password-form">
|
|
<div class="form-group">
|
|
<label for="old-password">Old password</label>
|
|
<input class="form-control" id="old-password" type="password">
|
|
</div>
|
|
|
|
<div class="form-group">
|
|
<label for="new-password1">New password</label>
|
|
<input class="form-control" id="new-password1" type="password">
|
|
</div>
|
|
|
|
<div class="form-group">
|
|
<label for="new-password2">New password once more</label>
|
|
<input class="form-control" id="new-password2" type="password">
|
|
</div>
|
|
|
|
<button class="btn btn-sm">Change password</button>
|
|
</form>
|
|
</div>
|
|
<div id="protected-session-timeout">
|
|
<p>Protected session timeout is a time period after which the protected session is wiped out from
|
|
browser's memory. This is measured from the last interaction with protected notes.</p>
|
|
|
|
<form id="protected-session-timeout-form">
|
|
<div class="form-group">
|
|
<label for="protected-session-timeout-in-seconds">Protected session timeout (in seconds)</label>
|
|
<input class="form-control" id="protected-session-timeout-in-seconds" type="number">
|
|
</div>
|
|
|
|
<button class="btn btn-sm">Save</button>
|
|
</form>
|
|
</div>
|
|
<div id="history-snapshot-time-interval">
|
|
<p>History snapshot time interval is time in seconds after which new history record will be created for the note.</p>
|
|
|
|
<form id="history-snapshot-time-interval-form">
|
|
<div class="form-group">
|
|
<label for="history-snapshot-time-interval-in-seconds">History snapshot time interval (in seconds)</label>
|
|
<input class="form-control" id="history-snapshot-time-interval-in-seconds" type="number">
|
|
</div>
|
|
|
|
<button class="btn btn-sm">Save</button>
|
|
</form>
|
|
</div>
|
|
<div id="advanced">
|
|
<h4 style="margin-top: 0px;">Sync</h4>
|
|
<button id="force-full-sync-button" class="btn btn-sm">Force full sync</button>
|
|
|
|
<br/>
|
|
<br/>
|
|
|
|
<button id="fill-sync-rows-button" class="btn btn-sm">Fill sync rows</button>
|
|
|
|
<h4>Debugging</h4>
|
|
|
|
<button id="anonymize-button" class="btn btn-sm">Save anonymized database</button><br/><br/>
|
|
|
|
<p>This action will create a new copy of the database and anonymise it (remove all note content and leave only structure and metadata)
|
|
for sharing online for debugging purposes without fear of leaking your personal data.</p>
|
|
|
|
<h4>Image cleanup</h4>
|
|
|
|
<p>This will remove all image data of images not used in any current version of note from the database (metadata will remain).
|
|
|
|
This means that some images can disappear from note history.</p>
|
|
|
|
<button id="cleanup-unused-images-button" class="btn btn-warning btn-sm">Permanently cleanup unused images</button>
|
|
|
|
<h4>Soft-delete cleanup</h4>
|
|
|
|
<p>This deletes all soft deleted rows from the database. This change isn't synced and should be done manually on all instances.
|
|
<strong>Use this only if you really know what you're doing.</strong></p>
|
|
|
|
<button id="cleanup-soft-deleted-items-button" class="btn btn-danger btn-sm">Permanently cleanup soft-deleted items</button>
|
|
|
|
<h4>Vacuum database</h4>
|
|
|
|
<p>This will rebuild database which will typically result in smaller database file. No data will be actually changed.</p>
|
|
|
|
<button id="vacuum-database-button" class="btn btn-sm">Vacuum database</button>
|
|
</div>
|
|
<div id="about">
|
|
<table class="table">
|
|
<tr>
|
|
<th>App version:</th>
|
|
<td id="app-version"></td>
|
|
</tr>
|
|
|
|
<tr>
|
|
<th>DB version:</th>
|
|
<td id="db-version"></td>
|
|
</tr>
|
|
|
|
<tr>
|
|
<th>Build date:</th>
|
|
<td id="build-date"></td>
|
|
</tr>
|
|
|
|
<tr>
|
|
<th>Build revision:</th>
|
|
<td><a href="" target="_blank" id="build-revision"></a></td>
|
|
</tr>
|
|
</table>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div id="note-history-dialog" title="Note history" style="display: none;">
|
|
<div style="display: flex;">
|
|
<select id="note-history-list" size="25" style="width: 150px; height: 630px;">
|
|
</select>
|
|
|
|
<div id="note-history-content-wrapper" style="flex-grow: 1; margin-left: 20px;">
|
|
<h3 id="note-history-title" style="margin: 3px;"></h3>
|
|
|
|
<div id="note-history-content" style="height: 600px; width: 600px; overflow: auto;"></div>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<div id="recent-changes-dialog" title="Recent changes" style="display: none; padding: 20px;">
|
|
</div>
|
|
|
|
<div id="event-log-dialog" title="Event log" style="display: none; padding: 20px;">
|
|
<ul id="event-log-list"></ul>
|
|
</div>
|
|
|
|
<div id="edit-tree-prefix-dialog" title="Edit tree prefix" style="display: none; padding: 20px;">
|
|
<form id="edit-tree-prefix-form">
|
|
<div class="form-group">
|
|
<label for="tree-prefix-input">Prefix</label>
|
|
<input id="tree-prefix-input" style="width: 20em;"> - <span id="tree-prefix-note-title"></span>
|
|
</div>
|
|
|
|
<button name="action" class="btn btn-sm">Save</button>
|
|
</form>
|
|
</div>
|
|
|
|
<div id="sql-console-dialog" title="SQL console" style="display: none; padding: 20px;">
|
|
<textarea style="width: 100%; height: 100px" id="sql-console-query"></textarea>
|
|
<button class="btn btn-danger" id="sql-console-execute">Execute <kbd>CTRL+ENTER</kbd></button>
|
|
|
|
<table id="sql-console-results" class="table table-striped" style="overflow: scroll; width: 100%;">
|
|
<thead></thead>
|
|
<tbody></tbody>
|
|
</table>
|
|
</div>
|
|
|
|
<div id="note-source-dialog" title="Note source" style="display: none; padding: 20px;">
|
|
<textarea id="note-source" readonly="readonly"></textarea>
|
|
</div>
|
|
|
|
<div id="attributes-dialog" title="Note attributes" style="display: none; padding: 20px;">
|
|
<form data-bind="submit: save">
|
|
<div style="text-align: center">
|
|
<button class="btn btn-large" style="width: 200px;" id="save-attributes-button" type="submit">Save <kbd>enter</kbd></button>
|
|
</div>
|
|
|
|
<div style="height: 97%; overflow: auto">
|
|
<table id="attributes-table" class="table">
|
|
<thead>
|
|
<tr>
|
|
<th>ID</th>
|
|
<th>Name</th>
|
|
<th>Value</th>
|
|
</tr>
|
|
</thead>
|
|
<tbody data-bind="foreach: attributes">
|
|
<tr>
|
|
<td data-bind="text: attributeId"></td>
|
|
<td>
|
|
<!-- Change to valueUpdate: blur is necessary because jQuery UI autocomplete hijacks change event -->
|
|
<input type="text" class="attribute-name" data-bind="value: name, valueUpdate: 'blur', event: { blur: $parent.attributeChanged }"/>
|
|
|
|
<div style="color: red" data-bind="if: $parent.isNotUnique($index())">Attribute name must be unique per note.</div>
|
|
<div style="color: red" data-bind="if: $parent.isEmptyName($index())">Attribute name can't be empty.</div>
|
|
</td>
|
|
<td>
|
|
<input type="text" class="attribute-value" data-bind="value: value, valueUpdate: 'blur', event: { blur: $parent.attributeChanged }" style="width: 300px"/>
|
|
</td>
|
|
</tr>
|
|
</tbody>
|
|
</table>
|
|
</div>
|
|
</form>
|
|
</div>
|
|
|
|
<div id="tooltip" style="display: none;"></div>
|
|
|
|
<script type="text/javascript">
|
|
const baseApiUrl = 'api/';
|
|
const glob = {
|
|
activeDialog: null,
|
|
sourceId: '<%= sourceId %>',
|
|
maxSyncIdAtLoad: <%= maxSyncIdAtLoad %>
|
|
};
|
|
</script>
|
|
|
|
<!-- Required for correct loading of scripts in Electron -->
|
|
<script>if (typeof module === 'object') {window.module = module; module = undefined;}</script>
|
|
|
|
<script src="libraries/jquery.min.js"></script>
|
|
|
|
<!-- bootstrap needs to be included before jQuery UI, otherwise close icon in the dialog will be missing -->
|
|
<link href="libraries/bootstrap/css/bootstrap.css" rel="stylesheet">
|
|
<script src="libraries/bootstrap/js/bootstrap.js"></script>
|
|
|
|
<link href="libraries/jqueryui/jquery-ui.min.css" rel="stylesheet">
|
|
<script src="libraries/jqueryui/jquery-ui.min.js"></script>
|
|
|
|
<script src="libraries/bootstrap-notify.min.js"></script>
|
|
|
|
<!-- Include Fancytree skin and library -->
|
|
<link href="libraries/fancytree/skin-win8/ui.fancytree.css" rel="stylesheet">
|
|
<script src="libraries/fancytree/jquery.fancytree-all.min.js"></script>
|
|
|
|
<script src="libraries/ckeditor/ckeditor.js"></script>
|
|
|
|
<script src="libraries/jquery.hotkeys.js"></script>
|
|
<script src="libraries/jquery.fancytree.hotkeys.js"></script>
|
|
|
|
<script src="libraries/jquery.ui-contextmenu.min.js"></script>
|
|
|
|
<script src="libraries/knockout.min.js"></script>
|
|
|
|
<script src="libraries/codemirror/codemirror.js"></script>
|
|
<link rel="stylesheet" href="libraries/codemirror/codemirror.css">
|
|
<script src="libraries/codemirror/addon/mode/loadmode.js"></script>
|
|
<script src="libraries/codemirror/addon/fold/xml-fold.js"></script>
|
|
<script src="libraries/codemirror/addon/edit/matchbrackets.js"></script>
|
|
<script src="libraries/codemirror/addon/edit/matchtags.js"></script>
|
|
<script src="libraries/codemirror/addon/search/match-highlighter.js"></script>
|
|
<script src="libraries/codemirror/mode/meta.js"></script>
|
|
|
|
<link href="stylesheets/style.css" rel="stylesheet">
|
|
|
|
<script src="javascripts/utils.js"></script>
|
|
<script src="javascripts/init.js"></script>
|
|
<script src="javascripts/server.js"></script>
|
|
|
|
<!-- Tree scripts -->
|
|
<script src="javascripts/note_tree.js"></script>
|
|
<script src="javascripts/tree_changes.js"></script>
|
|
<script src="javascripts/cloning.js"></script>
|
|
<script src="javascripts/tree_utils.js"></script>
|
|
<script src="javascripts/drag_and_drop.js"></script>
|
|
<script src="javascripts/context_menu.js"></script>
|
|
<script src="javascripts/search_tree.js"></script>
|
|
|
|
<!-- Note detail -->
|
|
<script src="javascripts/note_editor.js"></script>
|
|
<script src="javascripts/protected_session.js"></script>
|
|
<script src="javascripts/note_type.js"></script>
|
|
|
|
<!-- dialogs -->
|
|
<script src="javascripts/dialogs/recent_notes.js"></script>
|
|
<script src="javascripts/dialogs/add_link.js"></script>
|
|
<script src="javascripts/dialogs/jump_to_note.js"></script>
|
|
<script src="javascripts/dialogs/settings.js"></script>
|
|
<script src="javascripts/dialogs/note_history.js"></script>
|
|
<script src="javascripts/dialogs/recent_changes.js"></script>
|
|
<script src="javascripts/dialogs/event_log.js"></script>
|
|
<script src="javascripts/dialogs/edit_tree_prefix.js"></script>
|
|
<script src="javascripts/dialogs/sql_console.js"></script>
|
|
<script src="javascripts/dialogs/note_source.js"></script>
|
|
<script src="javascripts/dialogs/attributes.js"></script>
|
|
|
|
<script src="javascripts/link.js"></script>
|
|
<script src="javascripts/sync.js"></script>
|
|
<script src="javascripts/messaging.js"></script>
|
|
<script src="javascripts/api.js"></script>
|
|
|
|
<script type="text/javascript">
|
|
// we hide container initally because otherwise it is rendered first without CSS and then flickers into
|
|
// final form which is pretty ugly.
|
|
$("#container").show();
|
|
</script>
|
|
</body>
|
|
</html> |