feat: add Popover docs
This commit is contained in:
parent
a51d7eead1
commit
fdb4d0d70f
117
packages/react/src/docs/components/Popover.mdx
Normal file
117
packages/react/src/docs/components/Popover.mdx
Normal file
@ -0,0 +1,117 @@
|
||||
import { TabProvider, TabTrigger, TabContent, TabList } from "@components/Tabs";
|
||||
import { Story } from "@/components/Story";
|
||||
import { LoadedCode, GITHUB } from "@/components/LoadedCode";
|
||||
import { PopoverDemo } from "./PopoverBlocks/Preview";
|
||||
import Examples from "./PopoverBlocks/Examples";
|
||||
|
||||
# Popover
|
||||
Displays rich content in a portal, triggered by a button.
|
||||
|
||||
<TabProvider defaultName="preview">
|
||||
<TabList>
|
||||
<TabTrigger name="preview">Preview</TabTrigger>
|
||||
<TabTrigger name="code">Code</TabTrigger>
|
||||
</TabList>
|
||||
<TabContent name="preview">
|
||||
<Story layout="centered">
|
||||
<PopoverDemo />
|
||||
</Story>
|
||||
</TabContent>
|
||||
<TabContent name="code">
|
||||
<LoadedCode from={`${GITHUB}/packages/react/src/docs/components/PopoverBlocks/Preview.tsx`} />
|
||||
</TabContent>
|
||||
</TabProvider>
|
||||
|
||||
## Installation
|
||||
|
||||
1. Create a new file `Popover.tsx` in your component folder.
|
||||
2. Copy and paste the following code into the file.
|
||||
|
||||
<LoadedCode from={`${GITHUB}/packages/react/components/Popover.tsx`} />
|
||||
|
||||
## Usage
|
||||
|
||||
```tsx
|
||||
import { Popover, PopoverTrigger, PopoverContent } from "@components/popover"
|
||||
```
|
||||
|
||||
```html
|
||||
<Popover>
|
||||
<PopoverTrigger>
|
||||
<Button />
|
||||
</PopoverTrigger>
|
||||
<PopoverContent>
|
||||
{/* content of popover */}
|
||||
</PopoverContent>
|
||||
</Popover
|
||||
```
|
||||
|
||||
> Note:
|
||||
>
|
||||
> PopoverTrigger will merge its onClick event handler to its children.
|
||||
> Also, there is no default element for those.
|
||||
> So you always have to provide the clickable children for PopoverTrigger.
|
||||
>
|
||||
> It is easier to understand if you think of this component as always having the `asChild` prop applied to it.
|
||||
|
||||
## Props
|
||||
|
||||
### Popover
|
||||
|
||||
#### Special
|
||||
|
||||
| Prop | Type | Default | Description |
|
||||
|:----------|:----------|:--------|:------------------------------------------------------------------|
|
||||
| `opened` | `boolean` | `false` | Initial open state |
|
||||
| `asChild` | `boolean` | `false` | Whether the root of popover is rendered as a child of a component |
|
||||
|
||||
### PopoverContent
|
||||
|
||||
#### Variants
|
||||
|
||||
| Prop | Type | Default | Description |
|
||||
|:---------|:-------------------------------------------------------------------------|:-----------------|:--------------------------------|
|
||||
| `anchor` | `` `${"top" \| "middle" \| "bottom"}${"Left" \| "Center" \| "Right"}` `` | `"bottomCenter"` | Position of Popover content |
|
||||
| `offset` | `"sm" \| "md" \| "lg"` | `"md"` | Gap between trigger and popover |
|
||||
|
||||
#### Special
|
||||
|
||||
| Prop | Type | Default | Description |
|
||||
|:----------|:----------|:--------|:--------------------------------------------------------------------------------|
|
||||
| `asChild` | `boolean` | `false` | `Whether the container of popover content is rendered as a child of a component |
|
||||
|
||||
## Examples
|
||||
|
||||
### Theme Selector
|
||||
|
||||
<TabProvider defaultName={"preview"}>
|
||||
<TabList>
|
||||
<TabTrigger name={"preview"}>Preview</TabTrigger>
|
||||
<TabTrigger name={"code"}>Code</TabTrigger>
|
||||
</TabList>
|
||||
<TabContent name={"preview"}>
|
||||
<Story layout={"centered"}>
|
||||
<Examples.ThemeSelector />
|
||||
</Story>
|
||||
</TabContent>
|
||||
<TabContent name={"code"}>
|
||||
<LoadedCode from={`${GITHUB}/packages/react/src/docs/components/PopoverBlocks/Examples/ThemeSelector.tsx`} />
|
||||
</TabContent>
|
||||
</TabProvider>
|
||||
|
||||
### User Control
|
||||
|
||||
<TabProvider defaultName={"preview"}>
|
||||
<TabList>
|
||||
<TabTrigger name={"preview"}>Preview</TabTrigger>
|
||||
<TabTrigger name={"code"}>Code</TabTrigger>
|
||||
</TabList>
|
||||
<TabContent name={"preview"}>
|
||||
<Story layout={"centered"}>
|
||||
<Examples.UserControl />
|
||||
</Story>
|
||||
</TabContent>
|
||||
<TabContent name={"code"}>
|
||||
<LoadedCode from={`${GITHUB}/packages/react/src/docs/components/PopoverBlocks/Examples/UserControl.tsx`} />
|
||||
</TabContent>
|
||||
</TabProvider>
|
@ -0,0 +1,43 @@
|
||||
import { Popover, PopoverTrigger, PopoverContent } from "@components/Popover.tsx";
|
||||
import { Button } from "@components/Button.tsx";
|
||||
import { useState } from "react";
|
||||
|
||||
const DarkIcon = () => {
|
||||
// ic:baseline-dark-mode
|
||||
return <svg xmlns="http://www.w3.org/2000/svg" width="1.2em" height="1.2em" viewBox="0 0 24 24">
|
||||
<path fill="currentColor"
|
||||
d="M12 3a9 9 0 1 0 9 9c0-.46-.04-.92-.1-1.36a5.389 5.389 0 0 1-4.4 2.26a5.403 5.403 0 0 1-3.14-9.8c-.44-.06-.9-.1-1.36-.1"/>
|
||||
</svg>
|
||||
}
|
||||
|
||||
const LightIcon = () => {
|
||||
// ic:baseline-light-mode
|
||||
return <svg xmlns="http://www.w3.org/2000/svg" width="1.2em" height="1.2em" viewBox="0 0 24 24">
|
||||
<path fill="currentColor"
|
||||
d="M12 7c-2.76 0-5 2.24-5 5s2.24 5 5 5s5-2.24 5-5s-2.24-5-5-5M2 13h2c.55 0 1-.45 1-1s-.45-1-1-1H2c-.55 0-1 .45-1 1s.45 1 1 1m18 0h2c.55 0 1-.45 1-1s-.45-1-1-1h-2c-.55 0-1 .45-1 1s.45 1 1 1M11 2v2c0 .55.45 1 1 1s1-.45 1-1V2c0-.55-.45-1-1-1s-1 .45-1 1m0 18v2c0 .55.45 1 1 1s1-.45 1-1v-2c0-.55-.45-1-1-1s-1 .45-1 1M5.99 4.58a.996.996 0 0 0-1.41 0a.996.996 0 0 0 0 1.41l1.06 1.06c.39.39 1.03.39 1.41 0s.39-1.03 0-1.41zm12.37 12.37a.996.996 0 0 0-1.41 0a.996.996 0 0 0 0 1.41l1.06 1.06c.39.39 1.03.39 1.41 0a.996.996 0 0 0 0-1.41zm1.06-10.96a.996.996 0 0 0 0-1.41a.996.996 0 0 0-1.41 0l-1.06 1.06c-.39.39-.39 1.03 0 1.41s1.03.39 1.41 0zM7.05 18.36a.996.996 0 0 0 0-1.41a.996.996 0 0 0-1.41 0l-1.06 1.06c-.39.39-.39 1.03 0 1.41s1.03.39 1.41 0z"/>
|
||||
</svg>
|
||||
}
|
||||
|
||||
export const ThemeSelector = () => {
|
||||
const [theme, setTheme] = useState<"light" | "dark">("dark");
|
||||
|
||||
return <Popover>
|
||||
<PopoverTrigger>
|
||||
<Button preset={"default"} size={"icon"}>
|
||||
{
|
||||
theme === "light" ? <LightIcon /> : <DarkIcon />
|
||||
}
|
||||
</Button>
|
||||
</PopoverTrigger>
|
||||
<PopoverContent anchor={"bottomCenter"}>
|
||||
<Button onClick={() => setTheme("dark")} preset={"ghost"} className={"gap-2"}>
|
||||
<DarkIcon />
|
||||
<span className={"whitespace-nowrap"}>Dark Mode</span>
|
||||
</Button>
|
||||
<Button onClick={() => setTheme("light")} preset={"ghost"} className={"gap-2"}>
|
||||
<LightIcon />
|
||||
<span className={"whitespace-nowrap"}>Light Mode</span>
|
||||
</Button>
|
||||
</PopoverContent>
|
||||
</Popover>
|
||||
}
|
@ -0,0 +1,151 @@
|
||||
import {
|
||||
Popover,
|
||||
PopoverTrigger,
|
||||
PopoverContent,
|
||||
} from "@components/Popover.tsx";
|
||||
import { Button } from "@components/Button.tsx";
|
||||
import { useToast } from "@components/Toast.tsx";
|
||||
import {
|
||||
createContext,
|
||||
Dispatch,
|
||||
SetStateAction,
|
||||
SVGProps,
|
||||
useContext,
|
||||
useState,
|
||||
useTransition,
|
||||
} from "react";
|
||||
import { Label } from "@components/Label.tsx";
|
||||
import { Input } from "@components/Input.tsx";
|
||||
|
||||
interface UserControlState {
|
||||
signIn: boolean;
|
||||
}
|
||||
const initialState: UserControlState = {
|
||||
signIn: false,
|
||||
};
|
||||
const UserControlContext = createContext<
|
||||
[UserControlState, Dispatch<SetStateAction<UserControlState>>]
|
||||
>([initialState, () => {}]);
|
||||
|
||||
const logInServerAction = async () => {
|
||||
return new Promise((r) => setTimeout(r, 2000));
|
||||
};
|
||||
|
||||
const logOutServerAction = async () => {
|
||||
return new Promise((r) => setTimeout(r, 1000));
|
||||
};
|
||||
|
||||
function MdiLoading(props: SVGProps<SVGSVGElement>) {
|
||||
return (
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1.2em"
|
||||
height="1.2em"
|
||||
viewBox="0 0 24 24"
|
||||
{...props}
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="M12 4V2A10 10 0 0 0 2 12h2a8 8 0 0 1 8-8"
|
||||
></path>
|
||||
</svg>
|
||||
);
|
||||
}
|
||||
|
||||
const SignInForm = () => {
|
||||
const [isSigningIn, setIsSigningIn] = useState(false);
|
||||
const transition = useTransition();
|
||||
const [_, setState] = useContext(UserControlContext);
|
||||
const { toast } = useToast();
|
||||
|
||||
function startSignIn() {
|
||||
transition[1](() => {
|
||||
setIsSigningIn(true);
|
||||
const toasted = toast({
|
||||
title: "Logging In...",
|
||||
description: "Please wait until server responses",
|
||||
status: "loading",
|
||||
});
|
||||
logInServerAction().then(() => {
|
||||
toasted.update({
|
||||
title: "Log In Success",
|
||||
description: "Successfully logged in!",
|
||||
status: "success",
|
||||
});
|
||||
setIsSigningIn(false);
|
||||
setState((prev) => ({ ...prev, signIn: true }));
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
return (
|
||||
<PopoverContent anchor={"bottomLeft"} className={"p-4 space-y-3"}>
|
||||
<Label>
|
||||
<span>Username</span>
|
||||
<Input type={"text"} />
|
||||
</Label>
|
||||
<Label>
|
||||
<span>Password</span>
|
||||
<Input type={"password"} />
|
||||
</Label>
|
||||
<div className={"flex flex-row justify-end"}>
|
||||
<Button preset={"success"} onClick={startSignIn}>
|
||||
{isSigningIn ? <MdiLoading className={"animate-spin"} /> : "Sign In"}
|
||||
</Button>
|
||||
</div>
|
||||
</PopoverContent>
|
||||
);
|
||||
};
|
||||
|
||||
const UserControlContent = () => {
|
||||
const [isSigningOut, setIsSigningOut] = useState(false);
|
||||
const transition = useTransition();
|
||||
const [_, setState] = useContext(UserControlContext);
|
||||
const { toast } = useToast();
|
||||
|
||||
function startSignOut() {
|
||||
transition[1](() => {
|
||||
setIsSigningOut(true);
|
||||
const toasted = toast({
|
||||
title: "Logging Out",
|
||||
description: "Please wait until server responses",
|
||||
status: "loading",
|
||||
});
|
||||
logOutServerAction().then(() => {
|
||||
toasted.update({
|
||||
title: "Log Out Success",
|
||||
description: "Successfully logged out!",
|
||||
status: "success",
|
||||
});
|
||||
setIsSigningOut(false);
|
||||
setState((prev) => ({ ...prev, signIn: false }));
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
return (
|
||||
<PopoverContent anchor={"bottomLeft"}>
|
||||
<Button preset={"ghost"}>Dashboard</Button>
|
||||
<Button preset={"ghost"} onClick={startSignOut}>
|
||||
{isSigningOut ? <MdiLoading className={"animate-spin"} /> : "Sign Out"}
|
||||
</Button>
|
||||
</PopoverContent>
|
||||
);
|
||||
};
|
||||
|
||||
export const UserControl = () => {
|
||||
const [state, setState] = useState<UserControlState>({
|
||||
signIn: false,
|
||||
});
|
||||
|
||||
return (
|
||||
<Popover>
|
||||
<PopoverTrigger>
|
||||
<Button>{state.signIn ? "Log Out" : "Log In"}</Button>
|
||||
</PopoverTrigger>
|
||||
<UserControlContext.Provider value={[state, setState]}>
|
||||
{state.signIn ? <UserControlContent /> : <SignInForm />}
|
||||
</UserControlContext.Provider>
|
||||
</Popover>
|
||||
);
|
||||
};
|
@ -0,0 +1,7 @@
|
||||
import { ThemeSelector } from "./ThemeSelector";
|
||||
import { UserControl } from "./UserControl";
|
||||
|
||||
export default {
|
||||
ThemeSelector,
|
||||
UserControl,
|
||||
}
|
54
packages/react/src/docs/components/PopoverBlocks/Preview.tsx
Normal file
54
packages/react/src/docs/components/PopoverBlocks/Preview.tsx
Normal file
@ -0,0 +1,54 @@
|
||||
import { Button } from "@components/Button";
|
||||
import { Popover, PopoverContent, PopoverTrigger } from "@components/Popover";
|
||||
|
||||
export function PopoverDemo() {
|
||||
return (
|
||||
<Popover>
|
||||
<PopoverTrigger>
|
||||
<Button size="icon">
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1.2em"
|
||||
height="1.2em"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="M12 4a4 4 0 0 1 4 4a4 4 0 0 1-4 4a4 4 0 0 1-4-4a4 4 0 0 1 4-4m0 10c4.42 0 8 1.79 8 4v2H4v-2c0-2.21 3.58-4 8-4"
|
||||
/>
|
||||
</svg>
|
||||
</Button>
|
||||
</PopoverTrigger>
|
||||
<PopoverContent>
|
||||
<Button preset="ghost" className="gap-2">
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1.2em"
|
||||
height="1.2em"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="M3 6h18v2H3zm0 5h18v2H3zm0 5h18v2H3z"
|
||||
/>
|
||||
</svg>
|
||||
<span className="flex-grow text-left">Dashboard</span>
|
||||
</Button>
|
||||
<Button preset="ghost" className="gap-2">
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1.2em"
|
||||
height="1.2em"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="m17 7l-1.41 1.41L18.17 11H8v2h10.17l-2.58 2.58L17 17l5-5M4 5h8V3H4c-1.1 0-2 .9-2 2v14c0 1.1.9 2 2 2h8v-2H4z"
|
||||
/>
|
||||
</svg>
|
||||
<span className="flex-grow text-left">Log out</span>
|
||||
</Button>
|
||||
</PopoverContent>
|
||||
</Popover>
|
||||
);
|
||||
}
|
Loading…
x
Reference in New Issue
Block a user