Files
plane/packages/propel/src/dialog/dialog.stories.tsx
chuan 8ebde8aa05
Some checks failed
Branch Build CE / Build Setup (push) Has been cancelled
Branch Build CE / Build-Push Admin Docker Image (push) Has been cancelled
Branch Build CE / Build-Push Web Docker Image (push) Has been cancelled
Branch Build CE / Build-Push Space Docker Image (push) Has been cancelled
Branch Build CE / Build-Push Live Collaboration Docker Image (push) Has been cancelled
Branch Build CE / Build-Push API Server Docker Image (push) Has been cancelled
Branch Build CE / Build-Push Proxy Docker Image (push) Has been cancelled
Branch Build CE / Build-Push AIO Docker Image (push) Has been cancelled
Branch Build CE / Upload Build Assets (push) Has been cancelled
Branch Build CE / Build Release (push) Has been cancelled
CodeQL / Analyze (javascript) (push) Has been cancelled
CodeQL / Analyze (python) (push) Has been cancelled
Codespell / Check for spelling errors (push) Has been cancelled
Sync Repositories / sync_changes (push) Has been cancelled
Initial commit: Plane
Synced from upstream: 8853637e981ed7d8a6cff32bd98e7afe20f54362
2025-11-07 00:00:52 +08:00

400 lines
13 KiB
TypeScript

