Sorting of "folders" before leafs

This commit is contained in:
Magnus Åhall 2026-06-02 18:50:29 +02:00
parent 2f27aeffb3
commit cc69f7194e
3 changed files with 42 additions and 8 deletions

View file

@ -176,14 +176,32 @@ export class NodeStore {
const nodeStore = trx.objectStore('nodes') const nodeStore = trx.objectStore('nodes')
const index = nodeStore.index('byParent') const index = nodeStore.index('byParent')
const req = index.getAll(storeParent) const req = index.getAll(storeParent)
const hasChildrenPromises = []
req.onsuccess = (event) => { req.onsuccess = (event) => {
const nodes = [] const nodes = []
for (const i in event.target.result) { for (const i in event.target.result) {
const nodeData = event.target.result[i] const nodeData = event.target.result[i]
const node = this.node(nodeData.UUID, nodeData, newLevel) const node = this.node(nodeData.UUID, nodeData, newLevel)
// Look for the key of any children, a hopefully fast way
// to tell if any children exists at all and this node is a
// "folder". Needed quite early on for sorting.
const promise = new Promise((resolve, reject) => {
const countReq = index.getKey(nodeData.UUID)
countReq.onsuccess = event => {
node.setHasChildren(event.target.result !== undefined)
resolve()
}
})
hasChildrenPromises.push(promise)
nodes.push(node) nodes.push(node)
} }
resolve(nodes)
Promise.all(hasChildrenPromises)
.then(() => {
resolve(nodes)
})
} }
req.onerror = (event) => reject(event.target.error) req.onerror = (event) => reject(event.target.error)
}) })

View file

@ -257,8 +257,15 @@ customElements.define('n2-nodeui', N2PageNodeUI)
export class Node { export class Node {
static sort(a, b) {//{{{ static sort(a, b) {//{{{
if (a.data.Name < b.data.Name) return -1 // Nodes with children ("folders") are sorted first.
if (a.data.Name > b.data.Name) return 0 if (a._has_children && !b._has_children) return -1
if (!a._has_children && b._has_children) return 1
// Otherwise sort by lowercased name.
const an = a.data.Name.toLowerCase()
const bn = b.data.Name.toLowerCase()
if (an < bn) return -1
if (an > bn) return 1
return 0 return 0
}//}}} }//}}}
static create(name, parentUUID) {// {{{ static create(name, parentUUID) {// {{{
@ -286,6 +293,7 @@ export class Node {
this.ParentUUID = nodeData.ParentUUID this.ParentUUID = nodeData.ParentUUID
this._children_fetched = false this._children_fetched = false
this._has_children = null // this will be set by nodeStore.getTreeNodes
this.Children = [] this.Children = []
this.Ancestors = [] this.Ancestors = []
@ -322,6 +330,7 @@ export class Node {
this.Children.sort(Node.sort) this.Children.sort(Node.sort)
const numChildren = this.Children.length const numChildren = this.Children.length
this.setHasChildren(numChildren > 0)
for (let i = 0; i < numChildren; i++) { for (let i = 0; i < numChildren; i++) {
if (i > 0) if (i > 0)
this.Children[i]._sibling_before = this.Children[i - 1] this.Children[i]._sibling_before = this.Children[i - 1]
@ -336,8 +345,11 @@ export class Node {
return this.Children return this.Children
}//}}} }//}}}
setHasChildren(v) {// {{{
this._has_children = v
}// }}}
hasChildren() {//{{{ hasChildren() {//{{{
return this.Children.length > 0 return this._has_children
}//}}} }//}}}
getSiblingBefore() {// {{{ getSiblingBefore() {// {{{
return this._sibling_before return this._sibling_before

View file

@ -91,7 +91,11 @@ export class N2Tree extends CustomHTMLElement {
} }
_mbus.dispatch('TREE_TRUNK_FETCHED') _mbus.dispatch('TREE_TRUNK_FETCHED')
}) })
.catch(e => { console.log(e); console.log(e.type, e.error); alert(e.error) }) .catch(e => {
console.error(e)
console.log(e.type, e.error)
alert(e.error)
})
}//}}} }//}}}
getNodeExpanded(UUID) {//{{{ getNodeExpanded(UUID) {//{{{
if (this.expandedNodes[UUID] === undefined) if (this.expandedNodes[UUID] === undefined)
@ -396,7 +400,7 @@ export class N2TreeNode extends CustomHTMLElement {
this.render(true) this.render(true)
}) })
_mbus.subscribe(`NODE_EXPAND_${node.UUID}`, state => { _mbus.subscribe(`NODE_EXPAND_${node.UUID}`, _state => {
this.render(true) this.render(true)
}) })
@ -412,7 +416,7 @@ export class N2TreeNode extends CustomHTMLElement {
return this return this
// Fetch the next level of children if the parent tree node is expanded and our children thus will be visible. // Fetch the next level of children if the parent tree node is expanded and our children thus will be visible.
const expanded = this.node.Children.length > 0 && this.tree.getNodeExpanded(this.node.UUID) const expanded = this.node.hasChildren() && this.tree.getNodeExpanded(this.node.UUID)
if (!this.children_populated && this.tree.getNodeExpanded(this.parent?.node.UUID)) { if (!this.children_populated && this.tree.getNodeExpanded(this.parent?.node.UUID)) {
this.node.fetchChildren().then(() => this.children_populated = true) this.node.fetchChildren().then(() => this.children_populated = true)
@ -435,7 +439,7 @@ export class N2TreeNode extends CustomHTMLElement {
} }
// The expand icon <img> is only changed to not get a flickering when re-rendering. // The expand icon <img> is only changed to not get a flickering when re-rendering.
if (this.node.Children.length === 0) if (!this.node.hasChildren())
this.setImgSrc(this.elExpand, `/images/${window._VERSION}/leaf.svg`) this.setImgSrc(this.elExpand, `/images/${window._VERSION}/leaf.svg`)
else if (this.tree.getNodeExpanded(this.node.UUID)) else if (this.tree.getNodeExpanded(this.node.UUID))
this.setImgSrc(this.elExpand, `/images/${window._VERSION}/expanded.svg`) this.setImgSrc(this.elExpand, `/images/${window._VERSION}/expanded.svg`)