Compare commits
47 Commits
dependabot
...
main
Author | SHA1 | Date | |
---|---|---|---|
fb2002a0a4 | |||
dce7823955 | |||
701a77d234 | |||
05426efcaf | |||
1281e89275 | |||
a8b195cd02 | |||
c0b58d1737 | |||
b5edb5d03e | |||
fe71775ae8 | |||
b9c30fb5af | |||
ff202e4678 | |||
898b8a15a9 | |||
bde80e60eb | |||
c5bf58b558 | |||
da1a53ba29 | |||
cf77c6908b | |||
ca9036c1ad | |||
ae4453dd21 | |||
c5274444c0 | |||
52a78b146c | |||
80a34d733e | |||
75031437bc | |||
ed936f0378 | |||
14a7487c2e | |||
24260fe375 | |||
bc6533bf01 | |||
662aab0e2b | |||
e4d9c8bae4 | |||
4add04ab7e | |||
a68d864a00 | |||
58a8b8b241 | |||
bcac697a04 | |||
ba344d4159 | |||
09aa5e0d78 | |||
a7e2f7ac42 | |||
d7ecef09a0 | |||
f1b6e5336d | |||
9070bc3424 | |||
9a3ea58d1c | |||
dd67bc7685 | |||
d9fea1d8c9 | |||
3338fc65e2 | |||
8e7ee48266 | |||
59358f9deb | |||
c60a277826 | |||
f92be39455 | |||
b76014db0b |
17
package.json
17
package.json
@ -7,7 +7,6 @@
|
|||||||
"author": "Shinwoo PARK <devpysweb@gmail.com>",
|
"author": "Shinwoo PARK <devpysweb@gmail.com>",
|
||||||
"license": "MIT",
|
"license": "MIT",
|
||||||
"private": true,
|
"private": true,
|
||||||
"packageManager": "yarn@4.3.0+sha512.1606bef7c84bc7d83b8576063de2fd08f7d69f9939015bed800f9585a002390268ecc777e9feeba7e26e9556aef6beaad4806968db2182ab5dd3e955ab3b9a0b",
|
|
||||||
"type": "module",
|
"type": "module",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"postinstall": "lefthook install",
|
"postinstall": "lefthook install",
|
||||||
@ -19,14 +18,16 @@
|
|||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@mdx-js/react": "^3.0.1",
|
"@mdx-js/react": "^3.0.1",
|
||||||
"@stefanprobst/rehype-extract-toc": "^2.2.0",
|
"@stefanprobst/rehype-extract-toc": "^2.2.0",
|
||||||
|
"@tailwindcss/vite": "^4.0.12",
|
||||||
"highlight.js": "^11.9.0",
|
"highlight.js": "^11.9.0",
|
||||||
"react": "^18.3.1",
|
"react": "^18.3.1",
|
||||||
"react-dom": "^18.3.1",
|
"react-dom": "^18.3.1",
|
||||||
"react-router-dom": "^6.23.1",
|
"react-router-dom": "^6.23.1",
|
||||||
"react-syntax-highlighter": "^15.5.0",
|
"react-syntax-highlighter": "^15.6.1",
|
||||||
"rehype-slug": "^6.0.0",
|
"rehype-slug": "^6.0.0",
|
||||||
"remark-gfm": "^4.0.0",
|
"remark-gfm": "^4.0.0",
|
||||||
"tailwind-merge": "^2.3.0"
|
"tailwind-merge": "^2.3.0",
|
||||||
|
"typescript": "^5.4.5"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@biomejs/biome": "1.8.3",
|
"@biomejs/biome": "1.8.3",
|
||||||
@ -42,13 +43,11 @@
|
|||||||
"@typescript-eslint/eslint-plugin": "^7.13.0",
|
"@typescript-eslint/eslint-plugin": "^7.13.0",
|
||||||
"@typescript-eslint/parser": "^7.13.0",
|
"@typescript-eslint/parser": "^7.13.0",
|
||||||
"@vitejs/plugin-react": "^4.3.1",
|
"@vitejs/plugin-react": "^4.3.1",
|
||||||
"autoprefixer": "^10.4.19",
|
|
||||||
"lefthook": "^1.6.18",
|
"lefthook": "^1.6.18",
|
||||||
"postcss": "^8.4.38",
|
"tailwind-scrollbar": "^4.0.1",
|
||||||
"tailwind-scrollbar": "^3.1.0",
|
"tailwindcss": "^4.0.12",
|
||||||
"tailwindcss": "^3.4.4",
|
|
||||||
"typescript": "^5.4.5",
|
|
||||||
"vite": "^5.3.0",
|
"vite": "^5.3.0",
|
||||||
"vite-plugin-dynamic-import": "^1.5.0"
|
"vite-plugin-dynamic-import": "^1.5.0"
|
||||||
}
|
},
|
||||||
|
"packageManager": "yarn@4.4.0+sha512.91d93b445d9284e7ed52931369bc89a663414e5582d00eea45c67ddc459a2582919eece27c412d6ffd1bd0793ff35399381cb229326b961798ce4f4cc60ddfdb"
|
||||||
}
|
}
|
||||||
|
84
src/App.tsx
84
src/App.tsx
@ -9,19 +9,8 @@ import {
|
|||||||
import DocsLayout from "./DocsLayout";
|
import DocsLayout from "./DocsLayout";
|
||||||
import DynamicLayout from "./DynamicLayout";
|
import DynamicLayout from "./DynamicLayout";
|
||||||
import ErrorBoundary from "./ErrorHandler";
|
import ErrorBoundary from "./ErrorHandler";
|
||||||
import Home from "./Home";
|
|
||||||
import MainLayout from "./MainLayout";
|
import MainLayout from "./MainLayout";
|
||||||
|
|
||||||
import DocsConfiguration, {
|
|
||||||
tableOfContents as docsConfigurationToc,
|
|
||||||
} from "./docs/configuration.mdx";
|
|
||||||
import DocsInstallation, {
|
|
||||||
tableOfContents as docsInstallationToc,
|
|
||||||
} from "./docs/installation.mdx";
|
|
||||||
import DocsIntroduction, {
|
|
||||||
tableOfContents as docsIntroductionToc,
|
|
||||||
} from "./docs/introduction.mdx";
|
|
||||||
|
|
||||||
import { Tooltip, TooltipContent } from "@pswui/Tooltip";
|
import { Tooltip, TooltipContent } from "@pswui/Tooltip";
|
||||||
import React, {
|
import React, {
|
||||||
type ForwardedRef,
|
type ForwardedRef,
|
||||||
@ -225,12 +214,20 @@ const router = createBrowserRouter(
|
|||||||
>
|
>
|
||||||
<Route
|
<Route
|
||||||
index
|
index
|
||||||
loader={() =>
|
lazy={async () => {
|
||||||
REDIRECTED_404.test(window.location.search)
|
const { default: Component } = await import("./Home");
|
||||||
? redirect(REDIRECTED_404.exec(window.location.search)?.[1] ?? "/")
|
|
||||||
: true
|
return {
|
||||||
}
|
Component,
|
||||||
element={<Home />}
|
loader() {
|
||||||
|
return REDIRECTED_404.test(window.location.search)
|
||||||
|
? redirect(
|
||||||
|
REDIRECTED_404.exec(window.location.search)?.[1] ?? "/",
|
||||||
|
)
|
||||||
|
: true;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path="docs"
|
path="docs"
|
||||||
@ -242,27 +239,50 @@ const router = createBrowserRouter(
|
|||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path="introduction"
|
path="introduction"
|
||||||
element={
|
lazy={async () => {
|
||||||
<DynamicLayout toc={docsIntroductionToc}>
|
const { default: DocsIntroduction, tableOfContents } = await import(
|
||||||
<DocsIntroduction components={overrideComponents} />
|
"./docs/introduction.mdx"
|
||||||
</DynamicLayout>
|
);
|
||||||
}
|
|
||||||
|
return {
|
||||||
|
Component: () => (
|
||||||
|
<DynamicLayout toc={tableOfContents}>
|
||||||
|
<DocsIntroduction components={overrideComponents} />
|
||||||
|
</DynamicLayout>
|
||||||
|
),
|
||||||
|
};
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path="installation"
|
path="installation"
|
||||||
element={
|
lazy={async () => {
|
||||||
<DynamicLayout toc={docsInstallationToc}>
|
const { default: DocsInstallation, tableOfContents } = await import(
|
||||||
<DocsInstallation components={overrideComponents} />
|
"./docs/installation.mdx"
|
||||||
</DynamicLayout>
|
);
|
||||||
}
|
|
||||||
|
return {
|
||||||
|
Component: () => (
|
||||||
|
<DynamicLayout toc={tableOfContents}>
|
||||||
|
<DocsInstallation components={overrideComponents} />
|
||||||
|
</DynamicLayout>
|
||||||
|
),
|
||||||
|
};
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
<Route
|
<Route
|
||||||
path="configuration"
|
path="configuration"
|
||||||
element={
|
lazy={async () => {
|
||||||
<DynamicLayout toc={docsConfigurationToc}>
|
const { default: DocsConfiguration, tableOfContents } =
|
||||||
<DocsConfiguration components={overrideComponents} />
|
await import("./docs/configuration.mdx");
|
||||||
</DynamicLayout>
|
|
||||||
}
|
return {
|
||||||
|
Component: () => (
|
||||||
|
<DynamicLayout toc={tableOfContents}>
|
||||||
|
<DocsConfiguration components={overrideComponents} />
|
||||||
|
</DynamicLayout>
|
||||||
|
),
|
||||||
|
};
|
||||||
|
}}
|
||||||
/>
|
/>
|
||||||
<Route path="components">
|
<Route path="components">
|
||||||
<Route
|
<Route
|
||||||
|
@ -33,7 +33,7 @@ function SideNav() {
|
|||||||
|
|
||||||
function DocsLayout() {
|
function DocsLayout() {
|
||||||
return (
|
return (
|
||||||
<div className="flex-grow grid grid-cols-1 md:grid-cols-[12rem_1fr] lg:grid-cols-[12rem_1fr_10rem] w-full max-w-5xl mx-auto">
|
<div className="grow grid grid-cols-1 md:grid-cols-[12rem_1fr] lg:grid-cols-[12rem_1fr_10rem] w-full max-w-5xl mx-auto">
|
||||||
<SideNav />
|
<SideNav />
|
||||||
<Outlet />
|
<Outlet />
|
||||||
</div>
|
</div>
|
||||||
|
@ -3,7 +3,7 @@ import { Link } from "react-router-dom";
|
|||||||
|
|
||||||
function Home() {
|
function Home() {
|
||||||
return (
|
return (
|
||||||
<main className="flex-grow h-full flex flex-col p-4 justify-center items-center">
|
<main className="grow h-full flex flex-col p-4 justify-center items-center">
|
||||||
<section className="h-full flex flex-col justify-center items-center text-center gap-8">
|
<section className="h-full flex flex-col justify-center items-center text-center gap-8">
|
||||||
<header className="flex flex-col justify-center items-center gap-2">
|
<header className="flex flex-col justify-center items-center gap-2">
|
||||||
<h1 className="text-4xl font-bold">
|
<h1 className="text-4xl font-bold">
|
||||||
|
@ -62,7 +62,8 @@ function ThemeButton() {
|
|||||||
</Button>
|
</Button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent
|
<PopoverContent
|
||||||
anchor="bottomLeft"
|
anchor={"end"}
|
||||||
|
align={"end"}
|
||||||
className="w-32"
|
className="w-32"
|
||||||
>
|
>
|
||||||
<Button
|
<Button
|
||||||
@ -149,7 +150,7 @@ function TopNav() {
|
|||||||
</svg>
|
</svg>
|
||||||
</Button>
|
</Button>
|
||||||
</DrawerTrigger>
|
</DrawerTrigger>
|
||||||
<DrawerOverlay className="z-[99]">
|
<DrawerOverlay className="z-99">
|
||||||
<DrawerContent className="w-[300px] overflow-auto">
|
<DrawerContent className="w-[300px] overflow-auto">
|
||||||
<DrawerClose className="absolute top-4 right-4">
|
<DrawerClose className="absolute top-4 right-4">
|
||||||
<Button
|
<Button
|
||||||
|
@ -1,10 +1,39 @@
|
|||||||
import { Button } from "@pswui/Button";
|
import { Button } from "@pswui/Button";
|
||||||
import { useToast } from "@pswui/Toast";
|
import { useToast } from "@pswui/Toast";
|
||||||
import { forwardRef, useEffect, useMemo, useState } from "react";
|
import {
|
||||||
import SyntaxHighlighter from "react-syntax-highlighter";
|
type Component,
|
||||||
import { gruvboxDark } from "react-syntax-highlighter/dist/cjs/styles/hljs";
|
forwardRef,
|
||||||
|
useEffect,
|
||||||
|
useMemo,
|
||||||
|
useState,
|
||||||
|
} from "react";
|
||||||
|
import {
|
||||||
|
PrismLight,
|
||||||
|
type SyntaxHighlighterProps,
|
||||||
|
} from "react-syntax-highlighter";
|
||||||
|
import { duotoneSpace } from "react-syntax-highlighter/dist/cjs/styles/prism";
|
||||||
|
import css from "react-syntax-highlighter/dist/esm/languages/prism/css";
|
||||||
|
import js from "react-syntax-highlighter/dist/esm/languages/prism/javascript";
|
||||||
|
import jsx from "react-syntax-highlighter/dist/esm/languages/prism/jsx";
|
||||||
|
import markup from "react-syntax-highlighter/dist/esm/languages/prism/markup";
|
||||||
|
import tsx from "react-syntax-highlighter/dist/esm/languages/prism/tsx";
|
||||||
|
import ts from "react-syntax-highlighter/dist/esm/languages/prism/typescript";
|
||||||
import { twMerge } from "tailwind-merge";
|
import { twMerge } from "tailwind-merge";
|
||||||
|
|
||||||
|
const SyntaxHighlighter =
|
||||||
|
PrismLight as unknown as typeof Component<SyntaxHighlighterProps> & {
|
||||||
|
registerLanguage<T>(name: string, func: T): void;
|
||||||
|
alias(name: string, alias: string | string[]): void;
|
||||||
|
alias(aliases: Record<string, string | string[]>): void;
|
||||||
|
};
|
||||||
|
|
||||||
|
SyntaxHighlighter.registerLanguage("javascript", js);
|
||||||
|
SyntaxHighlighter.registerLanguage("typescript", ts);
|
||||||
|
SyntaxHighlighter.registerLanguage("tsx", tsx);
|
||||||
|
SyntaxHighlighter.registerLanguage("jsx", jsx);
|
||||||
|
SyntaxHighlighter.registerLanguage("markup", markup);
|
||||||
|
SyntaxHighlighter.registerLanguage("css", css);
|
||||||
|
|
||||||
export const GITHUB_UI = "https://raw.githubusercontent.com/pswui/ui/main";
|
export const GITHUB_UI = "https://raw.githubusercontent.com/pswui/ui/main";
|
||||||
export const GITHUB_DOCS = "https://raw.githubusercontent.com/pswui/docs/main";
|
export const GITHUB_DOCS = "https://raw.githubusercontent.com/pswui/docs/main";
|
||||||
export const GITHUB_COMP = (componentName: string) =>
|
export const GITHUB_COMP = (componentName: string) =>
|
||||||
@ -16,17 +45,19 @@ export const GITHUB_COMP_PREVIEW = (componentName: string) =>
|
|||||||
export const GITHUB_STORY = (componentName: string, storyName: string) =>
|
export const GITHUB_STORY = (componentName: string, storyName: string) =>
|
||||||
`${GITHUB_DOCS}/src/docs/components/${componentName}Blocks/Examples/${storyName}.tsx`;
|
`${GITHUB_DOCS}/src/docs/components/${componentName}Blocks/Examples/${storyName}.tsx`;
|
||||||
|
|
||||||
export type TEMPLATE = Record<string, Record<string, string | boolean>>;
|
export type TEMPLATE = Record<
|
||||||
|
string,
|
||||||
|
Record<string, string | boolean | number | undefined>
|
||||||
|
>;
|
||||||
|
|
||||||
export const LoadedCode = ({
|
const TEMPLATE_REMOVE_REGEX = /\/\*\s*remove\s*\*\/(.|\n)*?\/\*\s*end\s*\*\//g;
|
||||||
from,
|
const TEMPLATE_REPLACE_REGEX =
|
||||||
className,
|
/\/\*\s*replace\s*\*\/(.|\n)*?\/\*\s*with\s*\n((.|\n)+)\n\s*\*\//g;
|
||||||
template,
|
|
||||||
}: {
|
export const LoadedCode = forwardRef<
|
||||||
from: string;
|
HTMLDivElement,
|
||||||
className?: string;
|
{ from: string; className?: string; template?: TEMPLATE }
|
||||||
template?: TEMPLATE;
|
>(({ from, className, template }, ref) => {
|
||||||
}) => {
|
|
||||||
const [state, setState] = useState<string | undefined | null>();
|
const [state, setState] = useState<string | undefined | null>();
|
||||||
const { toast } = useToast();
|
const { toast } = useToast();
|
||||||
|
|
||||||
@ -44,6 +75,10 @@ export const LoadedCode = ({
|
|||||||
|
|
||||||
let templatedCode = state;
|
let templatedCode = state;
|
||||||
|
|
||||||
|
templatedCode = templatedCode
|
||||||
|
.replaceAll(TEMPLATE_REMOVE_REGEX, "")
|
||||||
|
.replaceAll(TEMPLATE_REPLACE_REGEX, "$2");
|
||||||
|
|
||||||
for (const [componentName, componentTemplateProps] of Object.entries(
|
for (const [componentName, componentTemplateProps] of Object.entries(
|
||||||
template,
|
template,
|
||||||
)) {
|
)) {
|
||||||
@ -51,13 +86,15 @@ export const LoadedCode = ({
|
|||||||
componentTemplateProps,
|
componentTemplateProps,
|
||||||
)) {
|
)) {
|
||||||
const regex = new RegExp(
|
const regex = new RegExp(
|
||||||
`(<${componentName}\s[^]*)\s${propName}=(\{(true|false|"[^"\n]*"|'[^'\n]*'|\`[^\`\n]*\`)\}|"[^"\n]*"|'[^'\n]*')`,
|
`(<${componentName.slice(0, componentName.length - 5)}\\b[^>]*?)(\n?\\s+)${propName}={${componentName}.${propName}}`,
|
||||||
);
|
);
|
||||||
templatedCode = templatedCode.replace(
|
templatedCode = templatedCode.replace(
|
||||||
regex,
|
regex,
|
||||||
typeof propValue === "string"
|
typeof propValue === "undefined"
|
||||||
? `\$1 ${propName}="${propValue}"`
|
? "$1"
|
||||||
: `$1 ${propName}={${propValue}}`,
|
: typeof propValue === "string"
|
||||||
|
? `\$1$2 ${propName}="${propValue}"`
|
||||||
|
: `$1$2 ${propName}={${propValue}}`,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -66,7 +103,10 @@ export const LoadedCode = ({
|
|||||||
}, [state, template]);
|
}, [state, template]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className={twMerge("relative", className)}>
|
<div
|
||||||
|
className={twMerge("relative", className)}
|
||||||
|
ref={ref}
|
||||||
|
>
|
||||||
<Button
|
<Button
|
||||||
preset="default"
|
preset="default"
|
||||||
size="icon"
|
size="icon"
|
||||||
@ -103,15 +143,16 @@ export const LoadedCode = ({
|
|||||||
</Button>
|
</Button>
|
||||||
<SyntaxHighlighter
|
<SyntaxHighlighter
|
||||||
language="typescript"
|
language="typescript"
|
||||||
style={gruvboxDark}
|
style={duotoneSpace}
|
||||||
className={`w-full h-64 rounded-lg ${!state ? "animate-pulse" : ""} scrollbar-none`}
|
className={`w-full h-64 rounded-lg ${!state ? "animate-pulse" : ""} scrollbar-none resize-y`}
|
||||||
customStyle={{ padding: "1rem" }}
|
customStyle={{ padding: "1rem" }}
|
||||||
>
|
>
|
||||||
{postProcessedCode}
|
{postProcessedCode}
|
||||||
</SyntaxHighlighter>
|
</SyntaxHighlighter>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
});
|
||||||
|
LoadedCode.displayName = "LoadedCode";
|
||||||
|
|
||||||
export const Code = forwardRef<
|
export const Code = forwardRef<
|
||||||
HTMLDivElement,
|
HTMLDivElement,
|
||||||
@ -152,7 +193,7 @@ export const Code = forwardRef<
|
|||||||
</Button>
|
</Button>
|
||||||
<SyntaxHighlighter
|
<SyntaxHighlighter
|
||||||
language={language}
|
language={language}
|
||||||
style={gruvboxDark}
|
style={duotoneSpace}
|
||||||
className={"w-full h-auto max-h-64 rounded-lg scrollbar-none"}
|
className={"w-full h-auto max-h-64 rounded-lg scrollbar-none"}
|
||||||
customStyle={{ padding: "1rem" }}
|
customStyle={{ padding: "1rem" }}
|
||||||
>
|
>
|
||||||
|
44
src/components/PgHooks.tsx
Normal file
44
src/components/PgHooks.tsx
Normal file
@ -0,0 +1,44 @@
|
|||||||
|
import useMutable from "@/utils/useMutable";
|
||||||
|
import { useMemo } from "react";
|
||||||
|
import type { TEMPLATE } from "./LoadedCode";
|
||||||
|
import type { ControlTemplate, Template } from "./Playground";
|
||||||
|
|
||||||
|
export function usePgProps<T extends TEMPLATE>(
|
||||||
|
t: Template,
|
||||||
|
): [T, ControlTemplate] {
|
||||||
|
const [props, mutate] = useMutable(t);
|
||||||
|
|
||||||
|
return useMemo(() => {
|
||||||
|
const rawProps: TEMPLATE = {};
|
||||||
|
const controlTemplate: ControlTemplate = {};
|
||||||
|
|
||||||
|
for (const [componentName, prop] of Object.entries(props)) {
|
||||||
|
const pre: ControlTemplate[string] = {};
|
||||||
|
const vals: TEMPLATE[string] = {};
|
||||||
|
|
||||||
|
for (const [propKey, propMeta] of Object.entries(prop)) {
|
||||||
|
pre[propKey] = {
|
||||||
|
...propMeta,
|
||||||
|
onChange(value: string | number | boolean) {
|
||||||
|
console.log(`mutating ${componentName}/${propKey}`);
|
||||||
|
mutate((state) => {
|
||||||
|
state[componentName][propKey].value = value;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
onToggle(v: boolean) {
|
||||||
|
console.log(`toggling ${componentName}/${propKey}`);
|
||||||
|
mutate((state) => {
|
||||||
|
state[componentName][propKey].disabled = v;
|
||||||
|
});
|
||||||
|
},
|
||||||
|
};
|
||||||
|
vals[propKey] = propMeta.disabled ? undefined : propMeta.value;
|
||||||
|
}
|
||||||
|
|
||||||
|
controlTemplate[componentName] = pre;
|
||||||
|
rawProps[componentName] = vals;
|
||||||
|
}
|
||||||
|
|
||||||
|
return [rawProps as T, controlTemplate];
|
||||||
|
}, [props, mutate]);
|
||||||
|
}
|
183
src/components/Playground.tsx
Normal file
183
src/components/Playground.tsx
Normal file
@ -0,0 +1,183 @@
|
|||||||
|
import { Button } from "@pswui/Button";
|
||||||
|
import { Checkbox } from "@pswui/Checkbox";
|
||||||
|
import { Input } from "@pswui/Input";
|
||||||
|
import { Label } from "@pswui/Label";
|
||||||
|
import { Popover, PopoverContent, PopoverTrigger } from "@pswui/Popover";
|
||||||
|
import { Switch } from "@pswui/Switch";
|
||||||
|
import { TabContent, TabList, TabProvider, TabTrigger } from "@pswui/Tabs";
|
||||||
|
import type { ReactNode } from "react";
|
||||||
|
import { GITHUB_COMP_PREVIEW, LoadedCode, type TEMPLATE } from "./LoadedCode";
|
||||||
|
import { Story } from "./Story";
|
||||||
|
|
||||||
|
export type Template = Record<
|
||||||
|
string,
|
||||||
|
Record<
|
||||||
|
string,
|
||||||
|
| { type: "boolean"; value: boolean; disabled?: boolean }
|
||||||
|
| { type: "select"; options: string[]; value: string; disabled?: boolean }
|
||||||
|
| { type: "string"; value: string; disabled?: boolean }
|
||||||
|
| { type: "number"; value: number; disabled?: boolean }
|
||||||
|
>
|
||||||
|
>;
|
||||||
|
|
||||||
|
export type ControlTemplate = Record<
|
||||||
|
string,
|
||||||
|
Record<
|
||||||
|
string,
|
||||||
|
| {
|
||||||
|
type: "boolean";
|
||||||
|
value: boolean;
|
||||||
|
disabled?: boolean;
|
||||||
|
onChange: (value: boolean) => void;
|
||||||
|
onToggle: (v: boolean) => void;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: "select";
|
||||||
|
options: string[];
|
||||||
|
value: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
onChange: (value: string) => void;
|
||||||
|
onToggle: (v: boolean) => void;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: "string";
|
||||||
|
value: string;
|
||||||
|
disabled?: boolean;
|
||||||
|
onChange: (value: string) => void;
|
||||||
|
onToggle: (v: boolean) => void;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
type: "number";
|
||||||
|
value: number;
|
||||||
|
disabled?: boolean;
|
||||||
|
onChange: (value: number) => void;
|
||||||
|
onToggle: (v: boolean) => void;
|
||||||
|
}
|
||||||
|
>
|
||||||
|
>;
|
||||||
|
|
||||||
|
export function PlaygroundControl<T extends ControlTemplate>(props: {
|
||||||
|
props: T;
|
||||||
|
}): ReactNode {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<h3>Controls</h3>
|
||||||
|
<div
|
||||||
|
className={
|
||||||
|
"rounded-lg p-4 border border-neutral-300 dark:border-neutral-700 flex flex-col justify-center items-start gap-12"
|
||||||
|
}
|
||||||
|
>
|
||||||
|
{Object.entries(props.props).map(([componentName, propEntries]) => (
|
||||||
|
<div
|
||||||
|
key={componentName}
|
||||||
|
className="w-full flex flex-col justify-center items-start gap-4"
|
||||||
|
>
|
||||||
|
<span className="font-thin opacity-50 w-full border-b border-b-current pb-2">
|
||||||
|
<{componentName.slice(0, componentName.length - 5)}>
|
||||||
|
</span>
|
||||||
|
{Object.entries(propEntries).map(([propName, propMeta]) => (
|
||||||
|
<div
|
||||||
|
key={componentName + propName}
|
||||||
|
className={`flex gap-2 justify-between w-full ${propMeta.type === "boolean" ? "flex-row items-center" : "flex-col md:flex-row md:items-center"}`}
|
||||||
|
>
|
||||||
|
<Label
|
||||||
|
direction="horizontal"
|
||||||
|
className="flex flex-row items-center gap-2"
|
||||||
|
>
|
||||||
|
<Switch
|
||||||
|
checked={!propMeta.disabled}
|
||||||
|
onChange={(e) => {
|
||||||
|
propMeta.onToggle(!e.currentTarget.checked);
|
||||||
|
}}
|
||||||
|
/>
|
||||||
|
{propMeta.disabled ? (
|
||||||
|
<s className="opacity-50">{propName}</s>
|
||||||
|
) : (
|
||||||
|
<span>{propName}</span>
|
||||||
|
)}
|
||||||
|
</Label>
|
||||||
|
{propMeta.type === "boolean" ? (
|
||||||
|
<Checkbox
|
||||||
|
checked={propMeta.value}
|
||||||
|
onChange={(e) => propMeta.onChange(e.currentTarget.checked)}
|
||||||
|
/>
|
||||||
|
) : propMeta.type === "string" ? (
|
||||||
|
<Input
|
||||||
|
type="text"
|
||||||
|
value={propMeta.value}
|
||||||
|
onChange={(e) => propMeta.onChange(e.currentTarget.value)}
|
||||||
|
className="w-full md:w-fit"
|
||||||
|
/>
|
||||||
|
) : propMeta.type === "select" ? (
|
||||||
|
<Popover>
|
||||||
|
<PopoverTrigger>
|
||||||
|
<Button
|
||||||
|
preset="default"
|
||||||
|
className="w-full md:w-fit"
|
||||||
|
>
|
||||||
|
{propMeta.value}
|
||||||
|
</Button>
|
||||||
|
</PopoverTrigger>
|
||||||
|
<PopoverContent className="min-w-36">
|
||||||
|
{propMeta.options.map((value) => (
|
||||||
|
<Button
|
||||||
|
preset="ghost"
|
||||||
|
key={value}
|
||||||
|
onClick={() => propMeta.onChange(value)}
|
||||||
|
>
|
||||||
|
{value}
|
||||||
|
</Button>
|
||||||
|
))}
|
||||||
|
</PopoverContent>
|
||||||
|
</Popover>
|
||||||
|
) : propMeta.type === "number" ? (
|
||||||
|
<Input
|
||||||
|
type="number"
|
||||||
|
value={propMeta.value}
|
||||||
|
onChange={(e) =>
|
||||||
|
propMeta.onChange(e.currentTarget.valueAsNumber)
|
||||||
|
}
|
||||||
|
className="w-full md:w-fit"
|
||||||
|
/>
|
||||||
|
) : null}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
|
</div>
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function PlaygroundLayout<T extends ControlTemplate>({
|
||||||
|
children,
|
||||||
|
compName,
|
||||||
|
props,
|
||||||
|
control,
|
||||||
|
}: {
|
||||||
|
children: ReactNode;
|
||||||
|
compName: string;
|
||||||
|
props: TEMPLATE;
|
||||||
|
control: T;
|
||||||
|
}) {
|
||||||
|
return (
|
||||||
|
<>
|
||||||
|
<TabProvider defaultName="preview">
|
||||||
|
<TabList>
|
||||||
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
|
</TabList>
|
||||||
|
<TabContent name="preview">
|
||||||
|
<Story layout="centered">{children}</Story>
|
||||||
|
</TabContent>
|
||||||
|
<TabContent name="code">
|
||||||
|
<LoadedCode
|
||||||
|
from={GITHUB_COMP_PREVIEW(compName)}
|
||||||
|
template={props}
|
||||||
|
/>
|
||||||
|
</TabContent>
|
||||||
|
</TabProvider>
|
||||||
|
<PlaygroundControl props={control} />
|
||||||
|
</>
|
||||||
|
);
|
||||||
|
}
|
@ -1,162 +1,152 @@
|
|||||||
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
||||||
import { Story } from "@/components/Story";
|
import { Story } from "@/components/Story";
|
||||||
import { LoadedCode, GITHUB_COMP, GITHUB_COMP_PREVIEW, GITHUB_STORY } from "@/components/LoadedCode";
|
import { LoadedCode, GITHUB_COMP, GITHUB_STORY } from "@/components/LoadedCode";
|
||||||
import { ButtonDemo } from "./ButtonBlocks/Preview";
|
import Examples from "./ButtonBlocks/Examples";
|
||||||
import Examples from "./ButtonBlocks/Examples";
|
import ButtonPlayground from "./ButtonBlocks/Playground";
|
||||||
|
|
||||||
# Button
|
# Button
|
||||||
Displays a button or a component that looks like a button.
|
|
||||||
|
Displays a button or a component that looks like a button.
|
||||||
<TabProvider defaultName="preview">
|
|
||||||
<TabList>
|
## Playground
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
<ButtonPlayground />
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
## Installation
|
||||||
<Story layout="centered">
|
|
||||||
<ButtonDemo />
|
1. Create a new file `Button.tsx` in your component folder.
|
||||||
</Story>
|
2. Copy and paste the following code into the file.
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
<LoadedCode from={GITHUB_COMP("Button")} />
|
||||||
<LoadedCode from={GITHUB_COMP_PREVIEW("Button")} />
|
|
||||||
</TabContent>
|
## Usage
|
||||||
</TabProvider>
|
|
||||||
|
```tsx
|
||||||
## Installation
|
import { Button } from "@components/Button";
|
||||||
|
```
|
||||||
1. Create a new file `Button.tsx` in your component folder.
|
|
||||||
2. Copy and paste the following code into the file.
|
```html
|
||||||
|
<button>Button</button>
|
||||||
<LoadedCode from={GITHUB_COMP("Button")} />
|
```
|
||||||
|
|
||||||
## Usage
|
## Props
|
||||||
|
|
||||||
```tsx
|
### Variants
|
||||||
import { Button } from "@components/Button";
|
|
||||||
```
|
| Prop | Type | Default | Description |
|
||||||
|
| :----------- | :---------------------------------------------------------------------------- | :---------- | :-------------------------------------- |
|
||||||
```html
|
| `size` | `"link" \| "sm" \| "md" \| "lg" \| "icon"` | `"md"` | The size of the button |
|
||||||
<Button>Button</Button>
|
| `border` | `"none" \| "solid" \| "success" \| "warning" \| "danger"` | `"solid"` | The border color of the button |
|
||||||
```
|
| `background` | `"default" \| "ghost" \| "success" \| "warning" \| "danger" \| "transparent"` | `"default"` | The background color of the button |
|
||||||
|
| `decoration` | `"none" \| "link"` | `"none"` | The inner text decoration of the button |
|
||||||
## Props
|
| `presets` | `"default" \| "ghost" \| "link" \| "success" \| "warning" \| "danger"` | `"default"` | The preset of the variant props |
|
||||||
|
|
||||||
### Variants
|
### Special
|
||||||
|
|
||||||
| Prop | Type | Default | Description |
|
| Prop | Type | Default | Description |
|
||||||
|:-------------|:------------------------------------------------------------------------------|:------------|:----------------------------------------|
|
| :-------- | :-------- | :------ | :------------------------------------------------------- |
|
||||||
| `size` | `"link" \| "sm" \| "md" \| "lg" \| "icon"` | `"md"` | The size of the button |
|
| `asChild` | `boolean` | `false` | Whether the button is rendered as a child of a component |
|
||||||
| `border` | `"none" \| "solid" \| "success" \| "warning" \| "danger"` | `"solid"` | The border color of the button |
|
|
||||||
| `background` | `"default" \| "ghost" \| "success" \| "warning" \| "danger" \| "transparent"` | `"default"` | The background color of the button |
|
## Examples
|
||||||
| `decoration` | `"none" \| "link"` | `"none"` | The inner text decoration of the button |
|
|
||||||
| `presets` | `"default" \| "ghost" \| "link" \| "success" \| "warning" \| "danger"` | `"default"` | The preset of the variant props |
|
### Default
|
||||||
|
|
||||||
### Special
|
<TabProvider defaultName="preview">
|
||||||
|
<TabList>
|
||||||
| Prop | Type | Default | Description |
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
|:----------|:----------|:--------|:---------------------------------------------------------|
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
| `asChild` | `boolean` | `false` | Whether the button is rendered as a child of a component |
|
</TabList>
|
||||||
|
<TabContent name="preview">
|
||||||
## Examples
|
<Story layout="centered">
|
||||||
|
<Examples.Default />
|
||||||
### Default
|
</Story>
|
||||||
|
</TabContent>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="code">
|
||||||
<TabList>
|
<LoadedCode from={GITHUB_STORY("Button", "Default")} />
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
</TabContent>
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</TabProvider>
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
### Ghost
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.Default />
|
<TabProvider defaultName="preview">
|
||||||
</Story>
|
<TabList>
|
||||||
</TabContent>
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
<TabContent name="code">
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
<LoadedCode from={GITHUB_STORY("Button", "Default")} />
|
</TabList>
|
||||||
</TabContent>
|
<TabContent name="preview">
|
||||||
</TabProvider>
|
<Story layout="centered">
|
||||||
|
<Examples.Ghost />
|
||||||
### Ghost
|
</Story>
|
||||||
|
</TabContent>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="code">
|
||||||
<TabList>
|
<LoadedCode from={GITHUB_STORY("Button", "Ghost")} />
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
</TabContent>
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</TabProvider>
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
### Link
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.Ghost />
|
<TabProvider defaultName="preview">
|
||||||
</Story>
|
<TabList>
|
||||||
</TabContent>
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
<TabContent name="code">
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
<LoadedCode from={GITHUB_STORY("Button", "Ghost")} />
|
</TabList>
|
||||||
</TabContent>
|
<TabContent name="preview">
|
||||||
</TabProvider>
|
<Story layout="centered">
|
||||||
|
<Examples.Link />
|
||||||
### Link
|
</Story>
|
||||||
|
</TabContent>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="code">
|
||||||
<TabList>
|
<LoadedCode from={GITHUB_STORY("Button", "Link")} />
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
</TabContent>
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</TabProvider>
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
### Success
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.Link />
|
<TabProvider defaultName="preview">
|
||||||
</Story>
|
<TabList>
|
||||||
</TabContent>
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
<TabContent name="code">
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
<LoadedCode from={GITHUB_STORY("Button", "Link")} />
|
</TabList>
|
||||||
</TabContent>
|
<TabContent name="preview">
|
||||||
</TabProvider>
|
<Story layout="centered">
|
||||||
|
<Examples.Success />
|
||||||
### Success
|
</Story>
|
||||||
|
</TabContent>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="code">
|
||||||
<TabList>
|
<LoadedCode from={GITHUB_STORY("Button", "Success")} />
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
</TabContent>
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</TabProvider>
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
### Warning
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.Success />
|
<TabProvider defaultName="preview">
|
||||||
</Story>
|
<TabList>
|
||||||
</TabContent>
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
<TabContent name="code">
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
<LoadedCode from={GITHUB_STORY("Button", "Success")} />
|
</TabList>
|
||||||
</TabContent>
|
<TabContent name="preview">
|
||||||
</TabProvider>
|
<Story layout="centered">
|
||||||
|
<Examples.Warning />
|
||||||
### Warning
|
</Story>
|
||||||
|
</TabContent>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="code">
|
||||||
<TabList>
|
<LoadedCode from={GITHUB_STORY("Button", "Warning")} />
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
</TabContent>
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</TabProvider>
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
### Danger
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.Warning />
|
<TabProvider defaultName="preview">
|
||||||
</Story>
|
<TabList>
|
||||||
</TabContent>
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
<TabContent name="code">
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
<LoadedCode from={GITHUB_STORY("Button", "Warning")} />
|
</TabList>
|
||||||
</TabContent>
|
<TabContent name="preview">
|
||||||
</TabProvider>
|
<Story layout="centered">
|
||||||
|
<Examples.Danger />
|
||||||
### Danger
|
</Story>
|
||||||
|
</TabContent>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="code">
|
||||||
<TabList>
|
<LoadedCode from={GITHUB_STORY("Button", "Danger")} />
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
</TabContent>
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</TabProvider>
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.Danger />
|
|
||||||
</Story>
|
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
|
||||||
<LoadedCode from={GITHUB_STORY("Button", "Danger")} />
|
|
||||||
</TabContent>
|
|
||||||
</TabProvider>
|
|
||||||
|
64
src/docs/components/ButtonBlocks/Playground.tsx
Normal file
64
src/docs/components/ButtonBlocks/Playground.tsx
Normal file
@ -0,0 +1,64 @@
|
|||||||
|
import type { TEMPLATE } from "@/components/LoadedCode";
|
||||||
|
import { usePgProps } from "@/components/PgHooks";
|
||||||
|
import { PlaygroundLayout } from "@/components/Playground";
|
||||||
|
import { ButtonDemo, type ControlledButtonDemoProps } from "./Preview";
|
||||||
|
|
||||||
|
interface TemplateProps extends TEMPLATE, ControlledButtonDemoProps {}
|
||||||
|
|
||||||
|
export default function ButtonPlayground() {
|
||||||
|
const [props, control] = usePgProps<TemplateProps>({
|
||||||
|
ButtonProps: {
|
||||||
|
preset: {
|
||||||
|
type: "select",
|
||||||
|
options: ["default", "ghost", "link", "success", "warning", "danger"],
|
||||||
|
value: "default",
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
type: "select",
|
||||||
|
options: ["link", "sm", "md", "lg", "icon"],
|
||||||
|
value: "md",
|
||||||
|
disabled: true,
|
||||||
|
},
|
||||||
|
border: {
|
||||||
|
type: "select",
|
||||||
|
options: ["none", "solid", "success", "warning", "danger"],
|
||||||
|
value: "solid",
|
||||||
|
disabled: true,
|
||||||
|
},
|
||||||
|
background: {
|
||||||
|
type: "select",
|
||||||
|
options: [
|
||||||
|
"default",
|
||||||
|
"ghost",
|
||||||
|
"success",
|
||||||
|
"warning",
|
||||||
|
"danger",
|
||||||
|
"transparent",
|
||||||
|
],
|
||||||
|
value: "default",
|
||||||
|
disabled: true,
|
||||||
|
},
|
||||||
|
decoration: {
|
||||||
|
type: "select",
|
||||||
|
options: ["none", "link"],
|
||||||
|
value: "none",
|
||||||
|
disabled: true,
|
||||||
|
},
|
||||||
|
disabled: {
|
||||||
|
type: "boolean",
|
||||||
|
value: false,
|
||||||
|
disabled: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PlaygroundLayout
|
||||||
|
compName="Button"
|
||||||
|
props={props}
|
||||||
|
control={control}
|
||||||
|
>
|
||||||
|
<ButtonDemo {...props} />
|
||||||
|
</PlaygroundLayout>
|
||||||
|
);
|
||||||
|
}
|
@ -1,5 +1,37 @@
|
|||||||
import { Button } from "@pswui/Button";
|
import { Button } from "@pswui/Button";
|
||||||
|
/* remove */
|
||||||
export function ButtonDemo() {
|
export interface ControlledButtonDemoProps {
|
||||||
return <Button>Button</Button>;
|
ButtonProps: {
|
||||||
|
preset?: "default" | "ghost" | "link" | "success" | "warning" | "danger";
|
||||||
|
size?: "link" | "sm" | "md" | "lg" | "icon";
|
||||||
|
border?: "none" | "solid" | "success" | "warning" | "danger";
|
||||||
|
background?:
|
||||||
|
| "default"
|
||||||
|
| "ghost"
|
||||||
|
| "success"
|
||||||
|
| "warning"
|
||||||
|
| "danger"
|
||||||
|
| "transparent";
|
||||||
|
decoration?: "none" | "link";
|
||||||
|
disabled?: boolean;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
/* end */
|
||||||
|
/* replace */
|
||||||
|
export function ButtonDemo({ ButtonProps }: ControlledButtonDemoProps) {
|
||||||
|
/* with
|
||||||
|
export function ButtonDemo() {
|
||||||
|
*/
|
||||||
|
return (
|
||||||
|
<Button
|
||||||
|
preset={ButtonProps.preset}
|
||||||
|
size={ButtonProps.size}
|
||||||
|
border={ButtonProps.border}
|
||||||
|
background={ButtonProps.background}
|
||||||
|
decoration={ButtonProps.decoration}
|
||||||
|
disabled={ButtonProps.disabled}
|
||||||
|
>
|
||||||
|
Button
|
||||||
|
</Button>
|
||||||
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,84 +1,79 @@
|
|||||||
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
||||||
import { Story } from "@/components/Story";
|
import { Story } from "@/components/Story";
|
||||||
import { LoadedCode, GITHUB_COMP, GITHUB_COMP_PREVIEW, GITHUB_STORY } from "@/components/LoadedCode";
|
import {
|
||||||
import { CheckboxDemo } from "./CheckboxBlocks/Preview";
|
LoadedCode,
|
||||||
import Examples from "./CheckboxBlocks/Examples";
|
GITHUB_COMP,
|
||||||
|
GITHUB_COMP_PREVIEW,
|
||||||
# Checkbox
|
GITHUB_STORY,
|
||||||
A control that allows the user to toggle between checked and not checked.
|
} from "@/components/LoadedCode";
|
||||||
|
import Examples from "./CheckboxBlocks/Examples";
|
||||||
<TabProvider defaultName="preview">
|
import Playground from "./CheckboxBlocks/Playground";
|
||||||
<TabList>
|
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
# Checkbox
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
|
||||||
</TabList>
|
A control that allows the user to toggle between checked and not checked.
|
||||||
<TabContent name="preview">
|
|
||||||
<Story layout="centered">
|
## Playground
|
||||||
<CheckboxDemo />
|
|
||||||
</Story>
|
<Playground />
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
## Installation
|
||||||
<LoadedCode from={GITHUB_COMP_PREVIEW("Checkbox")} />
|
|
||||||
</TabContent>
|
1. Create a new file `Checkbox.mdx` in your component folder.
|
||||||
</TabProvider>
|
2. Copy and paste the following code into the file.
|
||||||
|
|
||||||
## Installation
|
<LoadedCode from={GITHUB_COMP("Checkbox")} />
|
||||||
|
|
||||||
1. Create a new file `Checkbox.mdx` in your component folder.
|
## Usage
|
||||||
2. Copy and paste the following code into the file.
|
|
||||||
|
```tsx
|
||||||
<LoadedCode from={GITHUB_COMP("Checkbox")} />
|
import { Checkbox } from "@components/Checkbox";
|
||||||
|
```
|
||||||
## Usage
|
|
||||||
|
```html
|
||||||
```tsx
|
<Checkbox />
|
||||||
import { Checkbox } from "@components/Checkbox";
|
```
|
||||||
```
|
|
||||||
|
## Props
|
||||||
```html
|
|
||||||
<Checkbox />
|
### Variants
|
||||||
```
|
|
||||||
|
| Prop | Type | Default | Description |
|
||||||
## Props
|
| :----- | :----------------------- | :------ | :----------------------- |
|
||||||
|
| `size` | `"base" \| "md" \| "lg"` | `"md"` | The size of the checkbox |
|
||||||
### Variants
|
|
||||||
|
## Examples
|
||||||
| Prop | Type | Default | Description |
|
|
||||||
|:-------|:-------------------------|:--------|:-------------------------|
|
### Text
|
||||||
| `size` | `"base" \| "md" \| "lg"` | `"md"` | The size of the checkbox |
|
|
||||||
|
<TabProvider defaultName="preview">
|
||||||
## Examples
|
<TabList>
|
||||||
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
### Text
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
|
</TabList>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="preview">
|
||||||
<TabList>
|
<Story layout="centered">
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
<Examples.Text />
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</Story>
|
||||||
</TabList>
|
</TabContent>
|
||||||
<TabContent name="preview">
|
<TabContent name="code">
|
||||||
<Story layout="centered">
|
<LoadedCode from={GITHUB_STORY("Checkbox", "Text")} />
|
||||||
<Examples.Text />
|
</TabContent>
|
||||||
</Story>
|
</TabProvider>
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
### Disabled
|
||||||
<LoadedCode from={GITHUB_STORY("Checkbox", "Text")} />
|
|
||||||
</TabContent>
|
<TabProvider defaultName="preview">
|
||||||
</TabProvider>
|
<TabList>
|
||||||
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
### Disabled
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
|
</TabList>
|
||||||
<TabProvider defaultName="preview">
|
<TabContent name="preview">
|
||||||
<TabList>
|
<Story layout="centered">
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
<Examples.Disabled />
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
</Story>
|
||||||
</TabList>
|
</TabContent>
|
||||||
<TabContent name="preview">
|
<TabContent name="code">
|
||||||
<Story layout="centered">
|
<LoadedCode from={GITHUB_STORY("Checkbox", "Disabled")} />
|
||||||
<Examples.Disabled />
|
</TabContent>
|
||||||
</Story>
|
</TabProvider>
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
|
||||||
<LoadedCode from={GITHUB_STORY("Checkbox", "Disabled")} />
|
|
||||||
</TabContent>
|
|
||||||
</TabProvider>
|
|
||||||
|
28
src/docs/components/CheckboxBlocks/Playground.tsx
Normal file
28
src/docs/components/CheckboxBlocks/Playground.tsx
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
import type { TEMPLATE } from "@/components/LoadedCode";
|
||||||
|
import { usePgProps } from "@/components/PgHooks";
|
||||||
|
import { PlaygroundLayout } from "@/components/Playground";
|
||||||
|
import { CheckboxDemo, type CheckboxDemoPlaygroundProps } from "./Preview";
|
||||||
|
|
||||||
|
interface TemplateProps extends TEMPLATE, CheckboxDemoPlaygroundProps {}
|
||||||
|
|
||||||
|
export default function CheckboxPlayground() {
|
||||||
|
const [props, control] = usePgProps<TemplateProps>({
|
||||||
|
CheckboxProps: {
|
||||||
|
size: {
|
||||||
|
type: "select",
|
||||||
|
options: ["base", "md", "lg"],
|
||||||
|
value: "md",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PlaygroundLayout
|
||||||
|
compName="Checkbox"
|
||||||
|
props={props}
|
||||||
|
control={control}
|
||||||
|
>
|
||||||
|
<CheckboxDemo {...props} />
|
||||||
|
</PlaygroundLayout>
|
||||||
|
);
|
||||||
|
}
|
@ -1,10 +1,20 @@
|
|||||||
import { Checkbox } from "@pswui/Checkbox";
|
import { Checkbox } from "@pswui/Checkbox";
|
||||||
import { Label } from "@pswui/Label";
|
import { Label } from "@pswui/Label";
|
||||||
|
/* remove */
|
||||||
|
export interface CheckboxDemoPlaygroundProps {
|
||||||
|
CheckboxProps: {
|
||||||
|
size: "base" | "md" | "lg";
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/* replace */
|
||||||
|
export function CheckboxDemo({ CheckboxProps }: CheckboxDemoPlaygroundProps) {
|
||||||
|
/* with
|
||||||
export function CheckboxDemo() {
|
export function CheckboxDemo() {
|
||||||
|
*/
|
||||||
return (
|
return (
|
||||||
<Label direction="horizontal">
|
<Label direction="horizontal">
|
||||||
<Checkbox />
|
<Checkbox size={CheckboxProps.size} />
|
||||||
<span>Checkbox</span>
|
<span>Checkbox</span>
|
||||||
</Label>
|
</Label>
|
||||||
);
|
);
|
||||||
|
@ -1,181 +1,121 @@
|
|||||||
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
||||||
import { Story } from "@/components/Story";
|
import { Story } from "@/components/Story";
|
||||||
import { LoadedCode, GITHUB_DIR_COMP, GITHUB_COMP_PREVIEW, GITHUB_STORY } from "@/components/LoadedCode";
|
import {
|
||||||
import { DialogDemo } from "./DialogBlocks/Preview";
|
LoadedCode,
|
||||||
import Examples from "./DialogBlocks/Examples";
|
GITHUB_DIR_COMP,
|
||||||
|
GITHUB_COMP_PREVIEW,
|
||||||
# Dialog
|
GITHUB_STORY,
|
||||||
A modal window that prompts the user to take an action or provides critical information.
|
} from "@/components/LoadedCode";
|
||||||
|
import { DialogDemo } from "./DialogBlocks/Preview";
|
||||||
<TabProvider defaultName="preview">
|
import Examples from "./DialogBlocks/Examples";
|
||||||
<TabList>
|
import Playground from "./DialogBlocks/Playground";
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
# Dialog
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
A modal window that prompts the user to take an action or provides critical information.
|
||||||
<Story layout="centered">
|
|
||||||
<DialogDemo />
|
<Playground />
|
||||||
</Story>
|
|
||||||
</TabContent>
|
## Installation
|
||||||
<TabContent name="code">
|
|
||||||
<LoadedCode from={GITHUB_COMP_PREVIEW("Dialog")} />
|
1. Create a new directory named `Dialog` in your component folder.
|
||||||
</TabContent>
|
2. Create following files in the folder, and paste the code into the file.
|
||||||
</TabProvider>
|
|
||||||
|
- `index.ts`
|
||||||
## Installation
|
<LoadedCode from={GITHUB_DIR_COMP("Dialog", "index.ts")} />
|
||||||
|
- `Context.ts`
|
||||||
1. Create a new directory named `Dialog` in your component folder.
|
<LoadedCode from={GITHUB_DIR_COMP("Dialog", "Context.ts")} />
|
||||||
2. Create following files in the folder, and paste the code into the file.
|
- `Component.tsx`
|
||||||
* `index.ts`
|
<LoadedCode from={GITHUB_DIR_COMP("Dialog", "Component.tsx")} />
|
||||||
<LoadedCode from={GITHUB_DIR_COMP("Dialog", "index.ts")} />
|
|
||||||
* `Context.ts`
|
## Usage
|
||||||
<LoadedCode from={GITHUB_DIR_COMP("Dialog", "Context.ts")} />
|
|
||||||
* `Component.tsx`
|
```tsx
|
||||||
<LoadedCode from={GITHUB_DIR_COMP("Dialog", "Component.tsx")} />
|
import {
|
||||||
|
DialogRoot,
|
||||||
## Usage
|
DialogTrigger,
|
||||||
|
DialogOverlay,
|
||||||
```tsx
|
DialogContent,
|
||||||
import {
|
DialogHeader,
|
||||||
DialogRoot,
|
DialogTitle,
|
||||||
DialogTrigger,
|
DialogSubtitle,
|
||||||
DialogOverlay,
|
DialogFooter,
|
||||||
DialogContent,
|
DialogClose,
|
||||||
DialogHeader,
|
} from "@components/Dialog";
|
||||||
DialogTitle,
|
```
|
||||||
DialogSubtitle,
|
|
||||||
DialogFooter,
|
```html
|
||||||
DialogClose,
|
<DialogRoot>
|
||||||
} from "@components/Dialog";
|
<DialogTrigger>
|
||||||
```
|
<button>Open Dialog</button>
|
||||||
|
</DialogTrigger>
|
||||||
```html
|
<DialogOverlay>
|
||||||
<DialogRoot>
|
<DialogContent>
|
||||||
<DialogTrigger>
|
<DialogHeader>
|
||||||
<Button>Open Dialog</Button>
|
<DialogTitle>Dialog Title</DialogTitle>
|
||||||
</DialogTrigger>
|
<DialogSubtitle>Dialog Subtitle</DialogSubtitle>
|
||||||
<DialogOverlay>
|
</DialogHeader>
|
||||||
<DialogContent>
|
{/* Main Contents */}
|
||||||
<DialogHeader>
|
<DialogFooter>
|
||||||
<DialogTitle>Dialog Title</DialogTitle>
|
<DialogClose>
|
||||||
<DialogSubtitle>Dialog Subtitle</DialogSubtitle>
|
<button>Close</button>
|
||||||
</DialogHeader>
|
</DialogClose>
|
||||||
{/* Main Contents */}
|
</DialogFooter>
|
||||||
<DialogFooter>
|
</DialogContent>
|
||||||
<DialogClose>
|
</DialogOverlay>
|
||||||
<Button>Close</Button>
|
</DialogRoot>
|
||||||
</DialogClose>
|
```
|
||||||
</DialogFooter>
|
|
||||||
</DialogContent>
|
> Note:
|
||||||
</DialogOverlay>
|
>
|
||||||
</DialogRoot>
|
> DialogTrigger and DialogClose 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 DialogTrigger and DialogClose.
|
||||||
> Note:
|
>
|
||||||
>
|
> It is easier to understand if you think of this component as always having the `asChild` prop applied to it.
|
||||||
> DialogTrigger and DialogClose will merge its onClick event handler to its children.
|
|
||||||
> Also, there is no default element for those.
|
## Props
|
||||||
> So you always have to provide the clickable children for DialogTrigger and DialogClose.
|
|
||||||
>
|
### DialogOverlay
|
||||||
> It is easier to understand if you think of this component as always having the `asChild` prop applied to it.
|
|
||||||
|
#### Special
|
||||||
## Props
|
|
||||||
|
| Prop | Type | Default | Description |
|
||||||
### DialogOverlay
|
| :------------- | :-------- | :------ | :--------------------------------------------- |
|
||||||
|
| `closeOnClick` | `boolean` | `false` | Whether the dialog will be closed when clicked |
|
||||||
#### Variants
|
|
||||||
|
## Examples
|
||||||
| Prop | Type | Default | Description |
|
|
||||||
|:----------|:-----------------------|:--------|:---------------------------------------------|
|
### Basic Informational Dialog
|
||||||
| `blur` | `"sm" \| "md" \| "lg"` | `md` | Whether the background of dialog is blurred |
|
|
||||||
| `darken` | `"sm" \| "md" \| "lg"` | `md` | Whether the background of dialog is darkened |
|
<TabProvider defaultName="preview">
|
||||||
| `padding` | `"sm" \| "md" \| "lg"` | `md` | Minimum margin of the dialog |
|
<TabList>
|
||||||
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
#### Special
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
|
</TabList>
|
||||||
| Prop | Type | Default | Description |
|
<TabContent name="preview">
|
||||||
|:---------------|:----------|:--------|:-----------------------------------------------|
|
<Story layout="centered">
|
||||||
| `closeOnClick` | `boolean` | `false` | Whether the dialog will be closed when clicked |
|
<Examples.BasicInformationalDialog />
|
||||||
|
</Story>
|
||||||
### DialogContent
|
</TabContent>
|
||||||
|
<TabContent name="code">
|
||||||
#### Variants
|
<LoadedCode from={GITHUB_STORY("Dialog", "BasicInformationalDialog")} />
|
||||||
|
</TabContent>
|
||||||
| Prop | Type | Default | Description |
|
</TabProvider>
|
||||||
|:----------|:---------------------------------------------------------------------|:--------|:-----------------------------------------------|
|
|
||||||
| `size` | `"fit" \| "fullSm" \| "fullMd" \| "fullLg" \| "fullXl" \| "full2xl"` | `fit` | Size of the dialog - width and max width |
|
### Deleting Item
|
||||||
| `rounded` | `"sm" \| "md" \| "lg" \| "xl"` | `md` | Roundness of the dialog |
|
|
||||||
| `padding` | `"sm" \| "md" \| "lg"` | `md` | Padding of the dialog |
|
<TabProvider defaultName="preview">
|
||||||
| `gap` | `"sm" \| "md" \| "lg"` | `md` | Works like flex's gap - space between children |
|
<TabList>
|
||||||
|
<TabTrigger name="preview">Preview</TabTrigger>
|
||||||
### DialogHeader
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
|
</TabList>
|
||||||
#### Variants
|
<TabContent name="preview">
|
||||||
|
<Story layout="centered">
|
||||||
| Prop | Type | Default | Description |
|
<Examples.DeletingItem />
|
||||||
|:------|:-----------------------|:--------|:----------------------------------------------|
|
</Story>
|
||||||
| `gap` | `"sm" \| "md" \| "lg"` | `sm` | Gap between the children - title and subtitle |
|
</TabContent>
|
||||||
|
<TabContent name="code">
|
||||||
### DialogTitle
|
<LoadedCode from={GITHUB_STORY("Dialog", "DeletingItem")} />
|
||||||
|
</TabContent>
|
||||||
#### Variants
|
</TabProvider>
|
||||||
|
|
||||||
| Prop | Type | Default | Description |
|
|
||||||
|:---------|:-----------------------|:--------|:--------------------|
|
|
||||||
| `size` | `"sm" \| "md" \| "lg"` | `md` | Size of the title |
|
|
||||||
| `weight` | `"sm" \| "md" \| "lg"` | `lg` | Weight of the title |
|
|
||||||
|
|
||||||
### DialogSubtitle
|
|
||||||
|
|
||||||
#### Variants
|
|
||||||
|
|
||||||
| Prop | Type | Default | Description |
|
|
||||||
|:----------|:-----------------------|:--------|:------------------------|
|
|
||||||
| `size` | `"sm" \| "md" \| "lg"` | `sm` | Size of the subtitle |
|
|
||||||
| `weight` | `"sm" \| "md" \| "lg"` | `md` | Weight of the subtitle |
|
|
||||||
| `opacity` | `"sm" \| "md" \| "lg"` | `sm` | Opacity of the subtitle |
|
|
||||||
|
|
||||||
### DialogFooter
|
|
||||||
|
|
||||||
#### Variants
|
|
||||||
|
|
||||||
| Prop | Type | Default | Description |
|
|
||||||
|:------|:-----------------------|:--------|:-------------------------|
|
|
||||||
| `gap` | `"sm" \| "md" \| "lg"` | `sm` | Gap between the children |
|
|
||||||
|
|
||||||
## Examples
|
|
||||||
|
|
||||||
### Basic Informational Dialog
|
|
||||||
|
|
||||||
<TabProvider defaultName="preview">
|
|
||||||
<TabList>
|
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.BasicInformationalDialog />
|
|
||||||
</Story>
|
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
|
||||||
<LoadedCode from={GITHUB_STORY("Dialog", "BasicInformationalDialog")} />
|
|
||||||
</TabContent>
|
|
||||||
</TabProvider>
|
|
||||||
|
|
||||||
### Deleting Item
|
|
||||||
|
|
||||||
<TabProvider defaultName="preview">
|
|
||||||
<TabList>
|
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
|
||||||
<Story layout="centered">
|
|
||||||
<Examples.DeletingItem />
|
|
||||||
</Story>
|
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
|
||||||
<LoadedCode from={GITHUB_STORY("Dialog", "DeletingItem")} />
|
|
||||||
</TabContent>
|
|
||||||
</TabProvider>
|
|
||||||
|
|
||||||
|
@ -18,7 +18,7 @@ export function BasicInformationalDialog() {
|
|||||||
<Button preset="default">What is this?</Button>
|
<Button preset="default">What is this?</Button>
|
||||||
</DialogTrigger>
|
</DialogTrigger>
|
||||||
<DialogOverlay>
|
<DialogOverlay>
|
||||||
<DialogContent size={"fullMd"}>
|
<DialogContent>
|
||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle>Dialog Title</DialogTitle>
|
<DialogTitle>Dialog Title</DialogTitle>
|
||||||
<DialogSubtitle>Dialog Subtitle</DialogSubtitle>
|
<DialogSubtitle>Dialog Subtitle</DialogSubtitle>
|
||||||
|
@ -21,7 +21,7 @@ export function DeletingItem() {
|
|||||||
<Button preset="danger">Delete Item</Button>
|
<Button preset="danger">Delete Item</Button>
|
||||||
</DialogTrigger>
|
</DialogTrigger>
|
||||||
<DialogOverlay>
|
<DialogOverlay>
|
||||||
<DialogContent size={"fullMd"}>
|
<DialogContent>
|
||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle>Delete Item</DialogTitle>
|
<DialogTitle>Delete Item</DialogTitle>
|
||||||
<DialogSubtitle>
|
<DialogSubtitle>
|
||||||
|
27
src/docs/components/DialogBlocks/Playground.tsx
Normal file
27
src/docs/components/DialogBlocks/Playground.tsx
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
import type { TEMPLATE } from "@/components/LoadedCode";
|
||||||
|
import { usePgProps } from "@/components/PgHooks";
|
||||||
|
import { PlaygroundLayout } from "@/components/Playground";
|
||||||
|
import { DialogDemo, type DialogDemoPlaygroundProps } from "./Preview";
|
||||||
|
|
||||||
|
interface TemplateProps extends TEMPLATE, DialogDemoPlaygroundProps {}
|
||||||
|
|
||||||
|
export default function DialogPlayground() {
|
||||||
|
const [props, control] = usePgProps<TemplateProps>({
|
||||||
|
DialogOverlayProps: {
|
||||||
|
closeOnClick: {
|
||||||
|
type: "boolean",
|
||||||
|
value: false,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PlaygroundLayout
|
||||||
|
compName="Dialog"
|
||||||
|
control={control}
|
||||||
|
props={props}
|
||||||
|
>
|
||||||
|
<DialogDemo {...props} />
|
||||||
|
</PlaygroundLayout>
|
||||||
|
);
|
||||||
|
}
|
@ -10,15 +10,25 @@ import {
|
|||||||
DialogTitle,
|
DialogTitle,
|
||||||
DialogTrigger,
|
DialogTrigger,
|
||||||
} from "@pswui/Dialog";
|
} from "@pswui/Dialog";
|
||||||
|
/* remove */
|
||||||
|
export interface DialogDemoPlaygroundProps {
|
||||||
|
DialogOverlayProps: {
|
||||||
|
closeOnClick: boolean;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
/* end */
|
||||||
|
/* replace */
|
||||||
|
export function DialogDemo({ DialogOverlayProps }: DialogDemoPlaygroundProps) {
|
||||||
|
/* with
|
||||||
export function DialogDemo() {
|
export function DialogDemo() {
|
||||||
|
*/
|
||||||
return (
|
return (
|
||||||
<DialogRoot>
|
<DialogRoot>
|
||||||
<DialogTrigger>
|
<DialogTrigger>
|
||||||
<Button preset="default">Open Dialog</Button>
|
<Button preset="default">Open Dialog</Button>
|
||||||
</DialogTrigger>
|
</DialogTrigger>
|
||||||
<DialogOverlay>
|
<DialogOverlay closeOnClick={DialogOverlayProps.closeOnClick}>
|
||||||
<DialogContent size={"fullMd"}>
|
<DialogContent>
|
||||||
<DialogHeader>
|
<DialogHeader>
|
||||||
<DialogTitle>Dialog Title</DialogTitle>
|
<DialogTitle>Dialog Title</DialogTitle>
|
||||||
<DialogSubtitle>Dialog Subtitle</DialogSubtitle>
|
<DialogSubtitle>Dialog Subtitle</DialogSubtitle>
|
||||||
|
@ -1,26 +1,13 @@
|
|||||||
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
import { TabProvider, TabTrigger, TabContent, TabList } from "@pswui/Tabs";
|
||||||
import { Story } from "@/components/Story";
|
import { Story } from "@/components/Story";
|
||||||
import { LoadedCode, GITHUB_COMP, GITHUB_COMP_PREVIEW, GITHUB_STORY } from '@/components/LoadedCode';
|
import { LoadedCode, GITHUB_COMP, GITHUB_COMP_PREVIEW, GITHUB_STORY } from '@/components/LoadedCode';
|
||||||
import { DrawerDemo } from "./DrawerBlocks/Preview";
|
|
||||||
import Examples from "./DrawerBlocks/Examples";
|
import Examples from "./DrawerBlocks/Examples";
|
||||||
|
import Playground from "./DrawerBlocks/Playground";
|
||||||
|
|
||||||
# Drawer
|
# Drawer
|
||||||
Displays a panel that slides out from the edge of the screen, typically used for navigation or additional content.
|
Displays a panel that slides out from the edge of the screen, typically used for navigation or additional content.
|
||||||
|
|
||||||
<TabProvider defaultName="preview">
|
<Playground />
|
||||||
<TabList>
|
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
|
||||||
</TabList>
|
|
||||||
<TabContent name="preview">
|
|
||||||
<Story layout="centered">
|
|
||||||
<DrawerDemo />
|
|
||||||
</Story>
|
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
|
||||||
<LoadedCode from={GITHUB_COMP_PREVIEW("Drawer")} />
|
|
||||||
</TabContent>
|
|
||||||
</TabProvider>
|
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
|
|
||||||
@ -41,7 +28,7 @@ import {
|
|||||||
DrawerHeader,
|
DrawerHeader,
|
||||||
DrawerBody,
|
DrawerBody,
|
||||||
DrawerFooter,
|
DrawerFooter,
|
||||||
} from "@components/Drawer";
|
} from "@components/Drawer";
|
||||||
```
|
```
|
||||||
|
|
||||||
```html
|
```html
|
||||||
@ -68,7 +55,7 @@ import {
|
|||||||
```
|
```
|
||||||
|
|
||||||
> Note:
|
> Note:
|
||||||
>
|
>
|
||||||
> DrawerTrigger and DrawerClose will merge its onClick event handler to its children.
|
> DrawerTrigger and DrawerClose will merge its onClick event handler to its children.
|
||||||
> Also, there is no default element for those.
|
> Also, there is no default element for those.
|
||||||
> So you always have to provide the clickable children for DialogTrigger and DialogClose.
|
> So you always have to provide the clickable children for DialogTrigger and DialogClose.
|
||||||
@ -98,9 +85,10 @@ import {
|
|||||||
|
|
||||||
#### Variants
|
#### Variants
|
||||||
|
|
||||||
| Prop | Type | Default | Description |
|
| Prop | Type | Default | Description |
|
||||||
|:-----------|:-----------------------------------------|:---------|:---------------------------|
|
|:-----------|:-----------------------------------------|:---------|:--------------------------------------------|
|
||||||
| `position` | `"top" \| "bottom" \| "left" \| "right"` | `"left"` | The position of the drawer |
|
| `position` | `"top" \| "bottom" \| "left" \| "right"` | `"left"` | The position of the drawer |
|
||||||
|
| `maxSize` | `"sm" \| "md" \| "lg" \| "xl"` | `"sm"` | max width ignored on top or bottom position |
|
||||||
|
|
||||||
#### Special
|
#### Special
|
||||||
|
|
||||||
|
@ -16,7 +16,7 @@ export const Bottom = () => {
|
|||||||
<DrawerTrigger>
|
<DrawerTrigger>
|
||||||
<Button>Open Drawer</Button>
|
<Button>Open Drawer</Button>
|
||||||
</DrawerTrigger>
|
</DrawerTrigger>
|
||||||
<DrawerOverlay className="z-[99]">
|
<DrawerOverlay className="z-99">
|
||||||
<DrawerContent position="bottom">
|
<DrawerContent position="bottom">
|
||||||
<DrawerHeader>
|
<DrawerHeader>
|
||||||
<h1 className="text-2xl font-bold">Drawer</h1>
|
<h1 className="text-2xl font-bold">Drawer</h1>
|
||||||
|
@ -16,7 +16,7 @@ export const Left = () => {
|
|||||||
<DrawerTrigger>
|
<DrawerTrigger>
|
||||||
<Button>Open Drawer</Button>
|
<Button>Open Drawer</Button>
|
||||||
</DrawerTrigger>
|
</DrawerTrigger>
|
||||||
<DrawerOverlay className="z-[99]">
|
<DrawerOverlay className="z-99">
|
||||||
<DrawerContent
|
<DrawerContent
|
||||||
position="left"
|
position="left"
|
||||||
className="max-w-[320px]"
|
className="max-w-[320px]"
|
||||||
|
@ -16,7 +16,7 @@ export const Right = () => {
|
|||||||
<DrawerTrigger>
|
<DrawerTrigger>
|
||||||
<Button>Open Drawer</Button>
|
<Button>Open Drawer</Button>
|
||||||
</DrawerTrigger>
|
</DrawerTrigger>
|
||||||
<DrawerOverlay className="z-[99]">
|
<DrawerOverlay className="z-99">
|
||||||
<DrawerContent
|
<DrawerContent
|
||||||
position="right"
|
position="right"
|
||||||
className="max-w-[320px]"
|
className="max-w-[320px]"
|
||||||
|
@ -16,7 +16,7 @@ export const Top = () => {
|
|||||||
<DrawerTrigger>
|
<DrawerTrigger>
|
||||||
<Button>Open Drawer</Button>
|
<Button>Open Drawer</Button>
|
||||||
</DrawerTrigger>
|
</DrawerTrigger>
|
||||||
<DrawerOverlay className="z-[99]">
|
<DrawerOverlay className="z-99">
|
||||||
<DrawerContent position="top">
|
<DrawerContent position="top">
|
||||||
<DrawerHeader>
|
<DrawerHeader>
|
||||||
<h1 className="text-2xl font-bold">Drawer</h1>
|
<h1 className="text-2xl font-bold">Drawer</h1>
|
||||||
|
39
src/docs/components/DrawerBlocks/Playground.tsx
Normal file
39
src/docs/components/DrawerBlocks/Playground.tsx
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
import type { TEMPLATE } from "@/components/LoadedCode";
|
||||||
|
import { usePgProps } from "@/components/PgHooks";
|
||||||
|
import { PlaygroundLayout } from "@/components/Playground";
|
||||||
|
import { DrawerDemo, type DrawerDemoPlaygroundProps } from "./Preview";
|
||||||
|
|
||||||
|
interface TemplateProps extends TEMPLATE, DrawerDemoPlaygroundProps {}
|
||||||
|
|
||||||
|
export default function DrawerPlayground() {
|
||||||
|
const [props, control] = usePgProps<TemplateProps>({
|
||||||
|
DrawerRootProps: {
|
||||||
|
closeThreshold: {
|
||||||
|
type: "number",
|
||||||
|
value: 0.3,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
DrawerContentProps: {
|
||||||
|
position: {
|
||||||
|
type: "select",
|
||||||
|
options: ["top", "bottom", "left", "right"],
|
||||||
|
value: "left",
|
||||||
|
},
|
||||||
|
maxSize: {
|
||||||
|
type: "select",
|
||||||
|
options: ["sm", "md", "lg", "xl"],
|
||||||
|
value: "sm",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<PlaygroundLayout
|
||||||
|
compName="Drawer"
|
||||||
|
control={control}
|
||||||
|
props={props}
|
||||||
|
>
|
||||||
|
<DrawerDemo {...props} />
|
||||||
|
</PlaygroundLayout>
|
||||||
|
);
|
||||||
|
}
|
@ -9,15 +9,35 @@ import {
|
|||||||
DrawerRoot,
|
DrawerRoot,
|
||||||
DrawerTrigger,
|
DrawerTrigger,
|
||||||
} from "@pswui/Drawer";
|
} from "@pswui/Drawer";
|
||||||
|
/* remove */
|
||||||
export const DrawerDemo = () => {
|
export interface DrawerDemoPlaygroundProps {
|
||||||
|
DrawerRootProps: {
|
||||||
|
closeThreshold: number;
|
||||||
|
};
|
||||||
|
DrawerContentProps: {
|
||||||
|
position: "top" | "bottom" | "left" | "right";
|
||||||
|
maxSize: "sm" | "md" | "lg" | "xl";
|
||||||
|
};
|
||||||
|
}
|
||||||
|
/* end */
|
||||||
|
/* replace */
|
||||||
|
export function DrawerDemo({
|
||||||
|
DrawerRootProps,
|
||||||
|
DrawerContentProps,
|
||||||
|
}: DrawerDemoPlaygroundProps) {
|
||||||
|
/* with
|
||||||
|
export function DrawerDemo() {
|
||||||
|
*/
|
||||||
return (
|
return (
|
||||||
<DrawerRoot>
|
<DrawerRoot closeThreshold={DrawerRootProps.closeThreshold}>
|
||||||
<DrawerTrigger>
|
<DrawerTrigger>
|
||||||
<Button>Open Drawer</Button>
|
<Button>Open Drawer</Button>
|
||||||
</DrawerTrigger>
|
</DrawerTrigger>
|
||||||
<DrawerOverlay className="z-[99]">
|
<DrawerOverlay className="z-99">
|
||||||
<DrawerContent className="max-w-[320px]">
|
<DrawerContent
|
||||||
|
position={DrawerContentProps.position}
|
||||||
|
maxSize={DrawerContentProps.maxSize}
|
||||||
|
>
|
||||||
<DrawerHeader>
|
<DrawerHeader>
|
||||||
<h1 className="text-2xl font-bold">Drawer</h1>
|
<h1 className="text-2xl font-bold">Drawer</h1>
|
||||||
</DrawerHeader>
|
</DrawerHeader>
|
||||||
@ -37,4 +57,4 @@ export const DrawerDemo = () => {
|
|||||||
</DrawerOverlay>
|
</DrawerOverlay>
|
||||||
</DrawerRoot>
|
</DrawerRoot>
|
||||||
);
|
);
|
||||||
};
|
}
|
||||||
|
@ -51,7 +51,12 @@ export const ThemeSelector = () => {
|
|||||||
{theme === "light" ? <LightIcon /> : <DarkIcon />}
|
{theme === "light" ? <LightIcon /> : <DarkIcon />}
|
||||||
</Button>
|
</Button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent anchor={"bottomCenter"}>
|
<PopoverContent
|
||||||
|
direction={"col"}
|
||||||
|
position={"end"}
|
||||||
|
anchor={"middle"}
|
||||||
|
align={"middle"}
|
||||||
|
>
|
||||||
<Button
|
<Button
|
||||||
onClick={() => setTheme("dark")}
|
onClick={() => setTheme("dark")}
|
||||||
preset={"ghost"}
|
preset={"ghost"}
|
||||||
|
@ -77,7 +77,8 @@ const SignInForm = () => {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<PopoverContent
|
<PopoverContent
|
||||||
anchor={"bottomLeft"}
|
anchor={"end"}
|
||||||
|
align={"end"}
|
||||||
className={"p-4 space-y-3"}
|
className={"p-4 space-y-3"}
|
||||||
>
|
>
|
||||||
<Label>
|
<Label>
|
||||||
@ -127,7 +128,10 @@ const UserControlContent = () => {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<PopoverContent anchor={"bottomLeft"}>
|
<PopoverContent
|
||||||
|
anchor={"end"}
|
||||||
|
align={"end"}
|
||||||
|
>
|
||||||
<Button preset={"ghost"}>Dashboard</Button>
|
<Button preset={"ghost"}>Dashboard</Button>
|
||||||
<Button
|
<Button
|
||||||
preset={"ghost"}
|
preset={"ghost"}
|
||||||
|
@ -1,328 +1,58 @@
|
|||||||
import { GITHUB_COMP_PREVIEW, LoadedCode } from "@/components/LoadedCode.tsx";
|
import type { TEMPLATE } from "@/components/LoadedCode.tsx";
|
||||||
import { Story } from "@/components/Story";
|
import { usePgProps } from "@/components/PgHooks.tsx";
|
||||||
import useMutable from "@/utils/useMutable.ts";
|
import { PlaygroundLayout } from "@/components/Playground.tsx";
|
||||||
import { Button } from "@pswui/Button.tsx";
|
|
||||||
import { Checkbox } from "@pswui/Checkbox.tsx";
|
|
||||||
import { Label } from "@pswui/Label.tsx";
|
|
||||||
import { Popover, PopoverContent, PopoverTrigger } from "@pswui/Popover.tsx";
|
|
||||||
import { TabContent, TabList, TabProvider, TabTrigger } from "@pswui/Tabs";
|
|
||||||
import { type ControlledPopoverDemoProps, PopoverDemo } from "./Preview.tsx";
|
import { type ControlledPopoverDemoProps, PopoverDemo } from "./Preview.tsx";
|
||||||
|
|
||||||
|
interface TemplateProps extends TEMPLATE, ControlledPopoverDemoProps {}
|
||||||
|
|
||||||
export default function PopoverPlayground() {
|
export default function PopoverPlayground() {
|
||||||
const [props, mutate] = useMutable<ControlledPopoverDemoProps>({
|
const [props, control] = usePgProps<TemplateProps>({
|
||||||
PopoverProps: {
|
PopoverProps: {
|
||||||
opened: false,
|
opened: {
|
||||||
|
type: "boolean",
|
||||||
|
value: false,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
PopoverContentProps: {
|
PopoverContentProps: {
|
||||||
direction: "col",
|
direction: {
|
||||||
position: "end",
|
type: "select",
|
||||||
anchor: "middle",
|
options: ["col", "row"],
|
||||||
align: "middle",
|
value: "col",
|
||||||
offset: "md",
|
},
|
||||||
className: "",
|
position: {
|
||||||
|
type: "select",
|
||||||
|
options: ["start", "end"],
|
||||||
|
value: "end",
|
||||||
|
},
|
||||||
|
anchor: {
|
||||||
|
type: "select",
|
||||||
|
options: ["start", "middle", "end"],
|
||||||
|
value: "middle",
|
||||||
|
},
|
||||||
|
align: {
|
||||||
|
type: "select",
|
||||||
|
options: ["start", "middle", "end"],
|
||||||
|
value: "middle",
|
||||||
|
},
|
||||||
|
offset: {
|
||||||
|
type: "select",
|
||||||
|
options: ["sm", "md", "lg"],
|
||||||
|
value: "md",
|
||||||
|
},
|
||||||
|
className: {
|
||||||
|
type: "string",
|
||||||
|
value: "",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<PlaygroundLayout
|
||||||
<TabProvider defaultName="preview">
|
compName="Popover"
|
||||||
<TabList>
|
props={props}
|
||||||
<TabTrigger name="preview">Preview</TabTrigger>
|
control={control}
|
||||||
<TabTrigger name="code">Code</TabTrigger>
|
>
|
||||||
</TabList>
|
<PopoverDemo {...props} />
|
||||||
<TabContent name="preview">
|
</PlaygroundLayout>
|
||||||
<Story layout="centered">
|
|
||||||
<PopoverDemo {...props} />
|
|
||||||
</Story>
|
|
||||||
</TabContent>
|
|
||||||
<TabContent name="code">
|
|
||||||
<LoadedCode from={GITHUB_COMP_PREVIEW("Popover")} />
|
|
||||||
</TabContent>
|
|
||||||
</TabProvider>
|
|
||||||
<h3>Controls</h3>
|
|
||||||
<div
|
|
||||||
className={
|
|
||||||
"rounded-lg p-4 border border-neutral-300 dark:border-neutral-700 flex flex-col justify-center items-start gap-2"
|
|
||||||
}
|
|
||||||
>
|
|
||||||
<Label direction={"horizontal"}>
|
|
||||||
<span>opened </span>
|
|
||||||
<Checkbox
|
|
||||||
checked={props.PopoverProps.opened}
|
|
||||||
onChange={(e) => {
|
|
||||||
const v = e.currentTarget.checked;
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverProps.opened = v;
|
|
||||||
});
|
|
||||||
}}
|
|
||||||
/>
|
|
||||||
</Label>
|
|
||||||
<Label direction={"horizontal"}>
|
|
||||||
<span>direction = </span>
|
|
||||||
<Popover>
|
|
||||||
<PopoverTrigger>
|
|
||||||
<Button className={"gap-2"}>
|
|
||||||
{props.PopoverContentProps.direction}
|
|
||||||
<svg
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="1em"
|
|
||||||
height="1em"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<title>Expand</title>
|
|
||||||
<path
|
|
||||||
fill="currentColor"
|
|
||||||
d="M7.41 8.58L12 13.17l4.59-4.59L18 10l-6 6l-6-6z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</Button>
|
|
||||||
</PopoverTrigger>
|
|
||||||
<PopoverContent
|
|
||||||
anchor={"start"}
|
|
||||||
align={"start"}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.direction = "col";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Column
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.direction = "row";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Row
|
|
||||||
</Button>
|
|
||||||
</PopoverContent>
|
|
||||||
</Popover>
|
|
||||||
</Label>
|
|
||||||
<Label direction={"horizontal"}>
|
|
||||||
<span>position = </span>
|
|
||||||
<Popover>
|
|
||||||
<PopoverTrigger>
|
|
||||||
<Button className={"gap-2"}>
|
|
||||||
{props.PopoverContentProps.position}
|
|
||||||
<svg
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="1em"
|
|
||||||
height="1em"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<title>Expand</title>
|
|
||||||
<path
|
|
||||||
fill="currentColor"
|
|
||||||
d="M7.41 8.58L12 13.17l4.59-4.59L18 10l-6 6l-6-6z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</Button>
|
|
||||||
</PopoverTrigger>
|
|
||||||
<PopoverContent
|
|
||||||
anchor={"start"}
|
|
||||||
align={"start"}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.position = "start";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Start
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.position = "end";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
End
|
|
||||||
</Button>
|
|
||||||
</PopoverContent>
|
|
||||||
</Popover>
|
|
||||||
</Label>
|
|
||||||
<Label direction={"horizontal"}>
|
|
||||||
<span>anchor = </span>
|
|
||||||
<Popover>
|
|
||||||
<PopoverTrigger>
|
|
||||||
<Button className={"gap-2"}>
|
|
||||||
{props.PopoverContentProps.anchor}
|
|
||||||
<svg
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="1em"
|
|
||||||
height="1em"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<title>Expand</title>
|
|
||||||
<path
|
|
||||||
fill="currentColor"
|
|
||||||
d="M7.41 8.58L12 13.17l4.59-4.59L18 10l-6 6l-6-6z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</Button>
|
|
||||||
</PopoverTrigger>
|
|
||||||
<PopoverContent
|
|
||||||
anchor={"start"}
|
|
||||||
align={"start"}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.anchor = "start";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Start
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.anchor = "middle";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Middle
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.anchor = "end";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
End
|
|
||||||
</Button>
|
|
||||||
</PopoverContent>
|
|
||||||
</Popover>
|
|
||||||
</Label>
|
|
||||||
<Label direction={"horizontal"}>
|
|
||||||
<span>align = </span>
|
|
||||||
<Popover>
|
|
||||||
<PopoverTrigger>
|
|
||||||
<Button className={"gap-2"}>
|
|
||||||
{props.PopoverContentProps.align}
|
|
||||||
<svg
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="1em"
|
|
||||||
height="1em"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<title>Expand</title>
|
|
||||||
<path
|
|
||||||
fill="currentColor"
|
|
||||||
d="M7.41 8.58L12 13.17l4.59-4.59L18 10l-6 6l-6-6z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</Button>
|
|
||||||
</PopoverTrigger>
|
|
||||||
<PopoverContent
|
|
||||||
anchor={"start"}
|
|
||||||
align={"start"}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.align = "start";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Start
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.align = "middle";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Middle
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.align = "end";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
End
|
|
||||||
</Button>
|
|
||||||
</PopoverContent>
|
|
||||||
</Popover>
|
|
||||||
</Label>
|
|
||||||
<Label direction={"horizontal"}>
|
|
||||||
<span>offset = </span>
|
|
||||||
<Popover>
|
|
||||||
<PopoverTrigger>
|
|
||||||
<Button className={"gap-2"}>
|
|
||||||
{props.PopoverContentProps.offset}
|
|
||||||
<svg
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="1em"
|
|
||||||
height="1em"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
>
|
|
||||||
<title>Expand</title>
|
|
||||||
<path
|
|
||||||
fill="currentColor"
|
|
||||||
d="M7.41 8.58L12 13.17l4.59-4.59L18 10l-6 6l-6-6z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</Button>
|
|
||||||
</PopoverTrigger>
|
|
||||||
<PopoverContent
|
|
||||||
anchor={"start"}
|
|
||||||
align={"start"}
|
|
||||||
>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.offset = "sm";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Small
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.offset = "md";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Middle
|
|
||||||
</Button>
|
|
||||||
<Button
|
|
||||||
preset={"ghost"}
|
|
||||||
onClick={() =>
|
|
||||||
mutate((p) => {
|
|
||||||
p.PopoverContentProps.offset = "lg";
|
|
||||||
})
|
|
||||||
}
|
|
||||||
>
|
|
||||||
Large
|
|
||||||
</Button>
|
|
||||||
</PopoverContent>
|
|
||||||
</Popover>
|
|
||||||
</Label>
|
|
||||||
</div>
|
|
||||||
</>
|
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
@ -1,5 +1,6 @@
|
|||||||
import { Button } from "@pswui/Button";
|
import { Button } from "@pswui/Button";
|
||||||
import { Popover, PopoverContent, PopoverTrigger } from "@pswui/Popover";
|
import { Popover, PopoverContent, PopoverTrigger } from "@pswui/Popover";
|
||||||
|
/* remove */
|
||||||
|
|
||||||
export interface ControlledPopoverDemoProps {
|
export interface ControlledPopoverDemoProps {
|
||||||
PopoverProps: {
|
PopoverProps: {
|
||||||
@ -15,12 +16,17 @@ export interface ControlledPopoverDemoProps {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/* end */
|
||||||
|
/* replace */
|
||||||
export function PopoverDemo({
|
export function PopoverDemo({
|
||||||
PopoverProps,
|
PopoverProps,
|
||||||
PopoverContentProps,
|
PopoverContentProps,
|
||||||
}: ControlledPopoverDemoProps) {
|
}: ControlledPopoverDemoProps) {
|
||||||
|
/* with
|
||||||
|
export function PopoverDemo() {
|
||||||
|
*/
|
||||||
return (
|
return (
|
||||||
<Popover {...PopoverProps}>
|
<Popover opened={PopoverProps.opened}>
|
||||||
<PopoverTrigger>
|
<PopoverTrigger>
|
||||||
<Button size="icon">
|
<Button size="icon">
|
||||||
<svg
|
<svg
|
||||||
@ -37,7 +43,14 @@ export function PopoverDemo({
|
|||||||
</svg>
|
</svg>
|
||||||
</Button>
|
</Button>
|
||||||
</PopoverTrigger>
|
</PopoverTrigger>
|
||||||
<PopoverContent {...PopoverContentProps}>
|
<PopoverContent
|
||||||
|
direction={PopoverContentProps.direction}
|
||||||
|
position={PopoverContentProps.position}
|
||||||
|
anchor={PopoverContentProps.anchor}
|
||||||
|
align={PopoverContentProps.align}
|
||||||
|
offset={PopoverContentProps.offset}
|
||||||
|
className={PopoverContentProps.className}
|
||||||
|
>
|
||||||
<Button
|
<Button
|
||||||
preset="ghost"
|
preset="ghost"
|
||||||
className="gap-2"
|
className="gap-2"
|
||||||
@ -54,7 +67,7 @@ export function PopoverDemo({
|
|||||||
d="M3 6h18v2H3zm0 5h18v2H3zm0 5h18v2H3z"
|
d="M3 6h18v2H3zm0 5h18v2H3zm0 5h18v2H3z"
|
||||||
/>
|
/>
|
||||||
</svg>
|
</svg>
|
||||||
<span className="flex-grow text-left">Dashboard</span>
|
<span className="grow text-left">Dashboard</span>
|
||||||
</Button>
|
</Button>
|
||||||
<Button
|
<Button
|
||||||
preset="ghost"
|
preset="ghost"
|
||||||
@ -72,7 +85,7 @@ export function PopoverDemo({
|
|||||||
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"
|
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>
|
</svg>
|
||||||
<span className="flex-grow text-left">Log out</span>
|
<span className="grow text-left">Log out</span>
|
||||||
</Button>
|
</Button>
|
||||||
</PopoverContent>
|
</PopoverContent>
|
||||||
</Popover>
|
</Popover>
|
||||||
|
@ -12,7 +12,7 @@ Organizes content into multiple sections with tabbed navigation.
|
|||||||
<TabTrigger name="code">Code</TabTrigger>
|
<TabTrigger name="code">Code</TabTrigger>
|
||||||
</TabList>
|
</TabList>
|
||||||
<TabContent name="preview">
|
<TabContent name="preview">
|
||||||
<Story layout="centered" className="flex-col [&>*]:w-full">
|
<Story layout="centered" className="flex-col *:w-full">
|
||||||
<TabsDemo />
|
<TabsDemo />
|
||||||
</Story>
|
</Story>
|
||||||
</TabContent>
|
</TabContent>
|
||||||
|
@ -3,7 +3,7 @@ import { Link } from "react-router-dom";
|
|||||||
|
|
||||||
function PageNotFound() {
|
function PageNotFound() {
|
||||||
return (
|
return (
|
||||||
<main className="flex-grow h-full flex flex-col justify-center items-center gap-8">
|
<main className="grow h-full flex flex-col justify-center items-center gap-8">
|
||||||
<section className="flex flex-col justify-center items-center text-center gap-2">
|
<section className="flex flex-col justify-center items-center text-center gap-2">
|
||||||
<h1 className="text-4xl font-bold">Page not found</h1>
|
<h1 className="text-4xl font-bold">Page not found</h1>
|
||||||
<p className="text-base">
|
<p className="text-base">
|
||||||
|
@ -3,7 +3,7 @@ import { Link } from "react-router-dom";
|
|||||||
|
|
||||||
function UnexpectedError() {
|
function UnexpectedError() {
|
||||||
return (
|
return (
|
||||||
<main className="flex-grow h-full flex flex-col justify-center items-center gap-8">
|
<main className="grow h-full flex flex-col justify-center items-center gap-8">
|
||||||
<section className="flex flex-col justify-center items-center text-center gap-2">
|
<section className="flex flex-col justify-center items-center text-center gap-2">
|
||||||
<h1 className="text-4xl font-bold">Something went wrong</h1>
|
<h1 className="text-4xl font-bold">Something went wrong</h1>
|
||||||
<p className="text-base">
|
<p className="text-base">
|
||||||
|
@ -14,21 +14,18 @@ const colors = {
|
|||||||
danger: "border-red-400 dark:border-red-600",
|
danger: "border-red-400 dark:border-red-600",
|
||||||
},
|
},
|
||||||
background: {
|
background: {
|
||||||
default:
|
default: "bg-white dark:bg-black",
|
||||||
"bg-white dark:bg-black hover:bg-neutral-200 dark:hover:bg-neutral-800",
|
|
||||||
ghost:
|
ghost:
|
||||||
"bg-black/0 dark:bg-white/0 hover:bg-black/20 dark:hover:bg-white/20",
|
"bg-black/0 dark:bg-white/0 hover:bg-black/20 dark:hover:bg-white/20",
|
||||||
success:
|
success: "bg-green-100 dark:bg-green-900",
|
||||||
"bg-green-100 dark:bg-green-900 hover:bg-green-200 dark:hover:bg-green-800",
|
warning: "bg-yellow-100 dark:bg-yellow-900",
|
||||||
warning:
|
danger: "bg-red-100 dark:bg-red-900",
|
||||||
"bg-yellow-100 dark:bg-yellow-900 hover:bg-yellow-200 dark:hover:bg-yellow-800",
|
|
||||||
danger: "bg-red-100 dark:bg-red-900 hover:bg-red-200 dark:hover:bg-red-800",
|
|
||||||
},
|
},
|
||||||
underline: "decoration-current",
|
underline: "decoration-current",
|
||||||
};
|
};
|
||||||
|
|
||||||
const [buttonVariants, resolveVariants] = vcn({
|
const [buttonVariants, resolveVariants] = vcn({
|
||||||
base: `w-fit flex flex-row items-center justify-between rounded-md outline outline-1 outline-transparent outline-offset-2 ${colors.outline.focus} ${colors.disabled} transition-all cursor-pointer`,
|
base: `w-fit flex flex-row items-center justify-between rounded-md outline outline-1 outline-transparent outline-offset-2 hover:brightness-90 dark:hover:brightness-110 ${colors.outline.focus} ${colors.disabled} transition-all cursor-pointer`,
|
||||||
variants: {
|
variants: {
|
||||||
size: {
|
size: {
|
||||||
link: "p-0 text-base",
|
link: "p-0 text-base",
|
||||||
@ -111,21 +108,22 @@ export interface ButtonProps
|
|||||||
const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
|
const Button = React.forwardRef<HTMLButtonElement, ButtonProps>(
|
||||||
(props, ref) => {
|
(props, ref) => {
|
||||||
const [variantProps, otherPropsCompressed] = resolveVariants(props);
|
const [variantProps, otherPropsCompressed] = resolveVariants(props);
|
||||||
const { asChild, ...otherPropsExtracted } = otherPropsCompressed;
|
const { asChild, type, role, ...otherPropsExtracted } =
|
||||||
|
otherPropsCompressed;
|
||||||
|
|
||||||
const Comp = asChild ? Slot : "button";
|
const Comp = asChild ? Slot : "button";
|
||||||
const compProps = {
|
|
||||||
...otherPropsExtracted,
|
|
||||||
className: buttonVariants(variantProps),
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Comp
|
<Comp
|
||||||
ref={ref}
|
ref={ref}
|
||||||
{...compProps}
|
type={type ?? "button"}
|
||||||
|
className={buttonVariants(variantProps)}
|
||||||
|
role={role ?? "button"}
|
||||||
|
{...otherPropsExtracted}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
Button.displayName = "Button";
|
||||||
|
|
||||||
export { Button };
|
export { Button };
|
||||||
|
@ -18,12 +18,10 @@ const checkboxColors = {
|
|||||||
disabledCheckedHover:
|
disabledCheckedHover:
|
||||||
"has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-300 dark:has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-700",
|
"has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-300 dark:has-[input[type='checkbox']:disabled:checked]:hover:bg-neutral-700",
|
||||||
},
|
},
|
||||||
checkmark:
|
|
||||||
"text-black dark:text-white has-[input[type=checkbox]:disabled]:text-neutral-400 dark:has-[input[type=checkbox]:disabled]:text-neutral-500",
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const [checkboxVariant, resolveCheckboxVariantProps] = vcn({
|
const [checkboxVariant, resolveCheckboxVariantProps] = vcn({
|
||||||
base: `inline-block rounded-md ${checkboxColors.checkmark} ${checkboxColors.background.disabled} ${checkboxColors.background.default} ${checkboxColors.background.hover} ${checkboxColors.background.checked} ${checkboxColors.background.checkedHover} ${checkboxColors.background.disabledChecked} ${checkboxColors.background.disabledCheckedHover} has-[input[type="checkbox"]:disabled]:cursor-not-allowed transition-colors duration-75 ease-in-out`,
|
base: `inline-block rounded-md ${checkboxColors.background.disabled} ${checkboxColors.background.default} ${checkboxColors.background.hover} ${checkboxColors.background.checked} ${checkboxColors.background.checkedHover} ${checkboxColors.background.disabledChecked} ${checkboxColors.background.disabledCheckedHover} has-[input[type="checkbox"]:disabled]:cursor-not-allowed transition-colors duration-150 ease-in-out`,
|
||||||
variants: {
|
variants: {
|
||||||
size: {
|
size: {
|
||||||
base: "size-[1em] p-0 [&>svg]:size-[1em]",
|
base: "size-[1em] p-0 [&>svg]:size-[1em]",
|
||||||
@ -92,23 +90,11 @@ const Checkbox = React.forwardRef<HTMLInputElement, CheckboxProps>(
|
|||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<svg
|
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
|
||||||
width="1em"
|
|
||||||
height="1em"
|
|
||||||
viewBox="0 0 24 24"
|
|
||||||
className={`${checked ? "opacity-100" : "opacity-0"} transition-opacity duration-75 ease-in-out`}
|
|
||||||
>
|
|
||||||
<title>checked</title>
|
|
||||||
<path
|
|
||||||
fill="currentColor"
|
|
||||||
d="M21 7L9 19l-5.5-5.5l1.41-1.41L9 16.17L19.59 5.59z"
|
|
||||||
/>
|
|
||||||
</svg>
|
|
||||||
</label>
|
</label>
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
Checkbox.displayName = "Checkbox";
|
||||||
|
|
||||||
export { Checkbox };
|
export { Checkbox };
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
import { Slot, type VariantProps, vcn } from "@pswui-lib";
|
import { Slot, type VariantProps, useDocument, vcn } from "@pswui-lib";
|
||||||
import React, { useState } from "react";
|
import React, { type ReactNode, useState } from "react";
|
||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
@ -55,33 +55,15 @@ const DialogTrigger = ({ children }: DialogTriggerProps) => {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const [dialogOverlayVariant, resolveDialogOverlayVariant] = vcn({
|
const [dialogOverlayVariant, resolveDialogOverlayVariant] = vcn({
|
||||||
base: "fixed inset-0 z-50 w-full h-screen overflow-y-auto max-w-screen transition-all duration-300",
|
base: "fixed inset-0 z-50 w-full h-screen overflow-y-auto max-w-screen transition-all duration-300 backdrop-blur-md backdrop-brightness-75 [&>div]:p-6",
|
||||||
variants: {
|
variants: {
|
||||||
opened: {
|
opened: {
|
||||||
true: "pointer-events-auto opacity-100",
|
true: "pointer-events-auto opacity-100",
|
||||||
false: "pointer-events-none opacity-0",
|
false: "pointer-events-none opacity-0",
|
||||||
},
|
},
|
||||||
blur: {
|
|
||||||
sm: "backdrop-blur-sm",
|
|
||||||
md: "backdrop-blur-md",
|
|
||||||
lg: "backdrop-blur-lg",
|
|
||||||
},
|
|
||||||
darken: {
|
|
||||||
sm: "backdrop-brightness-90",
|
|
||||||
md: "backdrop-brightness-75",
|
|
||||||
lg: "backdrop-brightness-50",
|
|
||||||
},
|
|
||||||
padding: {
|
|
||||||
sm: "[&>div]:p-4",
|
|
||||||
md: "[&>div]:p-6",
|
|
||||||
lg: "[&>div]:p-8",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
defaults: {
|
defaults: {
|
||||||
opened: false,
|
opened: false,
|
||||||
blur: "md",
|
|
||||||
darken: "md",
|
|
||||||
padding: "md",
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -100,35 +82,36 @@ const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
|
|||||||
});
|
});
|
||||||
const { children, closeOnClick, onClick, ...otherPropsExtracted } =
|
const { children, closeOnClick, onClick, ...otherPropsExtracted } =
|
||||||
otherPropsCompressed;
|
otherPropsCompressed;
|
||||||
return (
|
|
||||||
<>
|
const document = useDocument();
|
||||||
{ReactDOM.createPortal(
|
if (!document) return null;
|
||||||
<div
|
|
||||||
{...otherPropsExtracted}
|
return ReactDOM.createPortal(
|
||||||
ref={ref}
|
<div
|
||||||
className={dialogOverlayVariant(variantProps)}
|
{...otherPropsExtracted}
|
||||||
onClick={(e) => {
|
ref={ref}
|
||||||
if (closeOnClick) {
|
className={dialogOverlayVariant(variantProps)}
|
||||||
setContext((p) => ({ ...p, opened: false }));
|
onClick={(e) => {
|
||||||
}
|
if (closeOnClick) {
|
||||||
onClick?.(e);
|
setContext((p) => ({ ...p, opened: false }));
|
||||||
}}
|
}
|
||||||
>
|
onClick?.(e);
|
||||||
<div
|
}}
|
||||||
className={
|
>
|
||||||
"w-screen max-w-full min-h-screen flex flex-col justify-center items-center"
|
<div
|
||||||
}
|
className={
|
||||||
>
|
"w-screen max-w-full min-h-screen flex flex-col justify-center items-center"
|
||||||
{/* Layer for overflow positioning */}
|
}
|
||||||
{children}
|
>
|
||||||
</div>
|
{/* Layer for overflow positioning */}
|
||||||
</div>,
|
{children}
|
||||||
document.body,
|
</div>
|
||||||
)}
|
</div>,
|
||||||
</>
|
document.body,
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DialogOverlay.displayName = "DialogOverlay";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* =========================
|
* =========================
|
||||||
@ -137,43 +120,15 @@ const DialogOverlay = React.forwardRef<HTMLDivElement, DialogOverlay>(
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const [dialogContentVariant, resolveDialogContentVariant] = vcn({
|
const [dialogContentVariant, resolveDialogContentVariant] = vcn({
|
||||||
base: "transition-transform duration-300 bg-white dark:bg-black border border-neutral-200 dark:border-neutral-800",
|
base: "transition-transform duration-300 bg-white dark:bg-black border border-neutral-200 dark:border-neutral-800 p-6 w-full max-w-xl rounded-md flex flex-col justify-start items-start gap-6",
|
||||||
variants: {
|
variants: {
|
||||||
opened: {
|
opened: {
|
||||||
true: "scale-100",
|
true: "scale-100",
|
||||||
false: "scale-50",
|
false: "scale-50",
|
||||||
},
|
},
|
||||||
size: {
|
|
||||||
fit: "w-fit",
|
|
||||||
fullSm: "w-full max-w-sm",
|
|
||||||
fullMd: "w-full max-w-md",
|
|
||||||
fullLg: "w-full max-w-lg",
|
|
||||||
fullXl: "w-full max-w-xl",
|
|
||||||
full2xl: "w-full max-w-2xl",
|
|
||||||
},
|
|
||||||
rounded: {
|
|
||||||
sm: "rounded-sm",
|
|
||||||
md: "rounded-md",
|
|
||||||
lg: "rounded-lg",
|
|
||||||
xl: "rounded-xl",
|
|
||||||
},
|
|
||||||
padding: {
|
|
||||||
sm: "p-4",
|
|
||||||
md: "p-6",
|
|
||||||
lg: "p-8",
|
|
||||||
},
|
|
||||||
gap: {
|
|
||||||
sm: "space-y-4",
|
|
||||||
md: "space-y-6",
|
|
||||||
lg: "space-y-8",
|
|
||||||
},
|
|
||||||
},
|
},
|
||||||
defaults: {
|
defaults: {
|
||||||
opened: false,
|
opened: false,
|
||||||
size: "fit",
|
|
||||||
rounded: "md",
|
|
||||||
padding: "md",
|
|
||||||
gap: "md",
|
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -204,6 +159,7 @@ const DialogContent = React.forwardRef<HTMLDivElement, DialogContentProps>(
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DialogContent.displayName = "DialogContent";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* =========================
|
* =========================
|
||||||
@ -234,17 +190,9 @@ const DialogClose = ({ children }: DialogCloseProps) => {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const [dialogHeaderVariant, resolveDialogHeaderVariant] = vcn({
|
const [dialogHeaderVariant, resolveDialogHeaderVariant] = vcn({
|
||||||
base: "flex flex-col",
|
base: "flex flex-col gap-2",
|
||||||
variants: {
|
variants: {},
|
||||||
gap: {
|
defaults: {},
|
||||||
sm: "gap-2",
|
|
||||||
md: "gap-4",
|
|
||||||
lg: "gap-6",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
defaults: {
|
|
||||||
gap: "sm",
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
interface DialogHeaderProps
|
interface DialogHeaderProps
|
||||||
@ -268,6 +216,8 @@ const DialogHeader = React.forwardRef<HTMLElement, DialogHeaderProps>(
|
|||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
|
||||||
|
DialogHeader.displayName = "DialogHeader";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* =========================
|
* =========================
|
||||||
* DialogTitle / DialogSubtitle
|
* DialogTitle / DialogSubtitle
|
||||||
@ -275,22 +225,9 @@ const DialogHeader = React.forwardRef<HTMLElement, DialogHeaderProps>(
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const [dialogTitleVariant, resolveDialogTitleVariant] = vcn({
|
const [dialogTitleVariant, resolveDialogTitleVariant] = vcn({
|
||||||
variants: {
|
base: "text-xl font-bold",
|
||||||
size: {
|
variants: {},
|
||||||
sm: "text-lg",
|
defaults: {},
|
||||||
md: "text-xl",
|
|
||||||
lg: "text-2xl",
|
|
||||||
},
|
|
||||||
weight: {
|
|
||||||
sm: "font-medium",
|
|
||||||
md: "font-semibold",
|
|
||||||
lg: "font-bold",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
defaults: {
|
|
||||||
size: "md",
|
|
||||||
weight: "lg",
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
interface DialogTitleProps
|
interface DialogTitleProps
|
||||||
@ -298,28 +235,9 @@ interface DialogTitleProps
|
|||||||
VariantProps<typeof dialogTitleVariant> {}
|
VariantProps<typeof dialogTitleVariant> {}
|
||||||
|
|
||||||
const [dialogSubtitleVariant, resolveDialogSubtitleVariant] = vcn({
|
const [dialogSubtitleVariant, resolveDialogSubtitleVariant] = vcn({
|
||||||
variants: {
|
base: "text-sm opacity-60 font-normal",
|
||||||
size: {
|
variants: {},
|
||||||
sm: "text-sm",
|
defaults: {},
|
||||||
md: "text-base",
|
|
||||||
lg: "text-lg",
|
|
||||||
},
|
|
||||||
opacity: {
|
|
||||||
sm: "opacity-60",
|
|
||||||
md: "opacity-70",
|
|
||||||
lg: "opacity-80",
|
|
||||||
},
|
|
||||||
weight: {
|
|
||||||
sm: "font-light",
|
|
||||||
md: "font-normal",
|
|
||||||
lg: "font-medium",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
defaults: {
|
|
||||||
size: "sm",
|
|
||||||
opacity: "sm",
|
|
||||||
weight: "md",
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
interface DialogSubtitleProps
|
interface DialogSubtitleProps
|
||||||
@ -342,6 +260,7 @@ const DialogTitle = React.forwardRef<HTMLHeadingElement, DialogTitleProps>(
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DialogTitle.displayName = "DialogTitle";
|
||||||
|
|
||||||
const DialogSubtitle = React.forwardRef<
|
const DialogSubtitle = React.forwardRef<
|
||||||
HTMLHeadingElement,
|
HTMLHeadingElement,
|
||||||
@ -360,6 +279,7 @@ const DialogSubtitle = React.forwardRef<
|
|||||||
</h2>
|
</h2>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
DialogSubtitle.displayName = "DialogSubtitle";
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* =========================
|
* =========================
|
||||||
@ -368,17 +288,9 @@ const DialogSubtitle = React.forwardRef<
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
const [dialogFooterVariant, resolveDialogFooterVariant] = vcn({
|
const [dialogFooterVariant, resolveDialogFooterVariant] = vcn({
|
||||||
base: "flex flex-col items-end sm:flex-row sm:items-center sm:justify-end",
|
base: "flex w-full flex-col items-end sm:flex-row sm:items-center sm:justify-end gap-2",
|
||||||
variants: {
|
variants: {},
|
||||||
gap: {
|
defaults: {},
|
||||||
sm: "gap-2",
|
|
||||||
md: "gap-4",
|
|
||||||
lg: "gap-6",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
defaults: {
|
|
||||||
gap: "md",
|
|
||||||
},
|
|
||||||
});
|
});
|
||||||
|
|
||||||
interface DialogFooterProps
|
interface DialogFooterProps
|
||||||
@ -401,6 +313,34 @@ const DialogFooter = React.forwardRef<HTMLDivElement, DialogFooterProps>(
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DialogFooter.displayName = "DialogFooter";
|
||||||
|
|
||||||
|
interface DialogControllers {
|
||||||
|
context: IDialogContext;
|
||||||
|
|
||||||
|
setContext: React.Dispatch<React.SetStateAction<IDialogContext>>;
|
||||||
|
close: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
interface DialogControllerProps {
|
||||||
|
children: (controllers: DialogControllers) => ReactNode;
|
||||||
|
}
|
||||||
|
|
||||||
|
const DialogController = (props: DialogControllerProps) => {
|
||||||
|
return (
|
||||||
|
<DialogContext.Consumer>
|
||||||
|
{([context, setContext]) =>
|
||||||
|
props.children({
|
||||||
|
context,
|
||||||
|
setContext,
|
||||||
|
close() {
|
||||||
|
setContext((p) => ({ ...p, opened: false }));
|
||||||
|
},
|
||||||
|
})
|
||||||
|
}
|
||||||
|
</DialogContext.Consumer>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
export {
|
export {
|
||||||
DialogRoot,
|
DialogRoot,
|
||||||
@ -412,4 +352,5 @@ export {
|
|||||||
DialogTitle,
|
DialogTitle,
|
||||||
DialogSubtitle,
|
DialogSubtitle,
|
||||||
DialogFooter,
|
DialogFooter,
|
||||||
|
DialogController,
|
||||||
};
|
};
|
||||||
|
@ -1,4 +1,11 @@
|
|||||||
import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib";
|
import {
|
||||||
|
type AsChild,
|
||||||
|
Slot,
|
||||||
|
type VariantProps,
|
||||||
|
useAnimatedMount,
|
||||||
|
useDocument,
|
||||||
|
vcn,
|
||||||
|
} from "@pswui-lib";
|
||||||
import React, {
|
import React, {
|
||||||
type ComponentPropsWithoutRef,
|
type ComponentPropsWithoutRef,
|
||||||
type TouchEvent as ReactTouchEvent,
|
type TouchEvent as ReactTouchEvent,
|
||||||
@ -15,6 +22,8 @@ interface IDrawerContext {
|
|||||||
closeThreshold: number;
|
closeThreshold: number;
|
||||||
movePercentage: number;
|
movePercentage: number;
|
||||||
isDragging: boolean;
|
isDragging: boolean;
|
||||||
|
isMounted: boolean;
|
||||||
|
isRendered: boolean;
|
||||||
leaveWhileDragging: boolean;
|
leaveWhileDragging: boolean;
|
||||||
}
|
}
|
||||||
const DrawerContextInitial: IDrawerContext = {
|
const DrawerContextInitial: IDrawerContext = {
|
||||||
@ -22,6 +31,8 @@ const DrawerContextInitial: IDrawerContext = {
|
|||||||
closeThreshold: 0.3,
|
closeThreshold: 0.3,
|
||||||
movePercentage: 0,
|
movePercentage: 0,
|
||||||
isDragging: false,
|
isDragging: false,
|
||||||
|
isMounted: false,
|
||||||
|
isRendered: false,
|
||||||
leaveWhileDragging: false,
|
leaveWhileDragging: false,
|
||||||
};
|
};
|
||||||
const DrawerContext = React.createContext<
|
const DrawerContext = React.createContext<
|
||||||
@ -96,8 +107,14 @@ interface DrawerOverlayProps
|
|||||||
|
|
||||||
const DrawerOverlay = forwardRef<HTMLDivElement, DrawerOverlayProps>(
|
const DrawerOverlay = forwardRef<HTMLDivElement, DrawerOverlayProps>(
|
||||||
(props, ref) => {
|
(props, ref) => {
|
||||||
|
const internalRef = useRef<HTMLDivElement | null>(null);
|
||||||
const [state, setState] = useContext(DrawerContext);
|
const [state, setState] = useContext(DrawerContext);
|
||||||
|
|
||||||
|
const { isMounted, isRendered } = useAnimatedMount(
|
||||||
|
state.isDragging ? true : state.opened,
|
||||||
|
internalRef,
|
||||||
|
);
|
||||||
|
|
||||||
const [variantProps, restPropsCompressed] =
|
const [variantProps, restPropsCompressed] =
|
||||||
resolveDrawerOverlayVariantProps(props);
|
resolveDrawerOverlayVariantProps(props);
|
||||||
const { asChild, ...restPropsExtracted } = restPropsCompressed;
|
const { asChild, ...restPropsExtracted } = restPropsCompressed;
|
||||||
@ -119,25 +136,46 @@ const DrawerOverlay = forwardRef<HTMLDivElement, DrawerOverlayProps>(
|
|||||||
: 1
|
: 1
|
||||||
})`;
|
})`;
|
||||||
|
|
||||||
return createPortal(
|
const document = useDocument();
|
||||||
<Comp
|
if (!document) return null;
|
||||||
{...restPropsExtracted}
|
|
||||||
className={drawerOverlayVariant({
|
return (
|
||||||
...variantProps,
|
<>
|
||||||
opened: state.isDragging ? true : state.opened,
|
<DrawerContext.Provider
|
||||||
})}
|
value={[{ ...state, isMounted, isRendered }, setState]}
|
||||||
onClick={onOutsideClick}
|
>
|
||||||
style={{
|
{isMounted
|
||||||
backdropFilter,
|
? createPortal(
|
||||||
WebkitBackdropFilter: backdropFilter,
|
<Comp
|
||||||
transitionDuration: state.isDragging ? "0s" : undefined,
|
{...restPropsExtracted}
|
||||||
}}
|
className={drawerOverlayVariant({
|
||||||
ref={ref}
|
...variantProps,
|
||||||
/>,
|
opened: isRendered,
|
||||||
document.body,
|
})}
|
||||||
|
onClick={onOutsideClick}
|
||||||
|
style={{
|
||||||
|
backdropFilter,
|
||||||
|
WebkitBackdropFilter: backdropFilter,
|
||||||
|
transitionDuration: state.isDragging ? "0s" : undefined,
|
||||||
|
}}
|
||||||
|
ref={(el: HTMLDivElement) => {
|
||||||
|
internalRef.current = el;
|
||||||
|
if (typeof ref === "function") {
|
||||||
|
ref(el);
|
||||||
|
} else if (ref) {
|
||||||
|
ref.current = el;
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
/>,
|
||||||
|
document.body,
|
||||||
|
)
|
||||||
|
: null}
|
||||||
|
</DrawerContext.Provider>
|
||||||
|
</>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DrawerOverlay.displayName = "DrawerOverlay";
|
||||||
|
|
||||||
const drawerContentColors = {
|
const drawerContentColors = {
|
||||||
background: "bg-white dark:bg-black",
|
background: "bg-white dark:bg-black",
|
||||||
@ -148,25 +186,52 @@ const [drawerContentVariant, resolveDrawerContentVariantProps] = vcn({
|
|||||||
base: `fixed ${drawerContentColors.background} ${drawerContentColors.border} transition-all p-4 flex flex-col justify-between gap-8 overflow-auto`,
|
base: `fixed ${drawerContentColors.background} ${drawerContentColors.border} transition-all p-4 flex flex-col justify-between gap-8 overflow-auto`,
|
||||||
variants: {
|
variants: {
|
||||||
position: {
|
position: {
|
||||||
top: "top-0 inset-x-0 w-full max-w-screen rounded-t-lg border-b-2",
|
top: "top-0 w-full max-w-screen rounded-t-lg border-b-2",
|
||||||
bottom: "bottom-0 inset-x-0 w-full max-w-screen rounded-b-lg border-t-2",
|
bottom: "bottom-0 w-full max-w-screen rounded-b-lg border-t-2",
|
||||||
left: "left-0 inset-y-0 h-screen rounded-l-lg border-r-2",
|
left: "left-0 h-screen rounded-l-lg border-r-2",
|
||||||
right: "right-0 inset-y-0 h-screen rounded-r-lg border-l-2",
|
right: "right-0 h-screen rounded-r-lg border-l-2",
|
||||||
|
},
|
||||||
|
maxSize: {
|
||||||
|
sm: "[&.left-0]:max-w-sm [&.right-0]:max-w-sm",
|
||||||
|
md: "[&.left-0]:max-w-md [&.right-0]:max-w-md",
|
||||||
|
lg: "[&.left-0]:max-w-lg [&.right-0]:max-w-lg",
|
||||||
|
xl: "[&.left-0]:max-w-xl [&.right-0]:max-w-xl",
|
||||||
},
|
},
|
||||||
opened: {
|
opened: {
|
||||||
true: "",
|
true: "",
|
||||||
false:
|
false:
|
||||||
"[&.top-0]:-translate-y-full [&.bottom-0]:translate-y-full [&.left-0]:-translate-x-full [&.right-0]:translate-x-full",
|
"[&.top-0]:-translate-y-full [&.bottom-0]:translate-y-full [&.left-0]:-translate-x-full [&.right-0]:translate-x-full",
|
||||||
},
|
},
|
||||||
|
internal: {
|
||||||
|
true: "relative",
|
||||||
|
false: "fixed",
|
||||||
|
},
|
||||||
},
|
},
|
||||||
defaults: {
|
defaults: {
|
||||||
position: "left",
|
position: "left",
|
||||||
opened: false,
|
opened: false,
|
||||||
|
maxSize: "sm",
|
||||||
|
internal: false,
|
||||||
},
|
},
|
||||||
|
dynamics: [
|
||||||
|
({ position, internal }) => {
|
||||||
|
if (!internal) {
|
||||||
|
if (["top", "bottom"].includes(position)) {
|
||||||
|
return "inset-x-0";
|
||||||
|
}
|
||||||
|
return "inset-y-0";
|
||||||
|
}
|
||||||
|
|
||||||
|
return "w-fit";
|
||||||
|
},
|
||||||
|
],
|
||||||
});
|
});
|
||||||
|
|
||||||
interface DrawerContentProps
|
interface DrawerContentProps
|
||||||
extends Omit<VariantProps<typeof drawerContentVariant>, "opened">,
|
extends Omit<
|
||||||
|
VariantProps<typeof drawerContentVariant>,
|
||||||
|
"opened" | "internal"
|
||||||
|
>,
|
||||||
AsChild,
|
AsChild,
|
||||||
ComponentPropsWithoutRef<"div"> {}
|
ComponentPropsWithoutRef<"div"> {}
|
||||||
|
|
||||||
@ -307,9 +372,9 @@ const DrawerContent = forwardRef<HTMLDivElement, DrawerContentProps>(
|
|||||||
<div
|
<div
|
||||||
className={drawerContentVariant({
|
className={drawerContentVariant({
|
||||||
...variantProps,
|
...variantProps,
|
||||||
opened: true,
|
opened: state.isRendered,
|
||||||
className: dragState.isDragging
|
className: dragState.isDragging
|
||||||
? "transition-[width_0ms]"
|
? "transition-[width] duration-0"
|
||||||
: variantProps.className,
|
: variantProps.className,
|
||||||
})}
|
})}
|
||||||
style={
|
style={
|
||||||
@ -321,6 +386,7 @@ const DrawerContent = forwardRef<HTMLDivElement, DrawerContentProps>(
|
|||||||
0) +
|
0) +
|
||||||
(position === "top" ? dragState.delta : -dragState.delta),
|
(position === "top" ? dragState.delta : -dragState.delta),
|
||||||
padding: 0,
|
padding: 0,
|
||||||
|
[`padding${position.slice(0, 1).toUpperCase()}${position.slice(1)}`]: `${dragState.delta}px`,
|
||||||
}
|
}
|
||||||
: {
|
: {
|
||||||
width:
|
width:
|
||||||
@ -328,6 +394,7 @@ const DrawerContent = forwardRef<HTMLDivElement, DrawerContentProps>(
|
|||||||
0) +
|
0) +
|
||||||
(position === "left" ? dragState.delta : -dragState.delta),
|
(position === "left" ? dragState.delta : -dragState.delta),
|
||||||
padding: 0,
|
padding: 0,
|
||||||
|
[`padding${position.slice(0, 1).toUpperCase()}${position.slice(1)}`]: `${dragState.delta}px`,
|
||||||
}
|
}
|
||||||
: { width: 0, height: 0, padding: 0 }
|
: { width: 0, height: 0, padding: 0 }
|
||||||
}
|
}
|
||||||
@ -336,14 +403,16 @@ const DrawerContent = forwardRef<HTMLDivElement, DrawerContentProps>(
|
|||||||
{...restPropsExtracted}
|
{...restPropsExtracted}
|
||||||
className={drawerContentVariant({
|
className={drawerContentVariant({
|
||||||
...variantProps,
|
...variantProps,
|
||||||
opened: state.opened,
|
opened: state.isRendered,
|
||||||
|
internal: true,
|
||||||
})}
|
})}
|
||||||
style={{
|
style={{
|
||||||
transform: dragState.isDragging
|
transform:
|
||||||
? `translate${["top", "bottom"].includes(position) ? "Y" : "X"}(${
|
dragState.isDragging &&
|
||||||
dragState.delta
|
((["top", "left"].includes(position) && dragState.delta < 0) ||
|
||||||
}px)`
|
(["bottom", "right"].includes(position) && dragState.delta > 0))
|
||||||
: undefined,
|
? `translate${["top", "bottom"].includes(position) ? "Y" : "X"}(${dragState.delta}px)`
|
||||||
|
: undefined,
|
||||||
transitionDuration: dragState.isDragging ? "0s" : undefined,
|
transitionDuration: dragState.isDragging ? "0s" : undefined,
|
||||||
userSelect: dragState.isDragging ? "none" : undefined,
|
userSelect: dragState.isDragging ? "none" : undefined,
|
||||||
}}
|
}}
|
||||||
@ -374,6 +443,7 @@ const DrawerContent = forwardRef<HTMLDivElement, DrawerContentProps>(
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DrawerContent.displayName = "DrawerContent";
|
||||||
|
|
||||||
const DrawerClose = forwardRef<
|
const DrawerClose = forwardRef<
|
||||||
HTMLButtonElement,
|
HTMLButtonElement,
|
||||||
@ -388,6 +458,7 @@ const DrawerClose = forwardRef<
|
|||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
DrawerClose.displayName = "DrawerClose";
|
||||||
|
|
||||||
const [drawerHeaderVariant, resolveDrawerHeaderVariantProps] = vcn({
|
const [drawerHeaderVariant, resolveDrawerHeaderVariantProps] = vcn({
|
||||||
base: "flex flex-col gap-2",
|
base: "flex flex-col gap-2",
|
||||||
@ -417,9 +488,10 @@ const DrawerHeader = forwardRef<HTMLDivElement, DrawerHeaderProps>(
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DrawerHeader.displayName = "DrawerHeader";
|
||||||
|
|
||||||
const [drawerBodyVariant, resolveDrawerBodyVariantProps] = vcn({
|
const [drawerBodyVariant, resolveDrawerBodyVariantProps] = vcn({
|
||||||
base: "flex-grow",
|
base: "grow",
|
||||||
variants: {},
|
variants: {},
|
||||||
defaults: {},
|
defaults: {},
|
||||||
});
|
});
|
||||||
@ -444,6 +516,7 @@ const DrawerBody = forwardRef<HTMLDivElement, DrawerBodyProps>((props, ref) => {
|
|||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
DrawerBody.displayName = "DrawerBody";
|
||||||
|
|
||||||
const [drawerFooterVariant, resolveDrawerFooterVariantProps] = vcn({
|
const [drawerFooterVariant, resolveDrawerFooterVariantProps] = vcn({
|
||||||
base: "flex flex-row justify-end gap-2",
|
base: "flex flex-row justify-end gap-2",
|
||||||
@ -473,6 +546,7 @@ const DrawerFooter = forwardRef<HTMLDivElement, DrawerFooterProps>(
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
DrawerFooter.displayName = "DrawerFooter";
|
||||||
|
|
||||||
export {
|
export {
|
||||||
DrawerRoot,
|
DrawerRoot,
|
||||||
|
183
src/pswui/components/Form.tsx
Normal file
183
src/pswui/components/Form.tsx
Normal file
@ -0,0 +1,183 @@
|
|||||||
|
import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib";
|
||||||
|
import {
|
||||||
|
type ComponentPropsWithoutRef,
|
||||||
|
createContext,
|
||||||
|
forwardRef,
|
||||||
|
useContext,
|
||||||
|
useEffect,
|
||||||
|
useRef,
|
||||||
|
} from "react";
|
||||||
|
|
||||||
|
/**
|
||||||
|
Form Item Context
|
||||||
|
**/
|
||||||
|
interface IFormItemContext {
|
||||||
|
invalid?: string | null | undefined;
|
||||||
|
}
|
||||||
|
const FormItemContext = createContext<IFormItemContext>({});
|
||||||
|
|
||||||
|
/**
|
||||||
|
FormItem
|
||||||
|
**/
|
||||||
|
const [formItemVariant, resolveFormItemVariantProps] = vcn({
|
||||||
|
base: "flex flex-col gap-2 items-start w-full",
|
||||||
|
variants: {},
|
||||||
|
defaults: {},
|
||||||
|
});
|
||||||
|
|
||||||
|
interface FormItemProps
|
||||||
|
extends VariantProps<typeof formItemVariant>,
|
||||||
|
AsChild,
|
||||||
|
ComponentPropsWithoutRef<"label"> {
|
||||||
|
invalid?: string | null | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
const FormItem = forwardRef<HTMLLabelElement, FormItemProps>((props, ref) => {
|
||||||
|
const [variantProps, restPropsCompressed] =
|
||||||
|
resolveFormItemVariantProps(props);
|
||||||
|
const { asChild, children, invalid, ...restPropsExtracted } =
|
||||||
|
restPropsCompressed;
|
||||||
|
const innerRef = useRef<HTMLLabelElement | null>(null);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const invalidAsString = invalid ? invalid : "";
|
||||||
|
|
||||||
|
const input = innerRef.current?.querySelector?.("input");
|
||||||
|
if (!input) return;
|
||||||
|
|
||||||
|
input.setCustomValidity(invalidAsString);
|
||||||
|
}, [invalid]);
|
||||||
|
|
||||||
|
const Comp = asChild ? Slot : "label";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<FormItemContext.Provider value={{ invalid }}>
|
||||||
|
<Comp
|
||||||
|
ref={(el: HTMLLabelElement | null) => {
|
||||||
|
innerRef.current = el;
|
||||||
|
if (typeof ref === "function") {
|
||||||
|
ref(el);
|
||||||
|
} else if (ref) {
|
||||||
|
ref.current = el;
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
className={formItemVariant(variantProps)}
|
||||||
|
{...restPropsExtracted}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Comp>
|
||||||
|
</FormItemContext.Provider>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
FormItem.displayName = "FormItem";
|
||||||
|
|
||||||
|
/**
|
||||||
|
FormLabel
|
||||||
|
**/
|
||||||
|
const [formLabelVariant, resolveFormLabelVariantProps] = vcn({
|
||||||
|
base: "text-sm font-bold",
|
||||||
|
variants: {},
|
||||||
|
defaults: {},
|
||||||
|
});
|
||||||
|
|
||||||
|
interface FormLabelProps
|
||||||
|
extends VariantProps<typeof formLabelVariant>,
|
||||||
|
AsChild,
|
||||||
|
ComponentPropsWithoutRef<"span"> {}
|
||||||
|
|
||||||
|
const FormLabel = forwardRef<HTMLSpanElement, FormLabelProps>((props, ref) => {
|
||||||
|
const [variantProps, otherPropsCompressed] =
|
||||||
|
resolveFormLabelVariantProps(props);
|
||||||
|
const { children, asChild, ...otherPropsExtracted } = otherPropsCompressed;
|
||||||
|
|
||||||
|
const Comp = asChild ? Slot : "span";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Comp
|
||||||
|
ref={ref}
|
||||||
|
className={formLabelVariant(variantProps)}
|
||||||
|
{...otherPropsExtracted}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Comp>
|
||||||
|
);
|
||||||
|
});
|
||||||
|
FormLabel.displayName = "FormLabel";
|
||||||
|
|
||||||
|
/**
|
||||||
|
FormHelper
|
||||||
|
**/
|
||||||
|
const [formHelperVariant, resolveFormHelperVariantProps] = vcn({
|
||||||
|
base: "opacity-75 text-sm font-light",
|
||||||
|
variants: {},
|
||||||
|
defaults: {},
|
||||||
|
});
|
||||||
|
|
||||||
|
interface FormHelperProps
|
||||||
|
extends VariantProps<typeof formHelperVariant>,
|
||||||
|
AsChild,
|
||||||
|
ComponentPropsWithoutRef<"span"> {
|
||||||
|
hiddenOnInvalid?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
const FormHelper = forwardRef<HTMLSpanElement, FormHelperProps>(
|
||||||
|
(props, ref) => {
|
||||||
|
const [variantProps, otherPropsCompressed] =
|
||||||
|
resolveFormHelperVariantProps(props);
|
||||||
|
const { asChild, children, hiddenOnInvalid, ...otherPropsExtracted } =
|
||||||
|
otherPropsCompressed;
|
||||||
|
const item = useContext(FormItemContext);
|
||||||
|
|
||||||
|
if (item.invalid && hiddenOnInvalid) return null;
|
||||||
|
|
||||||
|
const Comp = asChild ? Slot : "span";
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Comp
|
||||||
|
ref={ref}
|
||||||
|
className={formHelperVariant(variantProps)}
|
||||||
|
{...otherPropsExtracted}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
</Comp>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
FormHelper.displayName = "FormHelper";
|
||||||
|
|
||||||
|
/**
|
||||||
|
FormError
|
||||||
|
**/
|
||||||
|
const [formErrorVariant, resolveFormErrorVariantProps] = vcn({
|
||||||
|
base: "text-sm text-red-500",
|
||||||
|
variants: {},
|
||||||
|
defaults: {},
|
||||||
|
});
|
||||||
|
|
||||||
|
interface FormErrorProps
|
||||||
|
extends VariantProps<typeof formErrorVariant>,
|
||||||
|
AsChild,
|
||||||
|
Omit<ComponentPropsWithoutRef<"span">, "children"> {}
|
||||||
|
|
||||||
|
const FormError = forwardRef<HTMLSpanElement, FormErrorProps>((props, ref) => {
|
||||||
|
const [variantProps, otherPropsCompressed] =
|
||||||
|
resolveFormErrorVariantProps(props);
|
||||||
|
const { asChild, ...otherPropsExtracted } = otherPropsCompressed;
|
||||||
|
|
||||||
|
const item = useContext(FormItemContext);
|
||||||
|
|
||||||
|
const Comp = asChild ? Slot : "span";
|
||||||
|
|
||||||
|
return item.invalid ? (
|
||||||
|
<Comp
|
||||||
|
ref={ref}
|
||||||
|
className={formErrorVariant(variantProps)}
|
||||||
|
{...otherPropsExtracted}
|
||||||
|
>
|
||||||
|
{item.invalid}
|
||||||
|
</Comp>
|
||||||
|
) : null;
|
||||||
|
});
|
||||||
|
FormError.displayName = "FormError";
|
||||||
|
|
||||||
|
export { FormItem, FormLabel, FormHelper, FormError };
|
@ -1,37 +1,38 @@
|
|||||||
import { type VariantProps, vcn } from "@pswui-lib";
|
import { type AsChild, Slot, type VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
const inputColors = {
|
const inputColors = {
|
||||||
background: {
|
background: {
|
||||||
default: "bg-neutral-50 dark:bg-neutral-900",
|
default: "bg-neutral-50 dark:bg-neutral-900",
|
||||||
hover: "hover:bg-neutral-100 dark:hover:bg-neutral-800",
|
hover:
|
||||||
|
"hover:bg-neutral-100 dark:hover:bg-neutral-800 has-[input:hover]:bg-neutral-100 dark:has-[input:hover]:bg-neutral-800",
|
||||||
invalid:
|
invalid:
|
||||||
"invalid:bg-red-100 invalid:dark:bg-red-900 has-[input:invalid]:bg-red-100 dark:has-[input:invalid]:bg-red-900",
|
"invalid:bg-red-100 dark:invalid:bg-red-900 has-[input:invalid]:bg-red-100 dark:has-[input:invalid]:bg-red-900",
|
||||||
invalidHover:
|
invalidHover:
|
||||||
"hover:invalid:bg-red-200 dark:hover:invalid:bg-red-800 has-[input:invalid:hover]:bg-red-200 dark:has-[input:invalid:hover]:bg-red-800",
|
"hover:invalid:bg-red-200 dark:hover:invalid:bg-red-800 has-[input:invalid:hover]:bg-red-200 dark:has-[input:invalid:hover]:bg-red-800",
|
||||||
},
|
},
|
||||||
border: {
|
border: {
|
||||||
default: "border-neutral-400 dark:border-neutral-600",
|
default: "border-neutral-400 dark:border-neutral-600",
|
||||||
invalid:
|
invalid:
|
||||||
"invalid:border-red-400 invalid:dark:border-red-600 has-[input:invalid]:border-red-400 dark:has-[input:invalid]:border-red-600",
|
"invalid:border-red-400 dark:invalid:border-red-600 has-[input:invalid]:border-red-400 dark:has-[input:invalid]:border-red-600",
|
||||||
},
|
},
|
||||||
ring: {
|
ring: {
|
||||||
default: "ring-transparent focus-within:ring-current",
|
default: "ring-transparent focus-within:ring-current",
|
||||||
invalid:
|
invalid:
|
||||||
"invalid:focus-within:ring-red-400 invalid:focus-within:dark:ring-red-600 has-[input:invalid]:focus-within:ring-red-400 dark:has-[input:invalid]:focus-within:ring-red-600",
|
"invalid:focus-within:ring-red-400 dark:invalid:focus-within:ring-red-600 has-[input:invalid]:focus-within:ring-red-400 dark:has-[input:invalid]:focus-within:ring-red-600",
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const [inputVariant, resolveInputVariantProps] = vcn({
|
const [inputVariant, resolveInputVariantProps] = vcn({
|
||||||
base: `rounded-md p-2 border ring-1 outline-none transition-all duration-200 [appearance:textfield] disabled:brightness-50 disabled:saturate-0 disabled:cursor-not-allowed ${inputColors.background.default} ${inputColors.background.hover} ${inputColors.border.default} ${inputColors.ring.default} ${inputColors.background.invalid} ${inputColors.background.invalidHover} ${inputColors.border.invalid} ${inputColors.ring.invalid} [&:has(input)]:flex [&:has(input)]:w-fit`,
|
base: `rounded-md p-2 border ring-1 outline-hidden transition-all duration-200 [appearance:textfield] disabled:brightness-50 disabled:saturate-0 disabled:cursor-not-allowed ${inputColors.background.default} ${inputColors.background.hover} ${inputColors.border.default} ${inputColors.ring.default} ${inputColors.background.invalid} ${inputColors.background.invalidHover} ${inputColors.border.invalid} ${inputColors.ring.invalid} [&:has(input)]:flex`,
|
||||||
variants: {
|
variants: {
|
||||||
unstyled: {
|
unstyled: {
|
||||||
true: "bg-transparent border-none p-0 ring-0 hover:bg-transparent invalid:hover:bg-transparent invalid:focus-within:bg-transparent invalid:focus-within:ring-0",
|
true: "bg-transparent border-none p-0 ring-0 hover:bg-transparent invalid:hover:bg-transparent invalid:focus-within:bg-transparent invalid:focus-within:ring-0",
|
||||||
false: "",
|
false: "",
|
||||||
},
|
},
|
||||||
full: {
|
full: {
|
||||||
true: "w-full",
|
true: "[&:has(input)]:w-full w-full",
|
||||||
false: "w-fit",
|
false: "[&:has(input)]:w-fit w-fit",
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
defaults: {
|
defaults: {
|
||||||
@ -42,7 +43,8 @@ const [inputVariant, resolveInputVariantProps] = vcn({
|
|||||||
|
|
||||||
interface InputFrameProps
|
interface InputFrameProps
|
||||||
extends VariantProps<typeof inputVariant>,
|
extends VariantProps<typeof inputVariant>,
|
||||||
React.ComponentPropsWithoutRef<"label"> {
|
React.ComponentPropsWithoutRef<"label">,
|
||||||
|
AsChild {
|
||||||
children?: React.ReactNode;
|
children?: React.ReactNode;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -50,19 +52,22 @@ const InputFrame = React.forwardRef<HTMLLabelElement, InputFrameProps>(
|
|||||||
(props, ref) => {
|
(props, ref) => {
|
||||||
const [variantProps, otherPropsCompressed] =
|
const [variantProps, otherPropsCompressed] =
|
||||||
resolveInputVariantProps(props);
|
resolveInputVariantProps(props);
|
||||||
const { children, ...otherPropsExtracted } = otherPropsCompressed;
|
const { children, asChild, ...otherPropsExtracted } = otherPropsCompressed;
|
||||||
|
|
||||||
|
const Comp = asChild ? Slot : "label";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<label
|
<Comp
|
||||||
ref={ref}
|
ref={ref}
|
||||||
className={`group/input-frame ${inputVariant(variantProps)}`}
|
className={`group/input-frame ${inputVariant(variantProps)}`}
|
||||||
{...otherPropsExtracted}
|
{...otherPropsExtracted}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</label>
|
</Comp>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
InputFrame.displayName = "InputFrame";
|
||||||
|
|
||||||
interface InputProps
|
interface InputProps
|
||||||
extends VariantProps<typeof inputVariant>,
|
extends VariantProps<typeof inputVariant>,
|
||||||
@ -113,5 +118,6 @@ const Input = React.forwardRef<HTMLInputElement, InputProps>((props, ref) => {
|
|||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
Input.displayName = "Input";
|
||||||
|
|
||||||
export { InputFrame, Input };
|
export { InputFrame, Input };
|
||||||
|
@ -29,5 +29,6 @@ const Label = React.forwardRef<HTMLLabelElement, LabelProps>((props, ref) => {
|
|||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
Label.displayName = "Label";
|
||||||
|
|
||||||
export { Label };
|
export { Label };
|
||||||
|
@ -65,7 +65,7 @@ const popoverColors = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const [popoverContentVariant, resolvePopoverContentVariantProps] = vcn({
|
const [popoverContentVariant, resolvePopoverContentVariantProps] = vcn({
|
||||||
base: `absolute transition-all duration-150 border rounded-lg p-0.5 [&>*]:w-full z-10 ${popoverColors.background} ${popoverColors.border}`,
|
base: `absolute transition-all duration-150 border rounded-lg p-0.5 z-10 *:w-full ${popoverColors.background} ${popoverColors.border}`,
|
||||||
variants: {
|
variants: {
|
||||||
direction: {
|
direction: {
|
||||||
row: "",
|
row: "",
|
||||||
@ -200,7 +200,7 @@ const PopoverContent = React.forwardRef<HTMLDivElement, PopoverContentProps>(
|
|||||||
(props, ref) => {
|
(props, ref) => {
|
||||||
const [variantProps, otherPropsCompressed] =
|
const [variantProps, otherPropsCompressed] =
|
||||||
resolvePopoverContentVariantProps(props);
|
resolvePopoverContentVariantProps(props);
|
||||||
const { children, ...otherPropsExtracted } = otherPropsCompressed;
|
const { children, asChild, ...otherPropsExtracted } = otherPropsCompressed;
|
||||||
const [state, setState] = useContext(PopoverContext);
|
const [state, setState] = useContext(PopoverContext);
|
||||||
|
|
||||||
const internalRef = useRef<HTMLDivElement | null>(null);
|
const internalRef = useRef<HTMLDivElement | null>(null);
|
||||||
@ -221,14 +221,16 @@ const PopoverContent = React.forwardRef<HTMLDivElement, PopoverContentProps>(
|
|||||||
};
|
};
|
||||||
}, [state.controlled, setState]);
|
}, [state.controlled, setState]);
|
||||||
|
|
||||||
|
const Comp = asChild ? Slot : "div";
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<Comp
|
||||||
{...otherPropsExtracted}
|
{...otherPropsExtracted}
|
||||||
className={popoverContentVariant({
|
className={popoverContentVariant({
|
||||||
...variantProps,
|
...variantProps,
|
||||||
opened: state.opened,
|
opened: state.opened,
|
||||||
})}
|
})}
|
||||||
ref={(el) => {
|
ref={(el: HTMLDivElement) => {
|
||||||
internalRef.current = el;
|
internalRef.current = el;
|
||||||
if (typeof ref === "function") {
|
if (typeof ref === "function") {
|
||||||
ref(el);
|
ref(el);
|
||||||
@ -238,9 +240,10 @@ const PopoverContent = React.forwardRef<HTMLDivElement, PopoverContentProps>(
|
|||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{children}
|
{children}
|
||||||
</div>
|
</Comp>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
PopoverContent.displayName = "PopoverContent";
|
||||||
|
|
||||||
export { Popover, PopoverTrigger, PopoverContent };
|
export { Popover, PopoverTrigger, PopoverContent };
|
||||||
|
@ -80,5 +80,6 @@ const Switch = React.forwardRef<HTMLInputElement, SwitchProps>((props, ref) => {
|
|||||||
</label>
|
</label>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
Switch.displayName = "Switch";
|
||||||
|
|
||||||
export { Switch };
|
export { Switch };
|
||||||
|
@ -39,7 +39,7 @@ const TabList = (props: TabListProps) => {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const [TabTriggerVariant, resolveTabTriggerVariantProps] = vcn({
|
const [TabTriggerVariant, resolveTabTriggerVariantProps] = vcn({
|
||||||
base: "py-1.5 rounded-md flex-grow transition-all text-sm",
|
base: "py-1.5 rounded-md grow transition-all text-sm",
|
||||||
variants: {
|
variants: {
|
||||||
active: {
|
active: {
|
||||||
true: "bg-white/100 dark:bg-black/100 text-black dark:text-white",
|
true: "bg-white/100 dark:bg-black/100 text-black dark:text-white",
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import { type VariantProps, vcn } from "@pswui-lib";
|
import { type VariantProps, useDocument, vcn } from "@pswui-lib";
|
||||||
import React, { useEffect, useId, useRef } from "react";
|
import React, { useEffect, useId, useRef } from "react";
|
||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
|
|
||||||
@ -168,6 +168,10 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
};
|
};
|
||||||
}, [defaultOption]);
|
}, [defaultOption]);
|
||||||
|
|
||||||
|
const document = useDocument();
|
||||||
|
|
||||||
|
if (!document) return null;
|
||||||
|
|
||||||
const toasterInstance = document.querySelector("div[data-toaster-root]");
|
const toasterInstance = document.querySelector("div[data-toaster-root]");
|
||||||
if (toasterInstance && id !== toasterInstance.id) {
|
if (toasterInstance && id !== toasterInstance.id) {
|
||||||
if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) {
|
if (process.env.NODE_ENV === "development" && !muteDuplicationWarning) {
|
||||||
@ -208,5 +212,6 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
Toaster.displayName = "Toaster";
|
||||||
|
|
||||||
export { Toaster };
|
export { Toaster };
|
||||||
|
@ -71,6 +71,7 @@ const Tooltip = React.forwardRef<HTMLDivElement, TooltipProps>((props, ref) => {
|
|||||||
</TooltipContext.Provider>
|
</TooltipContext.Provider>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
Tooltip.displayName = "Tooltip";
|
||||||
|
|
||||||
const tooltipContentColors = {
|
const tooltipContentColors = {
|
||||||
variants: {
|
variants: {
|
||||||
@ -83,10 +84,10 @@ const tooltipContentColors = {
|
|||||||
};
|
};
|
||||||
|
|
||||||
const [tooltipContentVariant, resolveTooltipContentVariantProps] = vcn({
|
const [tooltipContentVariant, resolveTooltipContentVariantProps] = vcn({
|
||||||
base: `absolute py-1 px-3 rounded-md border opacity-0 transition-all
|
base: `absolute py-1 px-3 rounded-md border opacity-0 transition-all
|
||||||
group-[:not(.controlled):hover]/tooltip:opacity-100 group-[.opened]/tooltip:opacity-100
|
group-[:not(.controlled):hover]/tooltip:opacity-100 group-[.opened]/tooltip:opacity-100
|
||||||
select-none pointer-events-none
|
select-none pointer-events-none
|
||||||
group-[:not(.controlled):hover]/tooltip:select-auto group-[.opened]/tooltip:select-auto group-[:not(.controlled):hover]/tooltip:pointer-events-auto group-[.opened]/tooltip:pointer-events-auto
|
group-[:not(.controlled):hover]/tooltip:select-auto group-[.opened]/tooltip:select-auto group-[:not(.controlled):hover]/tooltip:pointer-events-auto group-[.opened]/tooltip:pointer-events-auto
|
||||||
group-[:not(.controlled):hover]/tooltip:[transition:transform_150ms_ease-out_var(--delay),opacity_150ms_ease-out_var(--delay),background-color_150ms_ease-in-out,color_150ms_ease-in-out,border-color_150ms_ease-in-out]`,
|
group-[:not(.controlled):hover]/tooltip:[transition:transform_150ms_ease-out_var(--delay),opacity_150ms_ease-out_var(--delay),background-color_150ms_ease-in-out,color_150ms_ease-in-out,border-color_150ms_ease-in-out]`,
|
||||||
variants: {
|
variants: {
|
||||||
position: {
|
position: {
|
||||||
@ -139,10 +140,12 @@ const TooltipContent = React.forwardRef<HTMLDivElement, TooltipContentProps>(
|
|||||||
...variantProps,
|
...variantProps,
|
||||||
position: contextState.position,
|
position: contextState.position,
|
||||||
})}
|
})}
|
||||||
|
role="tooltip"
|
||||||
{...rest}
|
{...rest}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
TooltipContent.displayName = "TooltipContent";
|
||||||
|
|
||||||
export { Tooltip, TooltipContent };
|
export { Tooltip, TooltipContent };
|
||||||
|
@ -85,7 +85,10 @@ export const Slot = React.forwardRef<
|
|||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
return React.cloneElement(children, {
|
return React.cloneElement(children, {
|
||||||
...mergeReactProps(safeSlotProps, children.props),
|
...mergeReactProps(
|
||||||
|
safeSlotProps,
|
||||||
|
children.props as Record<string, unknown>,
|
||||||
|
),
|
||||||
ref: combinedRef([
|
ref: combinedRef([
|
||||||
ref,
|
ref,
|
||||||
(children as unknown as { ref: React.Ref<HTMLElement> }).ref,
|
(children as unknown as { ref: React.Ref<HTMLElement> }).ref,
|
||||||
|
@ -1,2 +1,4 @@
|
|||||||
export * from "./vcn";
|
export * from "./vcn";
|
||||||
export * from "./Slot";
|
export * from "./Slot";
|
||||||
|
export * from "./useDocument";
|
||||||
|
export * from "./useAnimatedMount";
|
||||||
|
85
src/pswui/lib/useAnimatedMount.ts
Normal file
85
src/pswui/lib/useAnimatedMount.ts
Normal file
@ -0,0 +1,85 @@
|
|||||||
|
import { type MutableRefObject, useCallback, useEffect, useState } from "react";
|
||||||
|
|
||||||
|
function getCalculatedTransitionDuration(
|
||||||
|
ref: MutableRefObject<HTMLElement>,
|
||||||
|
): number {
|
||||||
|
let transitionDuration: {
|
||||||
|
value: number;
|
||||||
|
unit: string;
|
||||||
|
} | null;
|
||||||
|
if (ref.current.computedStyleMap !== undefined) {
|
||||||
|
transitionDuration = ref.current
|
||||||
|
.computedStyleMap()
|
||||||
|
.get("transition-duration") as { value: number; unit: string };
|
||||||
|
} else {
|
||||||
|
const style = /(\d+(\.\d+)?)(.+)/.exec(
|
||||||
|
window.getComputedStyle(ref.current).transitionDuration,
|
||||||
|
);
|
||||||
|
if (!style) return 0;
|
||||||
|
transitionDuration = {
|
||||||
|
value: Number.parseFloat(style[1] ?? "0"),
|
||||||
|
unit: style[3] ?? style[2] ?? "s",
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
return (
|
||||||
|
transitionDuration.value *
|
||||||
|
({
|
||||||
|
s: 1000,
|
||||||
|
ms: 1,
|
||||||
|
}[transitionDuration.unit] ?? 1)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* isMounted: true isRendered: true isRendered: false isMounted: false
|
||||||
|
* Component Mount Component Appear Component Disappear Component Unmount
|
||||||
|
* v v v v
|
||||||
|
* |-|=================|------------------------|======================|-|
|
||||||
|
*/
|
||||||
|
|
||||||
|
function useAnimatedMount(
|
||||||
|
visible: boolean,
|
||||||
|
ref: MutableRefObject<HTMLElement | null>,
|
||||||
|
callbacks?: { onMount: () => void; onUnmount: () => void },
|
||||||
|
) {
|
||||||
|
const [state, setState] = useState<{
|
||||||
|
isMounted: boolean;
|
||||||
|
isRendered: boolean;
|
||||||
|
}>({ isMounted: visible, isRendered: visible });
|
||||||
|
|
||||||
|
const umountCallback = useCallback(() => {
|
||||||
|
setState((p) => ({ ...p, isRendered: false }));
|
||||||
|
|
||||||
|
const calculatedTransitionDuration = ref.current
|
||||||
|
? getCalculatedTransitionDuration(ref as MutableRefObject<HTMLElement>)
|
||||||
|
: 0;
|
||||||
|
|
||||||
|
setTimeout(() => {
|
||||||
|
setState((p) => ({ ...p, isMounted: false }));
|
||||||
|
callbacks?.onUnmount?.();
|
||||||
|
}, calculatedTransitionDuration);
|
||||||
|
}, [ref, callbacks]);
|
||||||
|
|
||||||
|
const mountCallback = useCallback(() => {
|
||||||
|
setState((p) => ({ ...p, isMounted: true }));
|
||||||
|
callbacks?.onMount?.();
|
||||||
|
requestAnimationFrame(function onMount() {
|
||||||
|
if (!ref.current) return requestAnimationFrame(onMount);
|
||||||
|
setState((p) => ({ ...p, isRendered: true }));
|
||||||
|
});
|
||||||
|
}, [ref.current, callbacks]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
console.log(state);
|
||||||
|
if (!visible && state.isRendered) {
|
||||||
|
umountCallback();
|
||||||
|
} else if (visible && !state.isMounted) {
|
||||||
|
mountCallback();
|
||||||
|
}
|
||||||
|
}, [state, visible, mountCallback, umountCallback]);
|
||||||
|
|
||||||
|
return state;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { getCalculatedTransitionDuration, useAnimatedMount };
|
21
src/pswui/lib/useDocument.ts
Normal file
21
src/pswui/lib/useDocument.ts
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
"use client";
|
||||||
|
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This hook allows components to use `document` as like they're always in the client side.
|
||||||
|
* Return undefined if there is no `document` (which represents it's server side) or initial render(to avoid hydration error).
|
||||||
|
*/
|
||||||
|
function useDocument(): undefined | Document {
|
||||||
|
const [initialRender, setInitialState] = useState(true);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
setInitialState(false);
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (typeof document === "undefined" || initialRender) return undefined;
|
||||||
|
|
||||||
|
return document;
|
||||||
|
}
|
||||||
|
|
||||||
|
export { useDocument };
|
@ -270,6 +270,7 @@ export function vcn<
|
|||||||
for (const [variantName, variantKey] of Object.entries(
|
for (const [variantName, variantKey] of Object.entries(
|
||||||
otherVariantProps,
|
otherVariantProps,
|
||||||
) as VariantKVEntry<V>) {
|
) as VariantKVEntry<V>) {
|
||||||
|
if (typeof variantKey === "undefined") continue;
|
||||||
kv[variantName] = variantKey;
|
kv[variantName] = variantKey;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,8 +1,41 @@
|
|||||||
@import url("https://cdn.jsdelivr.net/gh/wanteddev/wanted-sans@v1.0.3/packages/wanted-sans/fonts/webfonts/variable/split/WantedSansVariable.min.css");
|
@import url('https://cdn.jsdelivr.net/gh/wanteddev/wanted-sans@v1.0.3/packages/wanted-sans/fonts/webfonts/variable/split/WantedSansVariable.min.css')
|
||||||
@tailwind base;
|
layer(base);
|
||||||
@tailwind components;
|
@import 'tailwindcss';
|
||||||
@tailwind utilities;
|
|
||||||
|
@plugin '@tailwindcss/typography';
|
||||||
|
@plugin 'tailwind-scrollbar';
|
||||||
|
|
||||||
|
@source '../{components,stories,src}/**/*.{js,jsx,ts,tsx,css,mdx}';
|
||||||
|
|
||||||
|
@custom-variant dark {
|
||||||
|
@media (prefers-color-scheme: dark) {
|
||||||
|
&:is(.system *) {
|
||||||
|
@slot;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
&:is(.dark *) {
|
||||||
|
@slot;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
The default border color has changed to `currentColor` in Tailwind CSS v4,
|
||||||
|
so we've added these compatibility styles to make sure everything still
|
||||||
|
looks the same as it did with Tailwind CSS v3.
|
||||||
|
|
||||||
|
If we ever want to remove these styles, we need to add an explicit border
|
||||||
|
color utility to any element that depends on these defaults.
|
||||||
|
*/
|
||||||
|
@layer base {
|
||||||
|
*,
|
||||||
|
::after,
|
||||||
|
::before,
|
||||||
|
::backdrop,
|
||||||
|
::file-selector-button {
|
||||||
|
border-color: var(--color-gray-200, currentColor);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@layer base {
|
@layer base {
|
||||||
:root {
|
:root {
|
||||||
--dark-bg-color: #000;
|
--dark-bg-color: #000;
|
||||||
|
@ -1,15 +0,0 @@
|
|||||||
/** @type {import('tailwindcss').Config} */
|
|
||||||
module.exports = {
|
|
||||||
content: ["./{components,stories,src}/**/*.{js,jsx,ts,tsx,css,mdx}"],
|
|
||||||
darkMode: [
|
|
||||||
"variant",
|
|
||||||
[
|
|
||||||
"@media (prefers-color-scheme: dark) { &:is(.system *) }",
|
|
||||||
"&:is(.dark *)",
|
|
||||||
],
|
|
||||||
],
|
|
||||||
theme: {
|
|
||||||
extend: {},
|
|
||||||
},
|
|
||||||
plugins: [require("@tailwindcss/typography"), require("tailwind-scrollbar")],
|
|
||||||
};
|
|
@ -2,10 +2,10 @@ import { resolve } from "node:path";
|
|||||||
import mdx from "@mdx-js/rollup";
|
import mdx from "@mdx-js/rollup";
|
||||||
import withToc from "@stefanprobst/rehype-extract-toc";
|
import withToc from "@stefanprobst/rehype-extract-toc";
|
||||||
import withTocExport from "@stefanprobst/rehype-extract-toc/mdx";
|
import withTocExport from "@stefanprobst/rehype-extract-toc/mdx";
|
||||||
|
import tailwindcss from "@tailwindcss/vite";
|
||||||
import react from "@vitejs/plugin-react";
|
import react from "@vitejs/plugin-react";
|
||||||
import withSlug from "rehype-slug";
|
import withSlug from "rehype-slug";
|
||||||
import remarkGfm from "remark-gfm";
|
import remarkGfm from "remark-gfm";
|
||||||
import tailwindcss from "tailwindcss";
|
|
||||||
import { defineConfig } from "vite";
|
import { defineConfig } from "vite";
|
||||||
import dynamicImport from "vite-plugin-dynamic-import";
|
import dynamicImport from "vite-plugin-dynamic-import";
|
||||||
|
|
||||||
@ -13,17 +13,13 @@ import dynamicImport from "vite-plugin-dynamic-import";
|
|||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
plugins: [
|
plugins: [
|
||||||
react(),
|
react(),
|
||||||
|
tailwindcss(),
|
||||||
mdx({
|
mdx({
|
||||||
rehypePlugins: [withSlug, withToc, withTocExport],
|
rehypePlugins: [withSlug, withToc, withTocExport],
|
||||||
remarkPlugins: [remarkGfm],
|
remarkPlugins: [remarkGfm],
|
||||||
}),
|
}),
|
||||||
dynamicImport(),
|
dynamicImport(),
|
||||||
],
|
],
|
||||||
css: {
|
|
||||||
postcss: {
|
|
||||||
plugins: [tailwindcss()],
|
|
||||||
},
|
|
||||||
},
|
|
||||||
resolve: {
|
resolve: {
|
||||||
alias: {
|
alias: {
|
||||||
"@pswui": resolve(__dirname, "./src/pswui/components"),
|
"@pswui": resolve(__dirname, "./src/pswui/components"),
|
||||||
|
Loading…
x
Reference in New Issue
Block a user