mirror of
https://github.com/wavetermdev/waveterm.git
synced 2024-12-23 16:58:27 +01:00
e85b0d205e
This PR is a large refactoring of the layout code to move as much of the layout state logic as possible into a unified model class, with atoms and derived atoms to notify the display logic of changes. It also fixes some latent bugs in the node resize code, significantly speeds up response times for resizing and dragging, and sets us up to fully replace the React-DnD library in the future.
127 lines
4.6 KiB
TypeScript
127 lines
4.6 KiB
TypeScript
// Copyright 2024, Command Line Inc.
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
import type { Meta, StoryObj } from "@storybook/react";
|
|
|
|
import { TileLayout } from "./TileLayout.jsx";
|
|
|
|
import { useState } from "react";
|
|
import { newLayoutTreeStateAtom, useLayoutTreeStateReducerAtom } from "./layoutAtom.js";
|
|
import { newLayoutNode } from "./layoutNode.js";
|
|
import "./tilelayout.stories.less";
|
|
import { LayoutTreeActionType, LayoutTreeInsertNodeAction, WritableLayoutTreeStateAtom } from "./types.js";
|
|
|
|
interface TestData {
|
|
name: string;
|
|
}
|
|
|
|
const renderTestData = (data: TestData) => <div>{data.name}</div>;
|
|
|
|
const meta = {
|
|
title: "TileLayout",
|
|
args: {
|
|
layoutTreeStateAtom: newLayoutTreeStateAtom<TestData>(
|
|
newLayoutNode(undefined, undefined, undefined, {
|
|
name: "Hello world!",
|
|
})
|
|
),
|
|
contents: {
|
|
renderContent: (
|
|
data: TestData,
|
|
_ready: boolean,
|
|
_onClose: () => void,
|
|
dragHandleRef: React.RefObject<HTMLDivElement>
|
|
) => (
|
|
<div ref={dragHandleRef} className="test-content" style={{ width: "100%", height: "100%" }}>
|
|
{renderTestData(data)}
|
|
</div>
|
|
),
|
|
renderPreview: renderTestData,
|
|
tabId: "",
|
|
},
|
|
},
|
|
component: TileLayout<TestData>,
|
|
// This component will have an automatically generated Autodocs entry: https://storybook.js.org/docs/writing-docs/autodocs
|
|
tags: ["autodocs"],
|
|
} satisfies Meta<typeof TileLayout<TestData>>;
|
|
|
|
export default meta;
|
|
type Story = StoryObj<typeof meta>;
|
|
|
|
export const Basic: Story = {
|
|
args: {
|
|
layoutTreeStateAtom: newLayoutTreeStateAtom<TestData>(
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world!" })
|
|
),
|
|
},
|
|
};
|
|
|
|
export const More: Story = {
|
|
args: {
|
|
layoutTreeStateAtom: newLayoutTreeStateAtom<TestData>(
|
|
newLayoutNode(undefined, undefined, [
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world1!" }),
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world2!" }),
|
|
newLayoutNode(undefined, undefined, [
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world3!" }),
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world4!" }),
|
|
]),
|
|
])
|
|
),
|
|
},
|
|
};
|
|
|
|
const evenMoreRootNode = newLayoutNode<TestData>(undefined, undefined, [
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world1!" }),
|
|
newLayoutNode(undefined, undefined, [
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world2!" }),
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world3!" }),
|
|
]),
|
|
newLayoutNode(undefined, undefined, [
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world4!" }),
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world5!" }),
|
|
newLayoutNode(undefined, undefined, [
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world6!" }),
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world7!" }),
|
|
newLayoutNode(undefined, undefined, undefined, { name: "Hello world8!" }),
|
|
]),
|
|
]),
|
|
]);
|
|
|
|
export const EvenMore: Story = {
|
|
args: {
|
|
layoutTreeStateAtom: newLayoutTreeStateAtom<TestData>(evenMoreRootNode),
|
|
},
|
|
};
|
|
|
|
const addNodeAtom = newLayoutTreeStateAtom(evenMoreRootNode);
|
|
|
|
export const AddNode: Story = {
|
|
render: () => {
|
|
const [, dispatch] = useLayoutTreeStateReducerAtom(addNodeAtom);
|
|
const [numAddedNodes, setNumAddedNodes] = useState(0);
|
|
const dispatchAddNode = () => {
|
|
const newNode = newLayoutNode(undefined, undefined, undefined, {
|
|
name: "New Node" + numAddedNodes,
|
|
});
|
|
const insertNodeAction: LayoutTreeInsertNodeAction<TestData> = {
|
|
type: LayoutTreeActionType.InsertNode,
|
|
node: newNode,
|
|
};
|
|
dispatch(insertNodeAction);
|
|
setNumAddedNodes(numAddedNodes + 1);
|
|
};
|
|
return (
|
|
<div style={{ display: "flex", flexDirection: "column", width: "100%", height: "100%" }}>
|
|
<div>
|
|
<button onClick={dispatchAddNode}>Add node</button>
|
|
</div>
|
|
<TileLayout
|
|
layoutTreeStateAtom={addNodeAtom as WritableLayoutTreeStateAtom<TestData>}
|
|
contents={meta.args.contents}
|
|
/>
|
|
</div>
|
|
);
|
|
},
|
|
};
|