import { useState } from "react";
import type { Meta, StoryObj } from "@storybook/react-vite";
import { useArgs } from "storybook/preview-api";
import { CloseIcon } from "../icons";
import { Dialog, EDialogWidth } from "./root";
const meta = {
title: "Components/Dialog",
component: Dialog,
subcomponents: {
DialogPanel: Dialog.Panel,
DialogTitle: Dialog.Title,
},
args: {
children: null,
open: false,
onOpenChange: () => {},
},
parameters: {
layout: "centered",
},
tags: ["autodocs"],
render(args) {
const [{ open }, updateArgs] = useArgs();
const setOpen = (value: boolean) => updateArgs({ open: value });
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-blue-500 px-4 py-2 text-white hover:bg-blue-600">
Open Dialog
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel>
<div className="p-6">
<Dialog.Title>Dialog Title</Dialog.Title>
<div className="mt-4">
<p className="text-sm text-gray-600">This is the dialog content. You can put any content here.</p>
</div>
<div className="mt-6 flex justify-end gap-2">
<button
onClick={() => setOpen(false)}
className="rounded bg-gray-200 px-4 py-2 text-sm hover:bg-gray-300"
>
Cancel
</button>
<button
onClick={() => setOpen(false)}
className="rounded bg-blue-500 px-4 py-2 text-sm text-white hover:bg-blue-600"
>
Confirm
</button>
</div>
</div>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
} satisfies Meta<typeof Dialog>;
export default meta;
type Story = StoryObj<typeof meta>;
export const Default: Story = {
args: {
children: null,
},
};
export const TopPosition: Story = {
render(args) {
const [open, setOpen] = useState(args.open);
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-blue-500 px-4 py-2 text-white hover:bg-blue-600">
Open Dialog (Top)
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel position="top">
<div className="p-6">
<Dialog.Title>Top Positioned Dialog</Dialog.Title>
<div className="mt-4">
<p className="text-sm text-gray-600">
This dialog appears at the top of the screen instead of centered.
</p>
</div>
<div className="mt-6 flex justify-end gap-2">
<button
onClick={() => setOpen(false)}
className="rounded bg-gray-200 px-4 py-2 text-sm hover:bg-gray-300"
>
Close
</button>
</div>
</div>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
};
export const SmallWidth: Story = {
render(args) {
const [open, setOpen] = useState(args.open);
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-blue-500 px-4 py-2 text-white hover:bg-blue-600">
Open Small Dialog
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel width={EDialogWidth.SM}>
<div className="p-6">
<Dialog.Title>Small Dialog</Dialog.Title>
<div className="mt-4">
<p className="text-sm text-gray-600">This is a small dialog.</p>
</div>
<div className="mt-6 flex justify-end">
<button
onClick={() => setOpen(false)}
className="rounded bg-blue-500 px-4 py-2 text-sm text-white hover:bg-blue-600"
>
Close
</button>
</div>
</div>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
};
export const LargeWidth: Story = {
render(args) {
const [open, setOpen] = useState(args.open);
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-blue-500 px-4 py-2 text-white hover:bg-blue-600">
Open Large Dialog
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel width={EDialogWidth.XXXXL}>
<div className="p-6">
<Dialog.Title>Large Dialog</Dialog.Title>
<div className="mt-4">
<p className="text-sm text-gray-600">
This is a large dialog with more horizontal space for content.
</p>
</div>
<div className="mt-6 flex justify-end">
<button
onClick={() => setOpen(false)}
className="rounded bg-blue-500 px-4 py-2 text-sm text-white hover:bg-blue-600"
>
Close
</button>
</div>
</div>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
};
export const WithCloseButton: Story = {
render(args) {
const [open, setOpen] = useState(args.open);
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-blue-500 px-4 py-2 text-white hover:bg-blue-600">
Open Dialog with Close Button
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel>
<div className="p-6">
<div className="flex items-start justify-between">
<Dialog.Title>Dialog with Close Button</Dialog.Title>
<button onClick={() => setOpen(false)} className="rounded-full p-1 hover:bg-gray-100">
<CloseIcon className="h-4 w-4" />
</button>
</div>
<div className="mt-4">
<p className="text-sm text-gray-600">This dialog has a close button in the header.</p>
</div>
</div>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
};
export const ConfirmationDialog: Story = {
render(args) {
const [open, setOpen] = useState(args.open);
const handleConfirm = () => {
alert("Confirmed!");
setOpen(false);
};
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-red-500 px-4 py-2 text-white hover:bg-red-600">
Delete Item
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel width={EDialogWidth.SM}>
<div className="p-6">
<Dialog.Title>Confirm Deletion</Dialog.Title>
<div className="mt-4">
<p className="text-sm text-gray-600">
Are you sure you want to delete this item? This action cannot be undone.
</p>
</div>
<div className="mt-6 flex justify-end gap-2">
<button
onClick={() => setOpen(false)}
className="rounded bg-gray-200 px-4 py-2 text-sm hover:bg-gray-300"
>
Cancel
</button>
<button
onClick={handleConfirm}
className="rounded bg-red-500 px-4 py-2 text-sm text-white hover:bg-red-600"
>
Delete
</button>
</div>
</div>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
};
export const FormDialog: Story = {
render(args) {
const [open, setOpen] = useState(args.open);
const handleSubmit = (e: React.FormEvent) => {
e.preventDefault();
alert("Form submitted!");
setOpen(false);
};
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-blue-500 px-4 py-2 text-white hover:bg-blue-600">
Open Form
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel width={EDialogWidth.MD}>
<form onSubmit={handleSubmit} className="p-6">
<Dialog.Title>Create New Item</Dialog.Title>
<div className="mt-4 space-y-4">
<div>
<label htmlFor="name" className="block text-sm font-medium text-gray-700">
Name
</label>
<input
type="text"
id="name"
className="mt-1 w-full rounded border border-gray-300 px-3 py-2 text-sm"
placeholder="Enter name"
/>
</div>
<div>
<label htmlFor="description" className="block text-sm font-medium text-gray-700">
Description
</label>
<textarea
id="description"
rows={3}
className="mt-1 w-full rounded border border-gray-300 px-3 py-2 text-sm"
placeholder="Enter description"
/>
</div>
</div>
<div className="mt-6 flex justify-end gap-2">
<button
type="button"
onClick={() => setOpen(false)}
className="rounded bg-gray-200 px-4 py-2 text-sm hover:bg-gray-300"
>
Cancel
</button>
<button type="submit" className="rounded bg-blue-500 px-4 py-2 text-sm text-white hover:bg-blue-600">
Create
</button>
</div>
</form>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
};
export const ScrollableContent: Story = {
render(args) {
const [open, setOpen] = useState(args.open);
return (
<>
<button onClick={() => setOpen(true)} className="rounded bg-blue-500 px-4 py-2 text-white hover:bg-blue-600">
Open Scrollable Dialog
</button>
{open && (
<Dialog {...args} open={open} onOpenChange={setOpen}>
<Dialog.Panel width={EDialogWidth.MD}>
<div className="p-6">
<Dialog.Title>Scrollable Content</Dialog.Title>
<div className="mt-4 max-h-96 overflow-y-auto">
{Array.from({ length: 20 }, (_, i) => (
<p key={i} className="mb-2 text-sm text-gray-600">
Lorem ipsum dolor sit amet, consectetur adipiscing elit. Sed do eiusmod tempor incididunt ut
labore et dolore magna aliqua.
</p>
))}
</div>
<div className="mt-6 flex justify-end">
<button
onClick={() => setOpen(false)}
className="rounded bg-blue-500 px-4 py-2 text-sm text-white hover:bg-blue-600"
>
Close
</button>
</div>
</div>
</Dialog.Panel>
</Dialog>
)}
</>
);
},
};
export const AllWidths: Story = {
render() {
const [openWidth, setOpenWidth] = useState<EDialogWidth | null>(null);
const widths = [
{ width: EDialogWidth.SM, label: "Small" },
{ width: EDialogWidth.MD, label: "Medium" },
{ width: EDialogWidth.LG, label: "Large" },
{ width: EDialogWidth.XL, label: "XL" },
{ width: EDialogWidth.XXL, label: "2XL" },
{ width: EDialogWidth.XXXL, label: "3XL" },
{ width: EDialogWidth.XXXXL, label: "4XL" },
];
return (
<div className="flex flex-wrap gap-2">
{widths.map(({ width, label }) => (
<button
key={width}
onClick={() => setOpenWidth(width)}
className="rounded bg-blue-500 px-4 py-2 text-sm text-white hover:bg-blue-600"
>
{label}
</button>
))}
{widths.map(({ width, label }) => (
<Dialog key={width} open={openWidth === width} onOpenChange={() => setOpenWidth(null)}>
<Dialog.Panel width={width}>
<div className="p-6">
<Dialog.Title>{label} Dialog</Dialog.Title>
<div className="mt-4">
<p className="text-sm text-gray-600">This dialog uses the {label} width variant.</p>
</div>
<div className="mt-6 flex justify-end">
<button
onClick={() => setOpenWidth(null)}
className="rounded bg-blue-500 px-4 py-2 text-sm text-white hover:bg-blue-600"
>
Close
</button>
</div>
</div>
</Dialog.Panel>
</Dialog>
))}
</div>
);
},
};