fix: split component file
This commit is contained in:
parent
6c35e54875
commit
7dd3bf7d9e
@ -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;
|
|
||||||
const 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,
|
||||||
@ -333,4 +204,4 @@ const Toaster = React.forwardRef<HTMLDivElement, ToasterProps>((props, ref) => {
|
|||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|
||||||
export { Toaster, useToast };
|
export { Toaster };
|
9
packages/react/components/Toast/Hook.ts
Normal file
9
packages/react/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
packages/react/components/Toast/Store.ts
Normal file
100
packages/react/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
packages/react/components/Toast/Variant.ts
Normal file
40
packages/react/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
packages/react/components/Toast/index.ts
Normal file
0
packages/react/components/Toast/index.ts
Normal file
Loading…
x
Reference in New Issue
Block a user