Break layout node into its own Wave Object (#21)

I am updating the layout node setup to write to its own wave object. 

The existing setup requires me to plumb the layout updates through every
time the tab gets updated, which produces a lot of annoying and
unintuitive design patterns. With this new setup, the tab object doesn't
get written to when the layout changes, only the layout object will get
written to. This prevents collisions when both the tab object and the
layout node object are getting updated, such as when a new block is
added or deleted.
This commit is contained in:
Evan Simkowitz 2024-06-05 17:21:40 -07:00 committed by GitHub
parent 28cef5f22f
commit f12e246c15
No known key found for this signature in database
GPG Key ID: B5690EEEBB952194
16 changed files with 149 additions and 84 deletions

View File

@ -5,4 +5,3 @@ DROP TABLE db_workspace;
DROP TABLE db_tab; DROP TABLE db_tab;
DROP TABLE db_block; DROP TABLE db_block;

View File

@ -27,3 +27,4 @@ CREATE TABLE db_block (
version int NOT NULL, version int NOT NULL,
data json NOT NULL data json NOT NULL
); );

View File

@ -0,0 +1 @@
DROP TABLE db_layout;

View File

@ -0,0 +1,5 @@
CREATE TABLE db_layout (
oid varchar(36) PRIMARY KEY,
version int NOT NULL,
data json NOT NULL
);

View File

