// Copyright 2024, Command Line Inc. // SPDX-License-Identifier: Apache-2.0 import { DefaultNodeSize, LayoutNode } from "./model"; import { FlexDirection, getCrypto, reverseFlexDirection } from "./utils"; const crypto = getCrypto(); /** * Creates a new node. * @param flexDirection The flex direction for the new node. * @param size The size for the new node. * @param children The children for the new node. * @param data The data for the new node. * @template T The type of data associated with the node. * @returns The new node. */ export function newLayoutNode( flexDirection?: FlexDirection, size?: number, children?: LayoutNode[], data?: T ): LayoutNode { const newNode: LayoutNode = { id: crypto.randomUUID(), flexDirection: flexDirection ?? FlexDirection.Row, size: size ?? DefaultNodeSize, children, data, }; if (!validateNode(newNode)) { throw new Error("Invalid node"); } return newNode; } /** * Adds new nodes to the tree at the given index. * @param node The parent node. * @param idx The index to insert at. * @param children The nodes to insert. * @template T The type of data associated with the node. * @returns The updated parent node. */ export function addChildAt(node: LayoutNode, idx: number, ...children: LayoutNode[]) { console.log("adding", children, "to", node, "at index", idx); if (children.length === 0) return; if (!node.children) { addIntermediateNode(node); } const childrenToAdd = children.flatMap((v) => { if (v.flexDirection !== node.flexDirection) { return v; } else if (v.children) { return v.children; } else { v.flexDirection = reverseFlexDirection(node.flexDirection); return v; } }); if (node.children.length <= idx) { node.children.push(...childrenToAdd); } else if (idx >= 0) { node.children.splice(idx, 0, ...childrenToAdd); } } /** * Adds an intermediate node as a direct child of the given node, moving the given node's children or data into it. * * If the node contains children, they are moved two levels deeper to preserve their flex direction. If the node only has data, it is moved one level deeper. * @param node The node to add the intermediate node to. * @template T The type of data associated with the node. * @returns The updated node and the node that was added. */ export function addIntermediateNode(node: LayoutNode): LayoutNode { let intermediateNode: LayoutNode; console.log(node); if (node.data) { intermediateNode = newLayoutNode(reverseFlexDirection(node.flexDirection), undefined, undefined, node.data); node.children = [intermediateNode]; node.data = undefined; } else { const intermediateNodeInner = newLayoutNode(node.flexDirection, undefined, node.children); intermediateNode = newLayoutNode(reverseFlexDirection(node.flexDirection), undefined, [intermediateNodeInner]); node.children = [intermediateNode]; } const intermediateNodeId = intermediateNode.id; intermediateNode.id = node.id; node.id = intermediateNodeId; return intermediateNode; } /** * Attempts to remove the specified node from its parent. * @param parent The parent node. * @param childToRemove The node to remove. * @param startingIndex The index in children to start the search from. * @template T The type of data associated with the node. * @returns The updated parent node, or undefined if the node was not found. */ export function removeChild(parent: LayoutNode, childToRemove: LayoutNode, startingIndex: number = 0) { if (!parent.children) return; const idx = parent.children.indexOf(childToRemove, startingIndex); if (idx === -1) return; parent.children?.splice(idx, 1); } /** * Finds the node with the given id. * @param node The node to search in. * @param id The id to search for. * @template T The type of data associated with the node. * @returns The node with the given id or undefined if no node with the given id was found. */ export function findNode(node: LayoutNode, id: string): LayoutNode | undefined { if (node.id === id) return node; if (!node.children) return; for (const child of node.children) { const result = findNode(child, id); if (result) return result; } return; } /** * Finds the node whose children contains the node with the given id. * @param node The node to start the search from. * @param id The id to search for. * @template T The type of data associated with the node. * @returns The parent node, or undefined if no node with the given id was found. */ export function findParent(node: LayoutNode, id: string): LayoutNode | undefined { if (node.id === id || !node.children) return; for (const child of node.children) { if (child.id === id) return node; const retVal = findParent(child, id); if (retVal) return retVal; } return; } /** * Determines whether a node is valid. * @param node The node to validate. * @template T The type of data associated with the node. * @returns True if the node is valid, false otherwise. */ export function validateNode(node: LayoutNode): boolean { if (!node.children == !node.data) { console.error("Either children or data must be defined for node, not both"); return false; } if (node.children?.length === 0) { console.error("Node cannot define an empty array of children"); return false; } return true; } /** * Recursively corrects the tree to minimize nested single-child nodes, remove invalid nodes, and correct invalid flex direction order. * Also finds all leaf nodes under the specified node. * @param node The node to start the balancing from. * @template T The type of data associated with the node. * @returns The corrected node and an array of leaf nodes. */ export function balanceNode(node: LayoutNode): { node: LayoutNode; leafs: LayoutNode[] } | undefined { const leafs: LayoutNode[] = []; const newNode = balanceNodeHelper(node, leafs); return { node: newNode, leafs }; } function balanceNodeHelper(node: LayoutNode, leafs: LayoutNode[]): LayoutNode { if (!node) return; if (!node.children) { leafs.push(node); return node; } if (node.children.length == 0) return; if (!validateNode(node)) throw new Error("Invalid node"); node.children = node.children .flatMap((child) => { if (child.flexDirection === node.flexDirection) { child.flexDirection = reverseFlexDirection(node.flexDirection); } if (child.children?.length == 1 && child.children[0].children) { return child.children[0].children; } return child; }) .map((child) => { return balanceNodeHelper(child, leafs); }) .filter((v) => v); if (node.children.length == 1 && !node.children[0].children) { node.data = node.children[0].data; node.id = node.children[0].id; node.children = undefined; } return node; } /** * Finds the first node in the tree where a new node can be inserted. * * This will attempt to fill each node until it has maxChildren children. If a node is full, it will move to its children and * fill each of them until it has maxChildren children. It will ensure that each child fills evenly before moving to the next * layer down. * * @param node The node to start the search from. * @param maxChildren The maximum number of children a node can have. * @returns The node to insert into and the index at which to insert. */ export function findNextInsertLocation( node: LayoutNode, maxChildren: number ): { node: LayoutNode; index: number } { const insertLoc = findNextInsertLocationHelper(node, maxChildren, 1); return { node: insertLoc?.node, index: insertLoc?.index }; } function findNextInsertLocationHelper( node: LayoutNode, maxChildren: number, curDepth: number = 1 ): { node: LayoutNode; index: number; depth: number } { if (!node) return; if (!node.children) return { node, index: 1, depth: curDepth }; let insertLocs: { node: LayoutNode; index: number; depth: number }[] = []; if (node.children.length < maxChildren) { insertLocs.push({ node, index: node.children.length, depth: curDepth }); } for (const child of node.children.slice().reverse()) { insertLocs.push(findNextInsertLocationHelper(child, maxChildren, curDepth + 1)); } insertLocs = insertLocs .filter((a) => a) .sort((a, b) => Math.pow(a.depth, a.index + maxChildren) - Math.pow(b.depth, b.index + maxChildren)); return insertLocs[0]; }