feat: reinstalled directory components
This commit is contained in:
parent
889a8c01b3
commit
0dcf22d6a8
@ -1,32 +1,13 @@
|
|||||||
import React, { Dispatch, SetStateAction, useState } from "react";
|
import React, { useState } from "react";
|
||||||
import { Slot, VariantProps, vcn } from "@pswui-lib";
|
import { Slot, VariantProps, vcn } from "@pswui-lib";
|
||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
|
|
||||||
/**
|
import {
|
||||||
* =========================
|
DialogContext,
|
||||||
* DialogContext
|
|
||||||
* =========================
|
|
||||||
*/
|
|
||||||
|
|
||||||
interface DialogContext {
|
|
||||||
opened: boolean;
|
|
||||||
}
|
|
||||||
|
|
||||||
const initialDialogContext: DialogContext = { opened: false };
|
|
||||||
const DialogContext = React.createContext<
|
|
||||||
[DialogContext, Dispatch<SetStateAction<DialogContext>>]
|
|
||||||
>([
|
|
||||||
initialDialogContext,
|
initialDialogContext,
|
||||||
() => {
|
useDialogContext,
|
||||||
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
IDialogContext,
|
||||||
console.warn(
|
} from "./Context";
|
||||||
"It seems like you're using DialogContext outside of a provider.",
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
|
|
||||||
const useDialogContext = () => React.useContext(DialogContext);
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* =========================
|
* =========================
|
||||||
@ -39,7 +20,7 @@ interface DialogRootProps {
|
|||||||
}
|
}
|
||||||
|
|
||||||
const DialogRoot = ({ children }: DialogRootProps) => {
|
const DialogRoot = ({ children }: DialogRootProps) => {
|
||||||
const state = useState<DialogContext>(initialDialogContext);
|
const state = useState<IDialogContext>(initialDialogContext);
|
||||||
return (
|
return (
|
||||||
<DialogContext.Provider value={state}>{children}</DialogContext.Provider>
|
<DialogContext.Provider value={state}>{children}</DialogContext.Provider>
|
||||||
);
|
);
|
||||||
@ -411,7 +392,6 @@ const DialogFooter = React.forwardRef<HTMLDivElement, DialogFooterProps>(
|
|||||||
);
|
);
|
||||||
|
|
||||||
export {
|
export {
|
||||||
useDialogContext,
|
|
||||||
DialogRoot,
|
DialogRoot,
|
||||||
DialogTrigger,
|
DialogTrigger,
|
||||||
DialogOverlay,
|
DialogOverlay,
|
27
src/pswui/components/Dialog/Context.ts
Normal file
27
src/pswui/components/Dialog/Context.ts
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
import { Dispatch, SetStateAction, useContext, createContext } from "react";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* =========================
|
||||||
|
* DialogContext
|
||||||
|
* =========================
|
||||||
|
*/
|
||||||
|
|
||||||
|
export interface IDialogContext {
|
||||||
|
opened: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const initialDialogContext: IDialogContext = { opened: false };
|
||||||
|
export const DialogContext = createContext<
|
||||||
|
[IDialogContext, Dispatch<SetStateAction<IDialogContext>>]
|
||||||
|
>([
|
||||||
|
initialDialogContext,
|
||||||
|
() => {
|
||||||
|
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
||||||
|
console.warn(
|
||||||
|
"It seems like you're using DialogContext outside of a provider.",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
|
||||||
|
export const useDialogContext = () => useContext(DialogContext);
|
2
src/pswui/components/Dialog/index.ts
Normal file
2
src/pswui/components/Dialog/index.ts
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
export * from "./Component";
|
||||||
|
export { useDialogContext } from "./Context";
|
@ -1,30 +1,7 @@
|
|||||||
import { AsChild, Slot, VariantProps, vcn } from "@pswui-lib";
|
import { AsChild, Slot, VariantProps, vcn } from "@pswui-lib";
|
||||||
import React from "react";
|
import React from "react";
|
||||||
|
|
||||||
interface Tab {
|
import { TabContextBody, TabContext, Tab } from "./Context";
|
||||||
name: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
interface TabContextBody {
|
|
||||||
tabs: Tab[];
|
|
||||||
active: [number, string] /* index, name */;
|
|
||||||
}
|
|
||||||
|
|
||||||
const TabContext = React.createContext<
|
|
||||||
[TabContextBody, React.Dispatch<React.SetStateAction<TabContextBody>>]
|
|
||||||
>([
|
|
||||||
{
|
|
||||||
tabs: [],
|
|
||||||
active: [0, ""],
|
|
||||||
},
|
|
||||||
() => {
|
|
||||||
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
|
||||||
console.warn(
|
|
||||||
"It seems like you're using TabContext outside of provider.",
|
|
||||||
);
|
|
||||||
}
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
|
|
||||||
interface TabProviderProps {
|
interface TabProviderProps {
|
||||||
defaultName: string;
|
defaultName: string;
|
||||||
@ -40,77 +17,6 @@ const TabProvider = ({ defaultName, children }: TabProviderProps) => {
|
|||||||
return <TabContext.Provider value={state}>{children}</TabContext.Provider>;
|
return <TabContext.Provider value={state}>{children}</TabContext.Provider>;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Provides current state for tab, using context.
|
|
||||||
* Also provides functions to control state.
|
|
||||||
*/
|
|
||||||
const useTabState = () => {
|
|
||||||
const [state, setState] = React.useContext(TabContext);
|
|
||||||
|
|
||||||
function getActiveTab() {
|
|
||||||
return state.active;
|
|
||||||
}
|
|
||||||
|
|
||||||
function setActiveTab(name: string): void;
|
|
||||||
function setActiveTab(index: number): void;
|
|
||||||
function setActiveTab(param: string | number) {
|
|
||||||
if (typeof param === "number") {
|
|
||||||
if (param < 0 || param >= state.tabs.length) {
|
|
||||||
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
|
||||||
console.error(
|
|
||||||
`Invalid index passed to setActiveTab: ${param}, valid indices are 0 to ${
|
|
||||||
state.tabs.length - 1
|
|
||||||
}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setState((prev) => {
|
|
||||||
return {
|
|
||||||
...prev,
|
|
||||||
active: [param, prev.tabs[param].name],
|
|
||||||
};
|
|
||||||
});
|
|
||||||
} else if (typeof param === "string") {
|
|
||||||
const index = state.tabs.findIndex((tab) => tab.name === param);
|
|
||||||
if (index === -1) {
|
|
||||||
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
|
||||||
console.error(
|
|
||||||
`Invalid name passed to setActiveTab: ${param}, valid names are ${state.tabs
|
|
||||||
.map((tab) => tab.name)
|
|
||||||
.join(", ")}`,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setActiveTab(index);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function setPreviousActive() {
|
|
||||||
if (state.active[0] === 0) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setActiveTab(state.active[0] - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
function setNextActive() {
|
|
||||||
if (state.active[0] === state.tabs.length - 1) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setActiveTab(state.active[0] + 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
return {
|
|
||||||
getActiveTab,
|
|
||||||
setActiveTab,
|
|
||||||
setPreviousActive,
|
|
||||||
setNextActive,
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
const [TabListVariant, resolveTabListVariantProps] = vcn({
|
const [TabListVariant, resolveTabListVariantProps] = vcn({
|
||||||
base: "flex flex-row bg-gray-100 dark:bg-neutral-800 rounded-lg p-1.5 gap-1",
|
base: "flex flex-row bg-gray-100 dark:bg-neutral-800 rounded-lg p-1.5 gap-1",
|
||||||
variants: {},
|
variants: {},
|
||||||
@ -169,6 +75,7 @@ const TabTrigger = (props: TabTriggerProps) => {
|
|||||||
};
|
};
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
}, [name]);
|
}, [name]);
|
||||||
|
|
||||||
const Comp = props.asChild ? Slot : "button";
|
const Comp = props.asChild ? Slot : "button";
|
||||||
@ -226,4 +133,4 @@ const TabContent = (props: TabContentProps) => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
export { TabProvider, useTabState, TabList, TabTrigger, TabContent };
|
export { TabProvider, TabList, TabTrigger, TabContent };
|
26
src/pswui/components/Tabs/Context.ts
Normal file
26
src/pswui/components/Tabs/Context.ts
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
import React from "react";
|
||||||
|
|
||||||
|
export interface Tab {
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface TabContextBody {
|
||||||
|
tabs: Tab[];
|
||||||
|
active: [number, string] /* index, name */;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const TabContext = React.createContext<
|
||||||
|
[TabContextBody, React.Dispatch<React.SetStateAction<TabContextBody>>]
|
||||||
|
>([
|
||||||
|
{
|
||||||
|
tabs: [],
|
||||||
|
active: [0, ""],
|
||||||
|
},
|
||||||
|
() => {
|
||||||
|
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
||||||
|
console.warn(
|
||||||
|
"It seems like you're using TabContext outside of provider.",
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
]);
|
74
src/pswui/components/Tabs/Hook.ts
Normal file
74
src/pswui/components/Tabs/Hook.ts
Normal file
@ -0,0 +1,74 @@
|
|||||||
|
import React from "react";
|
||||||
|
|
||||||
|
import { TabContext } from "./Context";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Provides current state for tab, using context.
|
||||||
|
* Also provides functions to control state.
|
||||||
|
*/
|
||||||
|
export const useTabState = () => {
|
||||||
|
const [state, setState] = React.useContext(TabContext);
|
||||||
|
|
||||||
|
function getActiveTab() {
|
||||||
|
return state.active;
|
||||||
|
}
|
||||||
|
|
||||||
|
function setActiveTab(name: string): void;
|
||||||
|
function setActiveTab(index: number): void;
|
||||||
|
function setActiveTab(param: string | number) {
|
||||||
|
if (typeof param === "number") {
|
||||||
|
if (param < 0 || param >= state.tabs.length) {
|
||||||
|
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
||||||
|
console.error(
|
||||||
|
`Invalid index passed to setActiveTab: ${param}, valid indices are 0 to ${
|
||||||
|
state.tabs.length - 1
|
||||||
|
}`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setState((prev) => {
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
active: [param, prev.tabs[param].name],
|
||||||
|
};
|
||||||
|
});
|
||||||
|
} else if (typeof param === "string") {
|
||||||
|
const index = state.tabs.findIndex((tab) => tab.name === param);
|
||||||
|
if (index === -1) {
|
||||||
|
if (process.env.NODE_ENV && process.env.NODE_ENV === "development") {
|
||||||
|
console.error(
|
||||||
|
`Invalid name passed to setActiveTab: ${param}, valid names are ${state.tabs
|
||||||
|
.map((tab) => tab.name)
|
||||||
|
.join(", ")}`,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
setActiveTab(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function setPreviousActive() {
|
||||||
|
if (state.active[0] === 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setActiveTab(state.active[0] - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
function setNextActive() {
|
||||||
|
if (state.active[0] === state.tabs.length - 1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setActiveTab(state.active[0] + 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
getActiveTab,
|
||||||
|
setActiveTab,
|
||||||
|
setPreviousActive,
|
||||||
|
setNextActive,
|
||||||
|
};
|
||||||
|
};
|
2
src/pswui/components/Tabs/index.ts
Normal file
2
src/pswui/components/Tabs/index.ts
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
export * from "./Component";
|
||||||
|
export * from "./Hook";
|
@ -2,148 +2,19 @@ import React, { useEffect, useId, useRef } from "react";
|
|||||||
import ReactDOM from "react-dom";
|
import ReactDOM from "react-dom";
|
||||||
import { VariantProps, vcn } from "@pswui-lib";
|
import { VariantProps, vcn } from "@pswui-lib";
|
||||||
|
|
||||||
interface ToastOption {
|
import { toastVariant } from "./Variant";
|
||||||
closeButton: boolean;
|
import {
|
||||||
closeTimeout: number | null;
|
ToastOption,
|
||||||
}
|
toasts,
|
||||||
|
subscribeSingle,
|
||||||
const defaultToastOption: ToastOption = {
|
getSingleSnapshot,
|
||||||
closeButton: true,
|
notifySingle,
|
||||||
closeTimeout: 3000,
|
|
||||||
};
|
|
||||||
|
|
||||||
const toastColors = {
|
|
||||||
background: "bg-white dark:bg-black",
|
|
||||||
borders: {
|
|
||||||
default: "border-black/10 dark:border-white/20",
|
|
||||||
error: "border-red-500/80",
|
|
||||||
success: "border-green-500/80",
|
|
||||||
warning: "border-yellow-500/80",
|
|
||||||
loading: "border-black/50 dark:border-white/50 animate-pulse",
|
|
||||||
},
|
|
||||||
};
|
|
||||||
|
|
||||||
const [toastVariant] = vcn({
|
|
||||||
base: `flex flex-col gap-2 border p-4 rounded-lg pr-8 pointer-events-auto ${toastColors.background} relative transition-all duration-150`,
|
|
||||||
variants: {
|
|
||||||
status: {
|
|
||||||
default: toastColors.borders.default,
|
|
||||||
error: toastColors.borders.error,
|
|
||||||
success: toastColors.borders.success,
|
|
||||||
warning: toastColors.borders.warning,
|
|
||||||
loading: toastColors.borders.loading,
|
|
||||||
},
|
|
||||||
life: {
|
|
||||||
born: "-translate-y-full md:translate-y-full scale-90 ease-[cubic-bezier(0,.6,.7,1)]",
|
|
||||||
normal: "translate-y-0 scale-100 ease-[cubic-bezier(0,.6,.7,1)]",
|
|
||||||
dead: "-translate-y-full md:translate-y-full scale-90 ease-[cubic-bezier(.6,0,1,.7)]",
|
|
||||||
},
|
|
||||||
},
|
|
||||||
defaults: {
|
|
||||||
status: "default",
|
|
||||||
life: "born",
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
interface ToastBody extends Omit<VariantProps<typeof toastVariant>, "preset"> {
|
|
||||||
title: string;
|
|
||||||
description: string;
|
|
||||||
}
|
|
||||||
|
|
||||||
let index = 0;
|
|
||||||
let toasts: Record<
|
|
||||||
`${number}`,
|
|
||||||
ToastBody & Partial<ToastOption> & { subscribers: (() => void)[] }
|
|
||||||
> = {};
|
|
||||||
let subscribers: (() => void)[] = [];
|
|
||||||
|
|
||||||
/**
|
|
||||||
* ====
|
|
||||||
* Controls
|
|
||||||
* ====
|
|
||||||
*/
|
|
||||||
|
|
||||||
function subscribe(callback: () => void) {
|
|
||||||
subscribers.push(callback);
|
|
||||||
return () => {
|
|
||||||
subscribers = subscribers.filter((subscriber) => subscriber !== callback);
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function getSnapshot() {
|
|
||||||
return { ...toasts };
|
|
||||||
}
|
|
||||||
|
|
||||||
function subscribeSingle(id: `${number}`) {
|
|
||||||
return (callback: () => void) => {
|
|
||||||
toasts[id].subscribers.push(callback);
|
|
||||||
return () => {
|
|
||||||
toasts[id].subscribers = toasts[id].subscribers.filter(
|
|
||||||
(subscriber) => subscriber !== callback,
|
|
||||||
);
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function getSingleSnapshot(id: `${number}`) {
|
|
||||||
return () => {
|
|
||||||
return {
|
|
||||||
...toasts[id],
|
|
||||||
};
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function notify() {
|
|
||||||
subscribers.forEach((subscriber) => subscriber());
|
|
||||||
}
|
|
||||||
|
|
||||||
function notifySingle(id: `${number}`) {
|
|
||||||
toasts[id].subscribers.forEach((subscriber) => subscriber());
|
|
||||||
}
|
|
||||||
|
|
||||||
function close(id: `${number}`) {
|
|
||||||
toasts[id] = {
|
|
||||||
...toasts[id],
|
|
||||||
life: "dead",
|
|
||||||
};
|
|
||||||
notifySingle(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
function update(
|
|
||||||
id: `${number}`,
|
|
||||||
toast: Partial<Omit<ToastBody, "life"> & Partial<ToastOption>>,
|
|
||||||
) {
|
|
||||||
toasts[id] = {
|
|
||||||
...toasts[id],
|
|
||||||
...toast,
|
|
||||||
};
|
|
||||||
notifySingle(id);
|
|
||||||
}
|
|
||||||
|
|
||||||
function addToast(toast: Omit<ToastBody, "life"> & Partial<ToastOption>) {
|
|
||||||
const id: `${number}` = `${index}`;
|
|
||||||
toasts[id] = {
|
|
||||||
...toast,
|
|
||||||
subscribers: [],
|
|
||||||
life: "born",
|
|
||||||
};
|
|
||||||
index += 1;
|
|
||||||
notify();
|
|
||||||
|
|
||||||
return {
|
|
||||||
update: (toast: Partial<Omit<ToastBody, "life"> & Partial<ToastOption>>) =>
|
|
||||||
update(id, toast),
|
|
||||||
close: () => close(id),
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function useToast() {
|
|
||||||
return {
|
|
||||||
toast: addToast,
|
|
||||||
update,
|
|
||||||
close,
|
close,
|
||||||
};
|
notify,
|
||||||
}
|
defaultToastOption,
|
||||||
|
subscribe,
|
||||||
|
getSnapshot,
|
||||||
|
} from "./Store";
|
||||||
|
|
||||||
const ToastTemplate = ({
|
const ToastTemplate = ({
|
||||||
id,
|
id,
|
||||||
@ -161,7 +32,7 @@ const ToastTemplate = ({
|
|||||||
subscribeSingle(id)(() => {
|
subscribeSingle(id)(() => {
|
||||||
setToast(getSingleSnapshot(id)());
|
setToast(getSingleSnapshot(id)());
|
||||||
});
|
});
|
||||||
}, []);
|
}, [id]);
|
||||||
|
|
||||||
const toastData = {
|
const toastData = {
|
||||||
...globalOption,
|
...globalOption,
|
||||||
@ -225,7 +96,7 @@ const ToastTemplate = ({
|
|||||||
}, calculatedTransitionDuration);
|
}, calculatedTransitionDuration);
|
||||||
return () => clearTimeout(timeout);
|
return () => clearTimeout(timeout);
|
||||||
}
|
}
|
||||||
}, [toastData.life, toastData.closeTimeout, toastData.closeButton]);
|
}, [id, toastData.life, toastData.closeTimeout, toastData.closeButton]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div
|
<div
|
||||||
@ -280,10 +151,9 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
const internalRef = useRef<HTMLDivElement | null>(null);
|
const internalRef = useRef<HTMLDivElement | null>(null);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const unsubscribe = subscribe(() => {
|
return subscribe(() => {
|
||||||
setToastList(getSnapshot());
|
setToastList(getSnapshot());
|
||||||
});
|
});
|
||||||
return unsubscribe;
|
|
||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
const option = React.useMemo(() => {
|
const option = React.useMemo(() => {
|
||||||
@ -308,7 +178,7 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
{ReactDOM.createPortal(
|
{ReactDOM.createPortal(
|
||||||
<div
|
<div
|
||||||
{...otherPropsExtracted}
|
{...otherPropsExtracted}
|
||||||
data-toaster-root
|
data-toaster-root={true}
|
||||||
className={toasterVariant(variantProps)}
|
className={toasterVariant(variantProps)}
|
||||||
ref={(el) => {
|
ref={(el) => {
|
||||||
internalRef.current = el;
|
internalRef.current = el;
|
||||||
@ -334,4 +204,4 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
export { Toaster, useToast };
|
export { Toaster };
|
9
src/pswui/components/Toast/Hook.ts
Normal file
9
src/pswui/components/Toast/Hook.ts
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
import { addToast, update, close } from "./Store";
|
||||||
|
|
||||||
|
export function useToast() {
|
||||||
|
return {
|
||||||
|
toast: addToast,
|
||||||
|
update,
|
||||||
|
close,
|
||||||
|
};
|
||||||
|
}
|
100
src/pswui/components/Toast/Store.ts
Normal file
100
src/pswui/components/Toast/Store.ts
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
import { ToastBody } from "./Variant";
|
||||||
|
|
||||||
|
export interface ToastOption {
|
||||||
|
closeButton: boolean;
|
||||||
|
closeTimeout: number | null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const defaultToastOption: ToastOption = {
|
||||||
|
closeButton: true,
|
||||||
|
closeTimeout: 3000,
|
||||||
|
};
|
||||||
|
|
||||||
|
let index = 0;
|
||||||
|
export const toasts: Record<
|
||||||
|
`${number}`,
|
||||||
|
ToastBody & Partial<ToastOption> & { subscribers: (() => void)[] }
|
||||||
|
> = {};
|
||||||
|
let subscribers: (() => void)[] = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* ====
|
||||||
|
* Controls
|
||||||
|
* ====
|
||||||
|
*/
|
||||||
|
|
||||||
|
export function subscribe(callback: () => void) {
|
||||||
|
subscribers.push(callback);
|
||||||
|
return () => {
|
||||||
|
subscribers = subscribers.filter((subscriber) => subscriber !== callback);
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getSnapshot() {
|
||||||
|
return { ...toasts };
|
||||||
|
}
|
||||||
|
|
||||||
|
export function subscribeSingle(id: `${number}`) {
|
||||||
|
return (callback: () => void) => {
|
||||||
|
toasts[id].subscribers.push(callback);
|
||||||
|
return () => {
|
||||||
|
toasts[id].subscribers = toasts[id].subscribers.filter(
|
||||||
|
(subscriber) => subscriber !== callback,
|
||||||
|
);
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getSingleSnapshot(id: `${number}`) {
|
||||||
|
return () => {
|
||||||
|
return {
|
||||||
|
...toasts[id],
|
||||||
|
};
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
export function notify() {
|
||||||
|
subscribers.forEach((subscriber) => subscriber());
|
||||||
|
}
|
||||||
|
|
||||||
|
export function notifySingle(id: `${number}`) {
|
||||||
|
toasts[id].subscribers.forEach((subscriber) => subscriber());
|
||||||
|
}
|
||||||
|
|
||||||
|
export function close(id: `${number}`) {
|
||||||
|
toasts[id] = {
|
||||||
|
...toasts[id],
|
||||||
|
life: "dead",
|
||||||
|
};
|
||||||
|
notifySingle(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function update(
|
||||||
|
id: `${number}`,
|
||||||
|
toast: Partial<Omit<ToastBody, "life"> & Partial<ToastOption>>,
|
||||||
|
) {
|
||||||
|
toasts[id] = {
|
||||||
|
...toasts[id],
|
||||||
|
...toast,
|
||||||
|
};
|
||||||
|
notifySingle(id);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function addToast(
|
||||||
|
toast: Omit<ToastBody, "life"> & Partial<ToastOption>,
|
||||||
|
) {
|
||||||
|
const id: `${number}` = `${index}`;
|
||||||
|
toasts[id] = {
|
||||||
|
...toast,
|
||||||
|
subscribers: [],
|
||||||
|
life: "born",
|
||||||
|
};
|
||||||
|
index += 1;
|
||||||
|
notify();
|
||||||
|
|
||||||
|
return {
|
||||||
|
update: (toast: Partial<Omit<ToastBody, "life"> & Partial<ToastOption>>) =>
|
||||||
|
update(id, toast),
|
||||||
|
close: () => close(id),
|
||||||
|
};
|
||||||
|
}
|
40
src/pswui/components/Toast/Variant.ts
Normal file
40
src/pswui/components/Toast/Variant.ts
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
import { VariantProps, vcn } from "@pswui-lib";
|
||||||
|
|
||||||
|
const toastColors = {
|
||||||
|
background: "bg-white dark:bg-black",
|
||||||
|
borders: {
|
||||||
|
default: "border-black/10 dark:border-white/20",
|
||||||
|
error: "border-red-500/80",
|
||||||
|
success: "border-green-500/80",
|
||||||
|
warning: "border-yellow-500/80",
|
||||||
|
loading: "border-black/50 dark:border-white/50 animate-pulse",
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const [toastVariant, resolveToastVariantProps] = vcn({
|
||||||
|
base: `flex flex-col gap-2 border p-4 rounded-lg pr-8 pointer-events-auto ${toastColors.background} relative transition-all duration-150`,
|
||||||
|
variants: {
|
||||||
|
status: {
|
||||||
|
default: toastColors.borders.default,
|
||||||
|
error: toastColors.borders.error,
|
||||||
|
success: toastColors.borders.success,
|
||||||
|
warning: toastColors.borders.warning,
|
||||||
|
loading: toastColors.borders.loading,
|
||||||
|
},
|
||||||
|
life: {
|
||||||
|
born: "-translate-y-full md:translate-y-full scale-90 ease-[cubic-bezier(0,.6,.7,1)]",
|
||||||
|
normal: "translate-y-0 scale-100 ease-[cubic-bezier(0,.6,.7,1)]",
|
||||||
|
dead: "-translate-y-full md:translate-y-full scale-90 ease-[cubic-bezier(.6,0,1,.7)]",
|
||||||
|
},
|
||||||
|
},
|
||||||
|
defaults: {
|
||||||
|
status: "default",
|
||||||
|
life: "born",
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
export interface ToastBody
|
||||||
|
extends Omit<VariantProps<typeof toastVariant>, "preset"> {
|
||||||
|
title: string;
|
||||||
|
description: string;
|
||||||
|
}
|
0
src/pswui/components/Toast/index.ts
Normal file
0
src/pswui/components/Toast/index.ts
Normal file
Loading…
x
Reference in New Issue
Block a user