@ -3,7 +3,6 @@
// WaveObjectStore // WaveObjectStore
import { LayoutNode } from "@/faraday/index";
import { Call as $Call, Events } from "@wailsio/runtime"; import { Call as $Call, Events } from "@wailsio/runtime";
import * as jotai from "jotai"; import * as jotai from "jotai";
import * as React from "react"; import * as React from "react";
@ -297,7 +296,7 @@ function getObjectValue<T>(oref: string, getFn?: jotai.Getter): T {
// sets the value of a WaveObject in the cache. // sets the value of a WaveObject in the cache.
// should provide setFn if it is available (e.g. inside of a jotai atom) // should provide setFn if it is available (e.g. inside of a jotai atom)
// otherwise it will use the globalStore.set function // otherwise it will use the globalStore.set function
function setObjectValue<T>(value: WaveObj, setFn?: jotai.Setter, pushToServer?: boolean) { function setObjectValue<T extends WaveObj>(value: T, setFn?: jotai.Setter, pushToServer?: boolean) {
const oref = makeORef(value.otype, value.oid); const oref = makeORef(value.otype, value.oid);
let wov = waveObjectValueCache.get(oref); let wov = waveObjectValueCache.get(oref);
if (wov == null) { if (wov == null) {
@ -309,7 +308,9 @@ function setObjectValue<T>(value: WaveObj, setFn?: jotai.Setter, pushToServer?:
} }
console.log("Setting", oref, "to", value); console.log("Setting", oref, "to", value);
setFn(wov.dataAtom, { value: value, loading: false }); setFn(wov.dataAtom, { value: value, loading: false });
console.log("Setting", oref, "to", value, "done");
if (pushToServer) { if (pushToServer) {
console.log("pushToServer", oref, value);
UpdateObject(value, false); UpdateObject(value, false);
} }
} }
@ -326,8 +327,8 @@ export function CreateBlock(blockDef: BlockDef, rtOpts: RuntimeOpts): Promise<{
return wrapObjectServiceCall("CreateBlock", blockDef, rtOpts); return wrapObjectServiceCall("CreateBlock", blockDef, rtOpts);
} }
export function DeleteBlock(blockId: string, newLayout?: LayoutNode<any>): Promise<void> { export function DeleteBlock(blockId: string): Promise<void> {
return wrapObjectServiceCall("DeleteBlock", blockId, newLayout); return wrapObjectServiceCall("DeleteBlock", blockId);
} }
export function CloseTab(tabId: string): Promise<void> { export function CloseTab(tabId: string): Promise<void> {
@ -339,9 +340,9 @@ export function UpdateObjectMeta(blockId: string, meta: MetadataType): Promise<v
} }
export function UpdateObject(waveObj: WaveObj, returnUpdates: boolean): Promise<WaveObjUpdate[]> { export function UpdateObject(waveObj: WaveObj, returnUpdates: boolean): Promise<WaveObjUpdate[]> {
console.log("UpdateObject", waveObj, returnUpdates);
return wrapObjectServiceCall("UpdateObject", waveObj, returnUpdates); return wrapObjectServiceCall("UpdateObject", waveObj, returnUpdates);
} }
export { export {
cleanWaveObjectCache, cleanWaveObjectCache,
clearWaveObjectCache, clearWaveObjectCache,

View File

@ -27,13 +27,10 @@ const TabContent = ({ tabId }: { tabId: string }) => {
return <Block blockId={tabData.blockId} onClose={onClose} />; return <Block blockId={tabData.blockId} onClose={onClose} />;
}, []); }, []);
const onNodeDelete = useCallback( const onNodeDelete = useCallback((data: TabLayoutData) => {
(data: TabLayoutData) => { console.log("onNodeDelete", data);
console.log("onNodeDelete", data, tabData); return WOS.DeleteBlock(data.blockId);
WOS.DeleteBlock(data.blockId, tabData.layout); }, []);
},
[tabData]
);
if (tabLoading) { if (tabLoading) {
return <CenteredLoadingDiv />; return <CenteredLoadingDiv />;

View File

@ -77,7 +77,7 @@ function Widgets() {
}; };
dispatchLayoutStateAction(insertNodeAction); dispatchLayoutStateAction(insertNodeAction);
}, },
[activeTabAtom] [activeTabAtom, dispatchLayoutStateAction]
); );
async function createBlock(blockDef: BlockDef) { async function createBlock(blockDef: BlockDef) {

View File

@ -22,7 +22,7 @@ import { setTransform as createTransform, debounce, determineDropDirection } fro
export interface TileLayoutProps<T> { export interface TileLayoutProps<T> {
layoutTreeStateAtom: WritableLayoutTreeStateAtom<T>; layoutTreeStateAtom: WritableLayoutTreeStateAtom<T>;
renderContent: ContentRenderer<T>; renderContent: ContentRenderer<T>;
onNodeDelete?: (data: T) => void; onNodeDelete?: (data: T) => Promise<void>;
className?: string; className?: string;
} }
@ -120,7 +120,7 @@ export const TileLayout = <T,>({ layoutTreeStateAtom, className, renderContent,
); );
} }
}, 30), }, 30),
[activeDrag, overlayContainerRef, displayContainerRef, layoutTreeState, nodeRefs] [activeDrag, overlayContainerRef, displayContainerRef, layoutTreeState.leafs, nodeRefs]
); );
// Update the transforms whenever we drag something and whenever the layout updates. // Update the transforms whenever we drag something and whenever the layout updates.
@ -133,6 +133,7 @@ export const TileLayout = <T,>({ layoutTreeStateAtom, className, renderContent,
// reattach the new callback. // reattach the new callback.
const [prevUpdateTransforms, setPrevUpdateTransforms] = useState<() => void>(undefined); const [prevUpdateTransforms, setPrevUpdateTransforms] = useState<() => void>(undefined);
useEffect(() => { useEffect(() => {
console.log("replace resize listener");
if (prevUpdateTransforms) window.removeEventListener("resize", prevUpdateTransforms); if (prevUpdateTransforms) window.removeEventListener("resize", prevUpdateTransforms);
window.addEventListener("resize", updateTransforms); window.addEventListener("resize", updateTransforms);
setPrevUpdateTransforms(updateTransforms); setPrevUpdateTransforms(updateTransforms);
@ -156,7 +157,7 @@ export const TileLayout = <T,>({ layoutTreeStateAtom, className, renderContent,
}, []); }, []);
const onLeafClose = useCallback( const onLeafClose = useCallback(
(node: LayoutNode<T>) => { async (node: LayoutNode<T>) => {
console.log("onLeafClose", node); console.log("onLeafClose", node);
const deleteAction: LayoutTreeDeleteNodeAction = { const deleteAction: LayoutTreeDeleteNodeAction = {
type: LayoutTreeActionType.DeleteNode, type: LayoutTreeActionType.DeleteNode,
@ -165,7 +166,7 @@ export const TileLayout = <T,>({ layoutTreeStateAtom, className, renderContent,
console.log("calling dispatch", deleteAction); console.log("calling dispatch", deleteAction);
dispatch(deleteAction); dispatch(deleteAction);
console.log("calling onNodeDelete", node); console.log("calling onNodeDelete", node);
onNodeDelete?.(node.data); await onNodeDelete?.(node.data);
console.log("node deleted"); console.log("node deleted");
}, },
[onNodeDelete, dispatch] [onNodeDelete, dispatch]

View File

@ -1,11 +1,13 @@
// Copyright 2024, Command Line Inc. // Copyright 2024, Command Line Inc.
// SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: Apache-2.0
import { PrimitiveAtom, WritableAtom, atom, useAtom } from "jotai"; import { WOS } from "@/app/store/global.js";
import { Atom, Getter, PrimitiveAtom, WritableAtom, atom, useAtom } from "jotai";
import { useCallback } from "react"; import { useCallback } from "react";
import { layoutTreeStateReducer, newLayoutTreeState } from "./layoutState.js"; import { layoutTreeStateReducer, newLayoutTreeState } from "./layoutState.js";
import { import {
LayoutNode, LayoutNode,
LayoutNodeWaveObj,
LayoutTreeAction, LayoutTreeAction,
LayoutTreeState, LayoutTreeState,
WritableLayoutNodeAtom, WritableLayoutNodeAtom,
@ -29,27 +31,16 @@ export function newLayoutTreeStateAtom<T>(rootNode: LayoutNode<T>): PrimitiveAto
* @returns The derived WritableLayoutTreeStateAtom. * @returns The derived WritableLayoutTreeStateAtom.
*/ */
export function withLayoutTreeState<T>(layoutNodeAtom: WritableLayoutNodeAtom<T>): WritableLayoutTreeStateAtom<T> { export function withLayoutTreeState<T>(layoutNodeAtom: WritableLayoutNodeAtom<T>): WritableLayoutTreeStateAtom<T> {
return atom( const pendingActionAtom = atom<LayoutTreeAction>(null) as PrimitiveAtom<LayoutTreeAction>;
(get) => newLayoutTreeState(get(layoutNodeAtom)),
(_get, set, value) => set(layoutNodeAtom, value.rootNode)
);
}
export function withLayoutStateFromTab(
tabAtom: WritableAtom<Tab, [value: Tab], void>
): WritableLayoutTreeStateAtom<TabLayoutData> {
return atom( return atom(
(get) => { (get) => {
const tabData = get(tabAtom); const layoutState = newLayoutTreeState(get(layoutNodeAtom));
console.log("get layout state from tab", tabData); layoutState.pendingAction = get(pendingActionAtom);
return newLayoutTreeState(tabData?.layout); return layoutState;
}, },
(get, set, value) => { (_get, set, value) => {
const tabValue = get(tabAtom); set(pendingActionAtom, value.pendingAction);
const newTabValue = { ...tabValue }; set(layoutNodeAtom, value.rootNode);
newTabValue.layout = value.rootNode;
console.log("set tab", tabValue, value);
set(tabAtom, newTabValue);
} }
); );
} }
@ -72,6 +63,38 @@ export function useLayoutTreeStateReducerAtom<T>(
const tabLayoutAtomCache = new Map<string, WritableLayoutTreeStateAtom<TabLayoutData>>(); const tabLayoutAtomCache = new Map<string, WritableLayoutTreeStateAtom<TabLayoutData>>();
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);
}
);
}
export function getLayoutStateAtomForTab( export function getLayoutStateAtomForTab(
tabId: string, tabId: string,
tabAtom: WritableAtom<Tab, [value: Tab], void> tabAtom: WritableAtom<Tab, [value: Tab], void>
@ -82,7 +105,7 @@ export function getLayoutStateAtomForTab(
return atom; return atom;
} }
console.log("Creating new atom for tab", tabId); console.log("Creating new atom for tab", tabId);
atom = withLayoutStateFromTab(tabAtom); atom = withLayoutTreeState(withLayoutNodeAtomFromTab<TabLayoutData>(tabAtom));
tabLayoutAtomCache.set(tabId, atom); tabLayoutAtomCache.set(tabId, atom);
return atom; return atom;
} }

View File

@ -131,3 +131,7 @@ export type WritableLayoutNodeAtom<T> = WritableAtom<LayoutNode<T>, [value: Layo
export type WritableLayoutTreeStateAtom<T> = WritableAtom<LayoutTreeState<T>, [value: LayoutTreeState<T>], void>; export type WritableLayoutTreeStateAtom<T> = WritableAtom<LayoutTreeState<T>, [value: LayoutTreeState<T>], void>;
export type ContentRenderer<T> = (data: T, ready: boolean, onClose?: () => void) => React.ReactNode; export type ContentRenderer<T> = (data: T, ready: boolean, onClose?: () => void) => React.ReactNode;
export interface LayoutNodeWaveObj<T> extends WaveObj {
node: LayoutNode<T>;
}

View File

@ -1,8 +1,6 @@
// Copyright 2024, Command Line Inc. // Copyright 2024, Command Line Inc.
// SPDX-License-Identifier: Apache-2.0 // SPDX-License-Identifier: Apache-2.0
import { LayoutNode } from "../faraday";
declare global { declare global {
type UIContext = { type UIContext = {
windowid: string; windowid: string;
@ -70,7 +68,7 @@ declare global {
version: number; version: number;
name: string; name: string;
blockids: string[]; blockids: string[];
layout: LayoutNode<TabLayoutData>; layoutNode: string;
}; };
type Point = { type Point = {

View File

@ -79,7 +79,7 @@ func TestCreate(t *testing.T) {
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
err := WFS.MakeFile(ctx, zoneId, "testfile", nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, "testfile", nil, FileOptsType{})
if err != nil { if err != nil {
t.Fatalf("error creating file: %v", err) t.Fatalf("error creating file: %v", err)
@ -153,7 +153,7 @@ func TestDelete(t *testing.T) {
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
err := WFS.MakeFile(ctx, zoneId, "testfile", nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, "testfile", nil, FileOptsType{})
if err != nil { if err != nil {
t.Fatalf("error creating file: %v", err) t.Fatalf("error creating file: %v", err)
@ -216,7 +216,7 @@ func TestSetMeta(t *testing.T) {
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
err := WFS.MakeFile(ctx, zoneId, "testfile", nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, "testfile", nil, FileOptsType{})
if err != nil { if err != nil {
t.Fatalf("error creating file: %v", err) t.Fatalf("error creating file: %v", err)
@ -319,7 +319,7 @@ func TestAppend(t *testing.T) {
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
fileName := "t2" fileName := "t2"
err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{})
if err != nil { if err != nil {
@ -347,7 +347,7 @@ func TestWriteFile(t *testing.T) {
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
fileName := "t3" fileName := "t3"
err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{})
if err != nil { if err != nil {
@ -391,7 +391,7 @@ func TestCircularWrites(t *testing.T) {
defer cleanupDb(t) defer cleanupDb(t)
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
err := WFS.MakeFile(ctx, zoneId, "c1", nil, FileOptsType{Circular: true, MaxSize: 50}) err := WFS.MakeFile(ctx, zoneId, "c1", nil, FileOptsType{Circular: true, MaxSize: 50})
if err != nil { if err != nil {
t.Fatalf("error creating file: %v", err) t.Fatalf("error creating file: %v", err)
@ -478,7 +478,7 @@ func TestMultiPart(t *testing.T) {
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
fileName := "m2" fileName := "m2"
data := makeText(80) data := makeText(80)
err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{})
@ -554,7 +554,7 @@ func TestSimpleDBFlush(t *testing.T) {
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
fileName := "t1" fileName := "t1"
err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{})
if err != nil { if err != nil {
@ -586,7 +586,7 @@ func TestConcurrentAppend(t *testing.T) {
defer cleanupDb(t) defer cleanupDb(t)
ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancelFn := context.WithTimeout(context.Background(), 5*time.Second)
defer cancelFn() defer cancelFn()
zoneId := uuid.New().String() zoneId := uuid.NewString()
fileName := "t1" fileName := "t1"
err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{}) err := WFS.MakeFile(ctx, zoneId, fileName, nil, FileOptsType{})
if err != nil { if err != nil {

View File

@ -147,11 +147,11 @@ func (svc *ObjectService) CreateBlock(uiContext wstore.UIContext, blockDef *wsto
return updatesRtn(ctx, rtn) return updatesRtn(ctx, rtn)
} }
func (svc *ObjectService) DeleteBlock(uiContext wstore.UIContext, blockId string, newLayout any) (any, error) { func (svc *ObjectService) DeleteBlock(uiContext wstore.UIContext, blockId string) (any, error) {
ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout) ctx, cancelFn := context.WithTimeout(context.Background(), DefaultTimeout)
defer cancelFn() defer cancelFn()
ctx = wstore.ContextWithUpdates(ctx) ctx = wstore.ContextWithUpdates(ctx)
err := wstore.DeleteBlock(ctx, uiContext.ActiveTabId, blockId, newLayout) err := wstore.DeleteBlock(ctx, uiContext.ActiveTabId, blockId)
if err != nil { if err != nil {
return nil, fmt.Errorf("error deleting block: %w", err) return nil, fmt.Errorf("error deleting block: %w", err)
} }

View File

@ -199,7 +199,7 @@ func (c *RpcClient) removeReqInfo(rpcId string, clearSend bool) {
} }
func (c *RpcClient) SimpleReq(ctx context.Context, command string, data any) (any, error) { func (c *RpcClient) SimpleReq(ctx context.Context, command string, data any) (any, error) {
rpcId := uuid.New().String() rpcId := uuid.NewString()
seqNum := c.NextSeqNum.Add(1) seqNum := c.NextSeqNum.Add(1)
var timeoutInfo *TimeoutInfo var timeoutInfo *TimeoutInfo
deadline, ok := ctx.Deadline() deadline, ok := ctx.Deadline()
@ -235,7 +235,7 @@ func (c *RpcClient) SimpleReq(ctx context.Context, command string, data any) (an
} }
func (c *RpcClient) StreamReq(ctx context.Context, command string, data any, respTimeout time.Duration) (chan *RpcPacket, error) { func (c *RpcClient) StreamReq(ctx context.Context, command string, data any, respTimeout time.Duration) (chan *RpcPacket, error) {
rpcId := uuid.New().String() rpcId := uuid.NewString()
seqNum := c.NextSeqNum.Add(1) seqNum := c.NextSeqNum.Add(1)
var timeoutInfo *TimeoutInfo = &TimeoutInfo{RespPacketTimeout: respTimeout.Milliseconds()} var timeoutInfo *TimeoutInfo = &TimeoutInfo{RespPacketTimeout: respTimeout.Milliseconds()}
deadline, ok := ctx.Deadline() deadline, ok := ctx.Deadline()

View File

@ -140,13 +140,19 @@ func CreateTab(ctx context.Context, workspaceId string, name string) (*Tab, erro
if ws == nil { if ws == nil {
return nil, fmt.Errorf("workspace not found: %q", workspaceId) return nil, fmt.Errorf("workspace not found: %q", workspaceId)
} }
layoutNodeId := uuid.NewString()
tab := &Tab{ tab := &Tab{
OID: uuid.New().String(), OID: uuid.NewString(),
Name: name, Name: name,
BlockIds: []string{}, BlockIds: []string{},
LayoutNode: layoutNodeId,
}
layoutNode := &LayoutNode{
OID: layoutNodeId,
} }
ws.TabIds = append(ws.TabIds, tab.OID) ws.TabIds = append(ws.TabIds, tab.OID)
DBInsert(tx.Context(), tab) DBInsert(tx.Context(), tab)
DBInsert(tx.Context(), layoutNode)
DBUpdate(tx.Context(), ws) DBUpdate(tx.Context(), ws)
return tab, nil return tab, nil
}) })
@ -154,7 +160,7 @@ func CreateTab(ctx context.Context, workspaceId string, name string) (*Tab, erro
func CreateWorkspace(ctx context.Context) (*Workspace, error) { func CreateWorkspace(ctx context.Context) (*Workspace, error) {
ws := &Workspace{ ws := &Workspace{
OID: uuid.New().String(), OID: uuid.NewString(),
TabIds: []string{}, TabIds: []string{},
} }
DBInsert(ctx, ws) DBInsert(ctx, ws)
@ -185,7 +191,7 @@ func CreateBlock(ctx context.Context, tabId string, blockDef *BlockDef, rtOpts *
if tab == nil { if tab == nil {
return nil, fmt.Errorf("tab not found: %q", tabId) return nil, fmt.Errorf("tab not found: %q", tabId)
} }
blockId := uuid.New().String() blockId := uuid.NewString()
blockData := &Block{ blockData := &Block{
OID: blockId, OID: blockId,
BlockDef: blockDef, BlockDef: blockDef,
@ -210,7 +216,7 @@ func findStringInSlice(slice []string, val string) int {
return -1 return -1
} }
func DeleteBlock(ctx context.Context, tabId string, blockId string, newLayout any) error { func DeleteBlock(ctx context.Context, tabId string, blockId string) error {
return WithTx(ctx, func(tx *TxWrap) error { return WithTx(ctx, func(tx *TxWrap) error {
tab, _ := DBGet[*Tab](tx.Context(), tabId) tab, _ := DBGet[*Tab](tx.Context(), tabId)
if tab == nil { if tab == nil {
@ -221,11 +227,8 @@ func DeleteBlock(ctx context.Context, tabId string, blockId string, newLayout an
return nil return nil
} }
tab.BlockIds = append(tab.BlockIds[:blockIdx], tab.BlockIds[blockIdx+1:]...) tab.BlockIds = append(tab.BlockIds[:blockIdx], tab.BlockIds[blockIdx+1:]...)
if newLayout != nil {
tab.Layout = newLayout
}
DBUpdate(tx.Context(), tab) DBUpdate(tx.Context(), tab)
DBDelete(tx.Context(), "block", blockId) DBDelete(tx.Context(), OType_Block, blockId)
return nil return nil
}) })
} }
@ -246,9 +249,10 @@ func CloseTab(ctx context.Context, workspaceId string, tabId string) error {
} }
ws.TabIds = append(ws.TabIds[:tabIdx], ws.TabIds[tabIdx+1:]...) ws.TabIds = append(ws.TabIds[:tabIdx], ws.TabIds[tabIdx+1:]...)
DBUpdate(tx.Context(), ws) DBUpdate(tx.Context(), ws)
DBDelete(tx.Context(), "tab", tabId) DBDelete(tx.Context(), OType_Tab, tabId)
DBDelete(tx.Context(), OType_LayoutNode, tab.LayoutNode)
for _, blockId := range tab.BlockIds { for _, blockId := range tab.BlockIds {
DBDelete(tx.Context(), "block", blockId) DBDelete(tx.Context(), OType_Block, blockId)
} }
return nil return nil
}) })
@ -300,11 +304,12 @@ func EnsureInitialData() error {
if clientCount > 0 { if clientCount > 0 {
return nil return nil
} }
windowId := uuid.New().String() windowId := uuid.NewString()
workspaceId := uuid.New().String() workspaceId := uuid.NewString()
tabId := uuid.New().String() tabId := uuid.NewString()
layoutNodeId := uuid.NewString()
client := &Client{ client := &Client{
OID: uuid.New().String(), OID: uuid.NewString(),
MainWindowId: windowId, MainWindowId: windowId,
} }
err = DBInsert(ctx, client) err = DBInsert(ctx, client)
@ -342,10 +347,19 @@ func EnsureInitialData() error {
OID: tabId, OID: tabId,
Name: "Tab-1", Name: "Tab-1",
BlockIds: []string{}, BlockIds: []string{},
LayoutNode: layoutNodeId,
} }
err = DBInsert(ctx, tab) err = DBInsert(ctx, tab)
if err != nil { if err != nil {
return fmt.Errorf("error inserting tab: %w", err) return fmt.Errorf("error inserting tab: %w", err)
} }
layoutNode := &LayoutNode{
OID: layoutNodeId,
}
err = DBInsert(ctx, layoutNode)
if err != nil {
return fmt.Errorf("error inserting layout node: %w", err)
}
return nil return nil
} }

View File

@ -21,6 +21,15 @@ const (
UpdateType_Delete = "delete" UpdateType_Delete = "delete"
) )
const (
OType_Client = "client"
OType_Window = "window"
OType_Workspace = "workspace"
OType_Tab = "tab"
OType_LayoutNode = "layout"
OType_Block = "block"
)
type WaveObjUpdate struct { type WaveObjUpdate struct {
UpdateType string `json:"updatetype"` UpdateType string `json:"updatetype"`
OType string `json:"otype"` OType string `json:"otype"`
@ -51,7 +60,7 @@ type Client struct {
} }
func (*Client) GetOType() string { func (*Client) GetOType() string {
return "client" return OType_Client
} }
// stores the ui-context of the window // stores the ui-context of the window
@ -69,7 +78,7 @@ type Window struct {
} }
func (*Window) GetOType() string { func (*Window) GetOType() string {
return "window" return OType_Window
} }
type Workspace struct { type Workspace struct {
@ -81,20 +90,31 @@ type Workspace struct {
} }
func (*Workspace) GetOType() string { func (*Workspace) GetOType() string {
return "workspace" return OType_Workspace
} }
type Tab struct { type Tab struct {
OID string `json:"oid"` OID string `json:"oid"`
Version int `json:"version"` Version int `json:"version"`
Name string `json:"name"` Name string `json:"name"`
Layout any `json:"layout,omitempty"` LayoutNode string `json:"layoutNode"`
BlockIds []string `json:"blockids"` BlockIds []string `json:"blockids"`
Meta map[string]any `json:"meta"` Meta map[string]any `json:"meta"`
} }
func (*Tab) GetOType() string { func (*Tab) GetOType() string {
return "tab" return OType_Tab
}
type LayoutNode struct {
OID string `json:"oid"`
Version int `json:"version"`
Node any `json:"node,omitempty"`
Meta map[string]any `json:"meta,omitempty"`
}
func (*LayoutNode) GetOType() string {
return OType_LayoutNode
} }
type FileDef struct { type FileDef struct {
@ -138,7 +158,7 @@ type Block struct {
} }
func (*Block) GetOType() string { func (*Block) GetOType() string {
return "block" return OType_Block
} }
func AllWaveObjTypes() []reflect.Type { func AllWaveObjTypes() []reflect.Type {
@ -148,5 +168,6 @@ func AllWaveObjTypes() []reflect.Type {
reflect.TypeOf(&Workspace{}), reflect.TypeOf(&Workspace{}),
reflect.TypeOf(&Tab{}), reflect.TypeOf(&Tab{}),
reflect.TypeOf(&Block{}), reflect.TypeOf(&Block{}),
reflect.TypeOf(&LayoutNode{}),
} }
} }