@difizen/mana-app
Version:
163 lines (142 loc) • 4.77 kB
text/typescript
import type { Event, Disposable } from '@difizen/mana-common';
import { Emitter } from '@difizen/mana-common';
import { singleton, inject, postConstruct } from '@difizen/mana-syringe';
import type { TreeNode } from './tree';
import { CompositeTreeNode, Tree } from './tree';
export const TreeExpansionService = Symbol('TreeExpansionService');
/**
* The tree expandable service.
*/
export type TreeExpansionService = {
/**
* Emit when the node is expanded or collapsed.
*/
readonly onExpansionChanged: Event<Readonly<ExpandableTreeNode>>;
/**
* Expand a node for the given node id if it is valid and collapsed.
* Expanding a node refreshes all its children.
*
* Return a valid expanded refreshed node or `undefined` if such does not exist.
*/
expandNode: (
node: Readonly<ExpandableTreeNode>,
) => Promise<Readonly<ExpandableTreeNode> | undefined>;
/**
* If the given node is valid and expanded then collapse it.
*
* Return true if a node has been collapsed; otherwise false.
*/
collapseNode: (node: Readonly<ExpandableTreeNode>) => Promise<boolean>;
/**
* If the given node is valid then collapse it recursively.
*
* Return true if a node has been collapsed; otherwise false.
*/
collapseAll: (node: Readonly<CompositeTreeNode>) => Promise<boolean>;
/**
* If the given node is invalid then does nothing.
* If the given node is collapsed then expand it; otherwise collapse it.
*/
toggleNodeExpansion: (node: Readonly<ExpandableTreeNode>) => Promise<void>;
} & Disposable;
/**
* The expandable tree node.
*/
export type ExpandableTreeNode = {
/**
* Test whether this tree node is expanded.
*/
expanded: boolean;
} & CompositeTreeNode;
export namespace ExpandableTreeNode {
export function is(node: Record<any, any> | undefined): node is ExpandableTreeNode {
return !!node && CompositeTreeNode.is(node) && 'expanded' in node;
}
export function isExpanded(
node: Record<any, any> | undefined,
): node is ExpandableTreeNode {
return ExpandableTreeNode.is(node) && node.expanded;
}
export function isCollapsed(
node: Record<any, any> | undefined,
): node is ExpandableTreeNode {
return ExpandableTreeNode.is(node) && !node.expanded;
}
}
({ contrib: TreeExpansionService })
export class TreeExpansionServiceImpl implements TreeExpansionService {
protected readonly onExpansionChangedEmitter = new Emitter<ExpandableTreeNode>();
protected readonly tree: Tree;
constructor((Tree) tree: Tree) {
this.tree = tree;
}
()
protected init(): void {
this.tree.onNodeRefreshed((node) => {
for (const child of node.children) {
if (ExpandableTreeNode.isExpanded(child) && node.waitUntil) {
node.waitUntil(this.tree.refresh(child));
}
}
});
}
dispose(): void {
this.onExpansionChangedEmitter.dispose();
}
get onExpansionChanged(): Event<ExpandableTreeNode> {
return this.onExpansionChangedEmitter.event;
}
protected fireExpansionChanged(node: ExpandableTreeNode): void {
this.onExpansionChangedEmitter.fire(node);
}
async expandNode(raw: ExpandableTreeNode): Promise<ExpandableTreeNode | undefined> {
const node = this.tree.validateNode(raw);
if (ExpandableTreeNode.isCollapsed(node)) {
return this.doExpandNode(node);
}
return undefined;
}
protected async doExpandNode(
node: ExpandableTreeNode,
): Promise<ExpandableTreeNode | undefined> {
const refreshed = await this.tree.refresh(node);
if (ExpandableTreeNode.is(refreshed)) {
refreshed.expanded = true;
this.fireExpansionChanged(refreshed);
return refreshed;
}
return undefined;
}
async collapseNode(raw: ExpandableTreeNode): Promise<boolean> {
const node = this.tree.validateNode(raw);
return this.doCollapseNode(node);
}
async collapseAll(raw: CompositeTreeNode): Promise<boolean> {
const node = this.tree.validateNode(raw);
return this.doCollapseAll(node);
}
protected doCollapseAll(node: TreeNode | undefined): boolean {
let result = false;
if (CompositeTreeNode.is(node)) {
for (const child of node.children) {
result = this.doCollapseAll(child) || result;
}
}
return this.doCollapseNode(node) || result;
}
protected doCollapseNode(node: TreeNode | undefined): boolean {
if (!ExpandableTreeNode.isExpanded(node)) {
return false;
}
node.expanded = false;
this.fireExpansionChanged(node);
return true;
}
async toggleNodeExpansion(node: ExpandableTreeNode): Promise<void> {
if (node.expanded) {
await this.collapseNode(node);
} else {
await this.expandNode(node);
}
}
}