mirror of
https://github.com/wavetermdev/waveterm.git
synced 2025-01-06 19:18:22 +01:00
75c9e211d9
Adds resizability to the layout system. Hovering in the margins of a block will highlight the available resize handle and show a cursor indicating its resize direction. Dragging will cause the resizing nodes to blur out and be replaced by an outline. Releasing the handle will commit the new resize operation and cause the underlying nodes to update to their new sizes. We'll want to refactor this in the future to move all layout and resize logic into a shared model that the TileLayout code can talk to, but that's a future improvement. For now, this makes some compromises, mainly that the logic is kind of distributed around. --------- Co-authored-by: sawka <mike.sawka@gmail.com>
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 { LayoutTreeActionType, LayoutTreeInsertNodeAction, WritableLayoutTreeStateAtom } from "./model.js";
|
|
import "./tilelayout.stories.less";
|
|
|
|
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>
|
|
);
|
|
},
|
|
};
|