note revisions as a bootstrap list instead of <select>

This commit is contained in:
zadam 2019-11-09 20:59:49 +01:00
parent 3a1c80c189
commit 169551c6a8
3 changed files with 59 additions and 19 deletions

View File

@ -5,13 +5,24 @@ import toastService from "../services/toast.js";
const $dialog = $("#note-revisions-dialog"); const $dialog = $("#note-revisions-dialog");
const $list = $("#note-revision-list"); const $list = $("#note-revision-list");
const $listDropdown = $("#note-revision-list-dropdown");
const $content = $("#note-revision-content"); const $content = $("#note-revision-content");
const $title = $("#note-revision-title"); const $title = $("#note-revision-title");
const $titleButtons = $("#note-revision-title-buttons"); const $titleButtons = $("#note-revision-title-buttons");
const $eraseAllRevisionsButton = $("#note-revisions-erase-all-revisions-button"); const $eraseAllRevisionsButton = $("#note-revisions-erase-all-revisions-button");
$listDropdown.dropdown();
$listDropdown.parent().on('hide.bs.dropdown', e => {
// prevent closing dropdown by clicking outside
if (e.clickEvent) {
e.preventDefault();
}
});
let revisionItems = []; let revisionItems = [];
let note; let note;
let noteRevisionId;
export async function showCurrentNoteRevisions() { export async function showCurrentNoteRevisions() {
await showNoteRevisionsDialog(noteDetailService.getActiveTabNoteId()); await showNoteRevisionsDialog(noteDetailService.getActiveTabNoteId());
@ -27,7 +38,7 @@ export async function showNoteRevisionsDialog(noteId, noteRevisionId) {
await loadNoteRevisions(noteId, noteRevisionId); await loadNoteRevisions(noteId, noteRevisionId);
} }
async function loadNoteRevisions(noteId, noteRevisionId) { async function loadNoteRevisions(noteId, noteRevId) {
$list.empty(); $list.empty();
$content.empty(); $content.empty();
@ -35,27 +46,34 @@ async function loadNoteRevisions(noteId, noteRevisionId) {
revisionItems = await server.get(`notes/${noteId}/revisions`); revisionItems = await server.get(`notes/${noteId}/revisions`);
for (const item of revisionItems) { for (const item of revisionItems) {
$list.append($('<option>', { $list.append($('<a class="dropdown-item" tabindex="0">')
value: item.noteRevisionId, .text(item.dateLastEdited.substr(0, 16) + ` (${item.contentLength} bytes)`)
text: item.dateLastEdited.substr(0, 16) + ` (${item.contentLength} bytes)` .attr('data-note-revision-id', item.noteRevisionId));
}));
} }
$listDropdown.dropdown('show');
noteRevisionId = noteRevId;
if (revisionItems.length > 0) { if (revisionItems.length > 0) {
if (!noteRevisionId) { if (!noteRevisionId) {
noteRevisionId = $list.find("option:first").val(); noteRevisionId = revisionItems[0].noteRevisionId;
} }
$list.val(noteRevisionId).trigger('change');
} else { } else {
$title.text("No revisions for this note yet..."); $title.text("No revisions for this note yet...");
noteRevisionId = null;
} }
} }
$list.on('change', async () => { $dialog.on('shown.bs.modal', () => {
const optVal = $list.find(":selected").val(); $list.find(`[data-note-revision-id="${noteRevisionId}"]`)
.trigger('focus');
});
const revisionItem = revisionItems.find(r => r.noteRevisionId === optVal); async function setContentPane() {
const noteRevisionId = $list.find(".active").attr('data-note-revision-id');
const revisionItem = revisionItems.find(r => r.noteRevisionId === noteRevisionId);
$titleButtons.empty(); $titleButtons.empty();
$content.empty(); $content.empty();
@ -102,7 +120,8 @@ $list.on('change', async () => {
// reason why we put this inline as base64 is that we do not want to let user to copy this // reason why we put this inline as base64 is that we do not want to let user to copy this
// as a URL to be used in a note. Instead if they copy and paste it into a note, it will be a uploaded as a new note // as a URL to be used in a note. Instead if they copy and paste it into a note, it will be a uploaded as a new note
.attr("src", `data:${note.mime};base64,` + fullNoteRevision.content) .attr("src", `data:${note.mime};base64,` + fullNoteRevision.content)
.css("width", "100%")); .css("max-width", "100%")
.css("max-height", "100%"));
} }
else if (revisionItem.type === 'file') { else if (revisionItem.type === 'file') {
const $table = $("<table cellpadding='10'>") const $table = $("<table cellpadding='10'>")
@ -130,7 +149,7 @@ $list.on('change', async () => {
else { else {
$content.text("Preview isn't available for this note type."); $content.text("Preview isn't available for this note type.");
} }
}); }
$eraseAllRevisionsButton.on('click', async () => { $eraseAllRevisionsButton.on('click', async () => {
const confirmDialog = await import('../dialogs/confirm.js'); const confirmDialog = await import('../dialogs/confirm.js');
@ -143,4 +162,17 @@ $eraseAllRevisionsButton.on('click', async () => {
toastService.showMessage('Note revisions has been deleted.'); toastService.showMessage('Note revisions has been deleted.');
} }
});
$list.on('click', '.dropdown-item', e => {
e.preventDefault();
return false;
});
$list.on('focus', '.dropdown-item', e => {
$list.find('.dropdown-item').each((i, el) => {
$(el).toggleClass('active', el === e.target);
});
setContentPane();
}); });

View File

@ -351,6 +351,11 @@ div.ui-tooltip {
color: #aaa !important; color: #aaa !important;
} }
.dropdown-item.active {
color: var(--active-item-text-color) !important;
background-color: var(--active-item-background-color) !important;
}
.note-detail-code { .note-detail-code {
overflow: auto; overflow: auto;
} }

View File

@ -15,18 +15,21 @@
<span aria-hidden="true">&times;</span> <span aria-hidden="true">&times;</span>
</button> </button>
</div> </div>
<div class="modal-body" style="display: flex;"> <div class="modal-body" style="display: flex; height: 80vh;">
<select id="note-revision-list" size="25" style="min-width: 230px; height: 630px;"> <div class="dropdown">
</select> <button id="note-revision-list-dropdown" type="button" style="display: none;" data-toggle="dropdown">Dropdown trigger</button>
<div id="note-revision-content-wrapper" style="flex-grow: 1; margin-left: 20px;"> <div id="note-revision-list" style="position: static; height: 100%; overflow: auto;" class="dropdown-menu"></div>
<div style="display: flex; justify-content: space-between;"> </div>
<div id="note-revision-content-wrapper" style="flex-grow: 1; margin-left: 20px; display: flex; flex-direction: column;">
<div style="flex-grow: 0; display: flex; justify-content: space-between;">
<h3 id="note-revision-title" style="margin: 3px; flex-grow: 100;"></h3> <h3 id="note-revision-title" style="margin: 3px; flex-grow: 100;"></h3>
<div id="note-revision-title-buttons"></div> <div id="note-revision-title-buttons"></div>
</div> </div>
<div id="note-revision-content" style="height: 600px; overflow: auto;"></div> <div id="note-revision-content" style="overflow: auto;"></div>
</div> </div>
</div> </div>
</div> </div>