import utils from "./utils.js"; import Branch from "../entities/branch.js"; import server from "./server.js"; import treeCache from "./tree_cache.js"; import ws from "./ws.js"; import hoistedNoteService from "./hoisted_note.js"; async function prepareTree() { const hoistedNoteId = await hoistedNoteService.getHoistedNoteId(); let hoistedBranch; if (hoistedNoteId === 'root') { hoistedBranch = await treeCache.getBranch('root'); } else { const hoistedNote = await treeCache.getNote(hoistedNoteId); hoistedBranch = (await hoistedNote.getBranches())[0]; } return [ await prepareNode(hoistedBranch) ]; } async function prepareBranch(note) { if (note.type === 'search') { return await prepareSearchBranch(note); } else { return await prepareRealBranch(note); } } const NOTE_TYPE_ICONS = { "file": "jam jam-attachment", "image": "jam jam-picture", "code": "jam jam-terminal", "render": "jam jam-play", "search": "jam jam-search-folder", "relation-map": "jam jam-map", "book": "jam jam-book" }; async function getIcon(note) { const hoistedNoteId = await hoistedNoteService.getHoistedNoteId(); if (note.noteId === 'root') { return "jam jam-chevrons-right"; } else if (note.noteId === hoistedNoteId) { return "jam jam-arrow-up"; } else if (note.type === 'text') { if (note.hasChildren()) { return "jam jam-folder"; } else { return "jam jam-file"; } } else { return NOTE_TYPE_ICONS[note.type]; } } async function prepareNode(branch) { const note = await branch.getNote(); const title = (branch.prefix ? (branch.prefix + " - ") : "") + note.title; const hoistedNoteId = await hoistedNoteService.getHoistedNoteId(); const node = { noteId: note.noteId, parentNoteId: branch.parentNoteId, branchId: branch.branchId, isProtected: note.isProtected, title: utils.escapeHtml(title), extraClasses: await getExtraClasses(note), icon: await getIcon(note), refKey: note.noteId, expanded: branch.isExpanded || hoistedNoteId === note.noteId, lazy: true, key: utils.randomString(12) // this should prevent some "duplicate key" errors }; if (note.hasChildren() || note.type === 'search') { node.folder = true; } return node; } async function prepareRealBranch(parentNote) { utils.assertArguments(parentNote); const childBranches = await parentNote.getChildBranches(); if (!childBranches) { ws.logError(`No children for ${parentNote}. This shouldn't happen.`); return; } const noteList = []; for (const branch of childBranches) { const node = await prepareNode(branch); noteList.push(node); } return noteList; } async function prepareSearchBranch(note) { const results = await server.get('search-note/' + note.noteId); // force to load all the notes at once instead of one by one await treeCache.getNotes(results.map(res => res.noteId)); for (const result of results) { const origBranch = await treeCache.getBranch(result.branchId); const branch = new Branch(treeCache, { branchId: "virt" + utils.randomString(10), noteId: result.noteId, parentNoteId: note.noteId, prefix: origBranch.prefix, virtual: true }); treeCache.addBranch(branch); } return await prepareRealBranch(note); } async function getExtraClasses(note) { utils.assertArguments(note); const extraClasses = []; if (note.isProtected) { extraClasses.push("protected"); } if (note.getParentNoteIds().length > 1) { extraClasses.push("multiple-parents"); } if (note.cssClass) { extraClasses.push(note.cssClass); } extraClasses.push(utils.getNoteTypeClass(note.type)); if (note.mime) { // some notes should not have mime type (e.g. render) extraClasses.push(utils.getMimeTypeClass(note.mime)); } if (note.archived) { extraClasses.push("archived"); } return extraClasses.join(" "); } export default { prepareTree, prepareBranch, getExtraClasses, getIcon }