2024-06-04 22:05:44 +02:00
|
|
|
// Copyright 2024, Command Line Inc.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
2024-06-06 02:21:40 +02:00
|
|
|
import { WOS } from "@/app/store/global.js";
|
|
|
|
import { Atom, Getter, PrimitiveAtom, WritableAtom, atom, useAtom } from "jotai";
|
2024-06-04 22:05:44 +02:00
|
|
|
import { useCallback } from "react";
|
|
|
|
import { layoutTreeStateReducer, newLayoutTreeState } from "./layoutState.js";
|
|
|
|
import {
|
|
|
|
LayoutNode,
|
2024-06-06 02:21:40 +02:00
|
|
|
LayoutNodeWaveObj,
|
2024-06-04 22:05:44 +02:00
|
|
|
LayoutTreeAction,
|
|
|
|
LayoutTreeState,
|
|
|
|
WritableLayoutNodeAtom,
|
|
|
|
WritableLayoutTreeStateAtom,
|
|
|
|
} from "./model.js";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a new layout tree state wrapped as an atom.
|
|
|
|
* @param rootNode The root node for the tree.
|
|
|
|
* @returns The state wrapped as an atom.
|
|
|
|
*
|
|
|
|
* @template T The type of data associated with the nodes of the tree.
|
|
|
|
*/
|
|
|
|
export function newLayoutTreeStateAtom<T>(rootNode: LayoutNode<T>): PrimitiveAtom<LayoutTreeState<T>> {
|
|
|
|
return atom(newLayoutTreeState(rootNode)) as PrimitiveAtom<LayoutTreeState<T>>;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Derives a WritableLayoutTreeStateAtom from a WritableLayoutNodeAtom, initializing the tree state.
|
|
|
|
* @param layoutNodeAtom The atom containing the root node for the LayoutTreeState.
|
|
|
|
* @returns The derived WritableLayoutTreeStateAtom.
|
|
|
|
*/
|
|
|
|
export function withLayoutTreeState<T>(layoutNodeAtom: WritableLayoutNodeAtom<T>): WritableLayoutTreeStateAtom<T> {
|
2024-06-06 02:21:40 +02:00
|
|
|
const pendingActionAtom = atom<LayoutTreeAction>(null) as PrimitiveAtom<LayoutTreeAction>;
|
2024-06-04 22:05:44 +02:00
|
|
|
return atom(
|
|
|
|
(get) => {
|
2024-06-06 02:21:40 +02:00
|
|
|
const layoutState = newLayoutTreeState(get(layoutNodeAtom));
|
|
|
|
layoutState.pendingAction = get(pendingActionAtom);
|
|
|
|
return layoutState;
|
2024-06-04 22:05:44 +02:00
|
|
|
},
|
2024-06-06 02:21:40 +02:00
|
|
|
(_get, set, value) => {
|
|
|
|
set(pendingActionAtom, value.pendingAction);
|
|
|
|
set(layoutNodeAtom, value.rootNode);
|
2024-06-04 22:05:44 +02:00
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Hook to subscribe to the tree state and dispatch actions to its reducer functon.
|
|
|
|
* @param layoutTreeStateAtom The atom holding the layout tree state.
|
|
|
|
* @returns The current state of the tree and the dispatch function.
|
|
|
|
*/
|
|
|
|
export function useLayoutTreeStateReducerAtom<T>(
|
|
|
|
layoutTreeStateAtom: WritableLayoutTreeStateAtom<T>
|
|
|
|
): readonly [LayoutTreeState<T>, (action: LayoutTreeAction) => void] {
|
|
|
|
const [state, setState] = useAtom(layoutTreeStateAtom);
|
|
|
|
const dispatch = useCallback(
|
|
|
|
(action: LayoutTreeAction) => setState(layoutTreeStateReducer(state, action)),
|
|
|
|
[state, setState]
|
|
|
|
);
|
|
|
|
return [state, dispatch];
|
|
|
|
}
|
|
|
|
|
|
|
|
const tabLayoutAtomCache = new Map<string, WritableLayoutTreeStateAtom<TabLayoutData>>();
|
|
|
|
|
2024-06-06 02:21:40 +02:00
|
|
|
function getLayoutNodeWaveObjAtomFromTab<T>(
|
|
|
|
tabAtom: Atom<Tab>,
|
|
|
|
get: Getter
|
|
|
|
): WritableAtom<LayoutNodeWaveObj<T>, [value: LayoutNodeWaveObj<T>], void> {
|
|
|
|
const tabValue = get(tabAtom);
|
|
|
|
console.log("getLayoutNodeWaveObjAtomFromTab tabValue", tabValue);
|
|
|
|
if (!tabValue) return;
|
|
|
|
const layoutNodeOref = WOS.makeORef("layout", tabValue.layoutNode);
|
|
|
|
console.log("getLayoutNodeWaveObjAtomFromTab oref", layoutNodeOref);
|
|
|
|
return WOS.getWaveObjectAtom<LayoutNodeWaveObj<T>>(layoutNodeOref);
|
|
|
|
}
|
|
|
|
|
|
|
|
export function withLayoutNodeAtomFromTab<T>(tabAtom: Atom<Tab>): WritableLayoutNodeAtom<T> {
|
|
|
|
return atom(
|
|
|
|
(get) => {
|
|
|
|
console.log("get withLayoutNodeAtomFromTab", tabAtom);
|
|
|
|
const atom = getLayoutNodeWaveObjAtomFromTab<T>(tabAtom, get);
|
|
|
|
if (!atom) return null;
|
|
|
|
const retVal = get(atom)?.node;
|
|
|
|
console.log("get withLayoutNodeAtomFromTab end", retVal);
|
|
|
|
return get(atom)?.node;
|
|
|
|
},
|
|
|
|
(get, set, value) => {
|
|
|
|
console.log("set withLayoutNodeAtomFromTab", value);
|
|
|
|
const waveObjAtom = getLayoutNodeWaveObjAtomFromTab<T>(tabAtom, get);
|
|
|
|
if (!waveObjAtom) return;
|
|
|
|
const newWaveObjAtom = { ...get(waveObjAtom), node: value };
|
|
|
|
set(waveObjAtom, newWaveObjAtom);
|
|
|
|
}
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2024-06-04 22:05:44 +02:00
|
|
|
export function getLayoutStateAtomForTab(
|
|
|
|
tabId: string,
|
|
|
|
tabAtom: WritableAtom<Tab, [value: Tab], void>
|
|
|
|
): WritableLayoutTreeStateAtom<TabLayoutData> {
|
|
|
|
let atom = tabLayoutAtomCache.get(tabId);
|
|
|
|
if (atom) {
|
|
|
|
console.log("Reusing atom for tab", tabId);
|
|
|
|
return atom;
|
|
|
|
}
|
|
|
|
console.log("Creating new atom for tab", tabId);
|
2024-06-06 02:21:40 +02:00
|
|
|
atom = withLayoutTreeState(withLayoutNodeAtomFromTab<TabLayoutData>(tabAtom));
|
2024-06-04 22:05:44 +02:00
|
|
|
tabLayoutAtomCache.set(tabId, atom);
|
|
|
|
return atom;
|
|
|
|
}
|
|
|
|
|
|
|
|
export function deleteLayoutStateAtomForTab(tabId: string) {
|
|
|
|
const atom = tabLayoutAtomCache.get(tabId);
|
|
|
|
if (atom) {
|
|
|
|
tabLayoutAtomCache.delete(tabId);
|
|
|
|
}
|
|
|
|
}
|