123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175 |
- /**
- * Copyright (c) 2023 mol* contributors, licensed under MIT, See LICENSE file for more info.
- *
- * @author Adam Midlik <midlik@gmail.com>
- */
- import { canonicalJsonString } from '../../../../mol-util/json';
- import { DefaultsForTree, Kind, SubTree, SubTreeOfKind, Tree, TreeFor, TreeSchema, TreeSchemaWithAllRequired, getParams } from './tree-schema';
- /** Run DFS (depth-first search) algorithm on a rooted tree.
- * Runs `visit` function when a node is discovered (before visiting any descendants).
- * Runs `postVisit` function when leaving a node (after all descendants have been visited). */
- export function dfs<TTree extends Tree>(root: TTree, visit?: (node: SubTree<TTree>, parent?: SubTree<TTree>) => any, postVisit?: (node: SubTree<TTree>, parent?: SubTree<TTree>) => any) {
- return _dfs<SubTree<TTree>>(root, undefined, visit, postVisit);
- }
- function _dfs<TTree extends Tree>(root: TTree, parent: SubTree<TTree> | undefined, visit?: (node: SubTree<TTree>, parent?: SubTree<TTree>) => any, postVisit?: (node: SubTree<TTree>, parent?: SubTree<TTree>) => any) {
- if (visit) visit(root, parent);
- for (const child of root.children ?? []) {
- _dfs<SubTree<TTree>>(child, root, visit, postVisit);
- }
- if (postVisit) postVisit(root, parent);
- }
- /** Convert a tree into a pretty-printed string. */
- export function treeToString(tree: Tree) {
- let level = 0;
- const lines: string[] = [];
- dfs(tree, node => lines.push(' '.repeat(level++) + `- ${node.kind} ${formatObject(node.params ?? {})}`), node => level--);
- return lines.join('\n');
- }
- /** Convert object to a human-friendly string (similar to JSON.stringify but without quoting keys) */
- export function formatObject(obj: {} | undefined): string {
- if (!obj) return 'undefined';
- return JSON.stringify(obj).replace(/,("\w+":)/g, ', $1').replace(/"(\w+)":/g, '$1: ');
- }
- /** Create a copy of a tree node, ignoring children. */
- export function copyNodeWithoutChildren<TTree extends Tree>(node: TTree): TTree {
- return {
- kind: node.kind,
- params: node.params ? { ...node.params } : undefined,
- } as TTree;
- }
- /** Create a copy of a tree node, including a shallow copy of children. */
- export function copyNode<TTree extends Tree>(node: TTree): TTree {
- return {
- kind: node.kind,
- params: node.params ? { ...node.params } : undefined,
- children: node.children ? [...node.children] : undefined,
- } as TTree;
- }
- /** Create a deep copy of a tree. */
- export function copyTree<T extends Tree>(root: T): T {
- return convertTree(root, {}) as T;
- }
- /** Set of rules for converting a tree of one schema into a different schema.
- * Each rule defines how to convert a node of a specific kind, e.g.
- * `{A: node => [], B: node => [{kind: 'X',...}], C: node => [{kind: 'Y',...}, {kind: 'Z',...}]}`:
- * nodes of kind `A` will be deleted (their children moved to parent),
- * nodes of kind `B` will be converted to kind `X`,
- * nodes of kind `C` will be converted to `Y` with a child `Z` (original children moved to `Z`),
- * nodes of other kinds will just be copied. */
- export type ConversionRules<A extends Tree, B extends Tree> = {
- [kind in Kind<SubTree<A>>]?: (node: SubTreeOfKind<A, kind>, parent?: SubTree<A>) => SubTree<B>[]
- };
- /** Apply a set of conversion rules to a tree to change to a different schema. */
- export function convertTree<A extends Tree, B extends Tree>(root: A, conversions: ConversionRules<A, B>): SubTree<B> {
- const mapping = new Map<SubTree<A>, SubTree<B>>();
- let convertedRoot: SubTree<B>;
- dfs<A>(root, (node, parent) => {
- const conversion = conversions[node.kind as (typeof node)['kind']] as ((n: typeof node, p?: SubTree<A>) => SubTree<B>[]) | undefined;
- if (conversion) {
- const convertidos = conversion(node, parent);
- if (!parent && convertidos.length === 0) throw new Error('Cannot convert root to empty path');
- let convParent = parent ? mapping.get(parent) : undefined;
- for (const conv of convertidos) {
- if (convParent) {
- (convParent.children ??= []).push(conv);
- } else {
- convertedRoot = conv;
- }
- convParent = conv;
- }
- mapping.set(node, convParent!);
- } else {
- const converted = copyNodeWithoutChildren(node);
- if (parent) {
- (mapping.get(parent)!.children ??= []).push(converted);
- } else {
- convertedRoot = converted;
- }
- mapping.set(node, converted);
- }
- });
- return convertedRoot!;
- }
- /** Create a copy of the tree where twins (siblings of the same kind with the same params) are merged into one node.
- * Applies only to the node kinds listed in `condenseNodes` (or all if undefined) except node kinds in `skipNodes`. */
- export function condenseTree<T extends Tree>(root: T, condenseNodes?: Set<Kind<Tree>>, skipNodes?: Set<Kind<Tree>>): T {
- const map = new Map<string, SubTree<T>>();
- const result = copyTree(root);
- dfs<T>(result, node => {
- map.clear();
- const newChildren: SubTree<T>[] = [];
- for (const child of node.children ?? []) {
- let twin: SubTree<T> | undefined = undefined;
- const doApply = (!condenseNodes || condenseNodes.has(child.kind)) && !skipNodes?.has(child.kind);
- if (doApply) {
- const key = child.kind + canonicalJsonString(getParams(child));
- twin = map.get(key);
- if (!twin) map.set(key, child);
- }
- if (twin) {
- (twin.children ??= []).push(...child.children ?? []);
- } else {
- newChildren.push(child as SubTree<T>);
- }
- }
- node.children = newChildren;
- });
- return result;
- }
- /** Create a copy of the tree where missing optional params for each node are added based on `defaults`. */
- export function addDefaults<S extends TreeSchema>(tree: TreeFor<S>, defaults: DefaultsForTree<S>): TreeFor<TreeSchemaWithAllRequired<S>> {
- const rules: ConversionRules<TreeFor<S>, TreeFor<S>> = {};
- for (const kind in defaults) {
- rules[kind] = node => [{ kind: node.kind, params: { ...defaults[kind], ...node.params } } as any];
- }
- return convertTree(tree, rules) as any;
- }
- /** Resolve any URI params in a tree, in place. URI params are those listed in `uriParamNames`.
- * Relative URIs are treated as relative to `baseUri`, which can in turn be relative to the window URL (if available). */
- export function resolveUris<T extends Tree>(tree: T, baseUri: string, uriParamNames: string[]): void {
- dfs(tree, node => {
- const params = node.params as Record<string, any> | undefined;
- if (!params) return;
- for (const name in params) {
- if (uriParamNames.includes(name)) {
- const uri = params[name];
- if (typeof uri === 'string') {
- params[name] = resolveUri(uri, baseUri, windowUrl());
- }
- }
- }
- });
- }
- /** Resolve a sequence of URI references (relative URIs), where each reference is either absolute or relative to the next one
- * (i.e. the last one is the base URI). Skip any `undefined`.
- * E.g. `resolveUri('./unexpected.png', '/spanish/inquisition/expectations.html', 'https://example.org/spam/spam/spam')`
- * returns `'https://example.org/spanish/inquisition/unexpected.png'`. */
- function resolveUri(...refs: (string | undefined)[]): string | undefined {
- let result: string | undefined = undefined;
- for (const ref of refs.reverse()) {
- if (ref !== undefined) {
- if (result === undefined) result = ref;
- else result = new URL(ref, result).href;
- }
- }
- return result;
- }
- /** Return URL of the current page when running in a browser; `undefined` when running in Node. */
- function windowUrl(): string | undefined {
- return (typeof window !== 'undefined') ? window.location.href : undefined;
- }
|