dashboard: fix toaster

This commit is contained in:
Carl-Gerhard Lindesvärd
2024-02-12 07:43:33 +01:00
parent 6f2aeffdff
commit 5afc49b7e4
20 changed files with 49 additions and 266 deletions

View File

@@ -24,25 +24,20 @@ export function ListDashboards({ dashboards }: ListDashboardsProps) {
const deletion = api.dashboard.delete.useMutation({
onError: (error, variables) => {
return handleErrorToastOptions({
action: (
<ToastAction
altText="Force delete"
onClick={() => {
deletion.mutate({
forceDelete: true,
id: variables.id,
});
}}
>
Force delete
</ToastAction>
),
action: {
label: 'Force delete',
onClick: () => {
deletion.mutate({
forceDelete: true,
id: variables.id,
});
},
},
})(error);
},
onSuccess() {
router.refresh();
toast({
title: 'Success',
toast('Success', {
description: 'Dashboard deleted.',
});
},

View File

@@ -25,13 +25,12 @@ export default function EditOrganization({
const router = useRouter();
const { register, handleSubmit, formState, reset } = useForm<IForm>({
defaultValues: organization,
defaultValues: organization ?? undefined,
});
const mutation = api.organization.update.useMutation({
onSuccess(res) {
toast({
title: 'Organization updated',
toast('Organization updated', {
description: 'Your organization has been updated.',
});
reset(res);
@@ -57,7 +56,7 @@ export default function EditOrganization({
<InputWithLabel
label="Name"
{...register('name')}
defaultValue={organization.name}
defaultValue={organization?.name}
/>
</WidgetBody>
</Widget>

View File

@@ -27,8 +27,7 @@ export function InviteUser() {
const mutation = api.organization.inviteUser.useMutation({
onSuccess() {
toast({
title: 'User invited!',
toast('User invited!', {
description: 'The user has been invited to the organization.',
});
reset();

View File

@@ -35,8 +35,7 @@ export default function EditProfile({ profile }: EditProfileProps) {
const mutation = api.user.update.useMutation({
onSuccess(res) {
toast({
title: 'Profile updated',
toast('Profile updated', {
description: 'Your profile has been updated.',
});
reset(res);

View File

@@ -1,8 +1,8 @@
import type { Toast } from '@/components/ui/use-toast';
import type { AppRouter } from '@/server/api/root';
import type { TRPCClientErrorBase } from '@trpc/react-query';
import { createTRPCReact } from '@trpc/react-query';
import type { inferRouterInputs, inferRouterOutputs } from '@trpc/server';
import type { ExternalToast } from 'sonner';
import { toast } from 'sonner';
export const api = createTRPCReact<AppRouter>({});
@@ -24,16 +24,14 @@ export type IChartData = RouterOutputs['chart']['chart'];
export type IChartSerieDataItem = IChartData['series'][number]['data'][number];
export function handleError(error: TRPCClientErrorBase<any>) {
toast({
title: 'Error',
toast('Error', {
description: error.message,
});
}
export function handleErrorToastOptions(options: Toast) {
export function handleErrorToastOptions(options: ExternalToast) {
return function (error: TRPCClientErrorBase<any>) {
toast({
title: 'Error',
toast('Error', {
description: error.message,
...options,
});

View File

@@ -2,7 +2,6 @@
import React, { useRef, useState } from 'react';
import { api } from '@/app/_trpc/client';
import { Toaster } from '@/components/ui/toaster';
import { TooltipProvider } from '@/components/ui/tooltip';
import { ModalProvider } from '@/modals';
import type { AppStore } from '@/redux';
@@ -50,7 +49,6 @@ export default function Providers({ children }: { children: React.ReactNode }) {
<QueryClientProvider client={queryClient}>
<TooltipProvider delayDuration={200}>
{children}
<Toaster />
<ModalProvider />
</TooltipProvider>
</QueryClientProvider>

View File

@@ -6,6 +6,7 @@ import type { IClientWithProject } from '@/types';
import { clipboard } from '@/utils/clipboard';
import { MoreHorizontal } from 'lucide-react';
import { useRouter } from 'next/navigation';
import { toast } from 'sonner';
import { Button } from '../ui/button';
import {
@@ -16,15 +17,13 @@ import {
DropdownMenuSeparator,
DropdownMenuTrigger,
} from '../ui/dropdown-menu';
import { toast } from '../ui/use-toast';
export function ClientActions(client: IClientWithProject) {
const { id } = client;
const router = useRouter();
const deletion = api.client.remove.useMutation({
onSuccess() {
toast({
title: 'Success',
toast('Success', {
description: 'Client revoked, incoming requests will be rejected.',
});
router.refresh();

View File

@@ -6,6 +6,7 @@ import type { IProject } from '@/types';
import { clipboard } from '@/utils/clipboard';
import { MoreHorizontal } from 'lucide-react';
import { useRouter } from 'next/navigation';
import { toast } from 'sonner';
import { Button } from '../ui/button';
import {
@@ -16,15 +17,13 @@ import {
DropdownMenuSeparator,
DropdownMenuTrigger,
} from '../ui/dropdown-menu';
import { toast } from '../ui/use-toast';
export function ProjectActions(project: IProject) {
const { id } = project;
const router = useRouter();
const deletion = api.project.remove.useMutation({
onSuccess() {
toast({
title: 'Success',
toast('Success', {
description: 'Project deleted successfully.',
});
router.refresh();

View File

@@ -19,8 +19,7 @@ export function ReportSaveButton({ className }: ReportSaveButtonProps) {
const update = api.report.update.useMutation({
onSuccess() {
dispatch(resetDirty());
toast({
title: 'Success',
toast('Success', {
description: 'Report updated.',
});
},

View File

@@ -1,29 +1,29 @@
import { useTheme } from "next-themes"
import { Toaster as Sonner } from "sonner"
import { useTheme } from 'next-themes';
import { Toaster as Sonner } from 'sonner';
type ToasterProps = React.ComponentProps<typeof Sonner>
type ToasterProps = React.ComponentProps<typeof Sonner>;
const Toaster = ({ ...props }: ToasterProps) => {
const { theme = "system" } = useTheme()
const { theme = 'system' } = useTheme();
return (
<Sonner
theme={theme as ToasterProps["theme"]}
theme={theme as ToasterProps['theme']}
className="toaster group"
toastOptions={{
classNames: {
toast:
"group toast group-[.toaster]:bg-background group-[.toaster]:text-foreground group-[.toaster]:border-border group-[.toaster]:shadow-lg",
description: "group-[.toast]:text-muted-foreground",
'group toast group-[.toaster]:bg-background group-[.toaster]:text-foreground group-[.toaster]:border-border group-[.toaster]:shadow-lg',
description: 'group-[.toast]:text-muted-foreground',
actionButton:
"group-[.toast]:bg-primary group-[.toast]:text-primary-foreground",
'group-[.toast]:bg-primary group-[.toast]:text-primary-foreground',
cancelButton:
"group-[.toast]:bg-muted group-[.toast]:text-muted-foreground",
'group-[.toast]:bg-muted group-[.toast]:text-muted-foreground',
},
}}
{...props}
/>
)
}
);
};
export { Toaster }
export { Toaster };

View File

@@ -1,190 +0,0 @@
'use client';
// Inspired by react-hot-toast library
import * as React from 'react';
import type { ToastActionElement, ToastProps } from '@/components/ui/toast';
const TOAST_LIMIT = 1;
const TOAST_REMOVE_DELAY = 1000000;
type ToasterToast = ToastProps & {
id: string;
title?: React.ReactNode;
description?: React.ReactNode;
action?: ToastActionElement;
};
const actionTypes = {
ADD_TOAST: 'ADD_TOAST',
UPDATE_TOAST: 'UPDATE_TOAST',
DISMISS_TOAST: 'DISMISS_TOAST',
REMOVE_TOAST: 'REMOVE_TOAST',
} as const;
let count = 0;
function genId() {
count = (count + 1) % Number.MAX_VALUE;
return count.toString();
}
type ActionType = typeof actionTypes;
type Action =
| {
type: ActionType['ADD_TOAST'];
toast: ToasterToast;
}
| {
type: ActionType['UPDATE_TOAST'];
toast: Partial<ToasterToast>;
}
| {
type: ActionType['DISMISS_TOAST'];
toastId?: ToasterToast['id'];
}
| {
type: ActionType['REMOVE_TOAST'];
toastId?: ToasterToast['id'];
};
interface State {
toasts: ToasterToast[];
}
const toastTimeouts = new Map<string, ReturnType<typeof setTimeout>>();
const addToRemoveQueue = (toastId: string) => {
if (toastTimeouts.has(toastId)) {
return;
}
const timeout = setTimeout(() => {
toastTimeouts.delete(toastId);
dispatch({
type: 'REMOVE_TOAST',
toastId: toastId,
});
}, TOAST_REMOVE_DELAY);
toastTimeouts.set(toastId, timeout);
};
export const reducer = (state: State, action: Action): State => {
switch (action.type) {
case 'ADD_TOAST':
return {
...state,
toasts: [action.toast, ...state.toasts].slice(0, TOAST_LIMIT),
};
case 'UPDATE_TOAST':
return {
...state,
toasts: state.toasts.map((t) =>
t.id === action.toast.id ? { ...t, ...action.toast } : t
),
};
case 'DISMISS_TOAST': {
const { toastId } = action;
// ! Side effects ! - This could be extracted into a dismissToast() action,
// but I'll keep it here for simplicity
if (toastId) {
addToRemoveQueue(toastId);
} else {
state.toasts.forEach((toast) => {
addToRemoveQueue(toast.id);
});
}
return {
...state,
toasts: state.toasts.map((t) =>
t.id === toastId || toastId === undefined
? {
...t,
open: false,
}
: t
),
};
}
case 'REMOVE_TOAST':
if (action.toastId === undefined) {
return {
...state,
toasts: [],
};
}
return {
...state,
toasts: state.toasts.filter((t) => t.id !== action.toastId),
};
}
};
const listeners: ((state: State) => void)[] = [];
let memoryState: State = { toasts: [] };
function dispatch(action: Action) {
memoryState = reducer(memoryState, action);
listeners.forEach((listener) => {
listener(memoryState);
});
}
export type Toast = Omit<ToasterToast, 'id'>;
function toast({ ...props }: Toast) {
const id = genId();
const update = (props: ToasterToast) =>
dispatch({
type: 'UPDATE_TOAST',
toast: { ...props, id },
});
const dismiss = () => dispatch({ type: 'DISMISS_TOAST', toastId: id });
dispatch({
type: 'ADD_TOAST',
toast: {
...props,
id,
open: true,
onOpenChange: (open) => {
if (!open) dismiss();
},
},
});
return {
id: id,
dismiss,
update,
};
}
function useToast() {
const [state, setState] = React.useState<State>(memoryState);
React.useEffect(() => {
listeners.push(setState);
return () => {
const index = listeners.indexOf(setState);
if (index > -1) {
listeners.splice(index, 1);
}
};
}, [state]);
return {
...state,
toast,
dismiss: (toastId?: string) => dispatch({ type: 'DISMISS_TOAST', toastId }),
};
}
export { useToast, toast };

View File

@@ -40,8 +40,7 @@ export default function AddClient({ organizationId }: AddClientProps) {
const mutation = api.client.create.useMutation({
onError: handleError,
onSuccess() {
toast({
title: 'Success',
toast('Success', {
description: 'Client created!',
});
router.refresh();

View File

@@ -35,8 +35,7 @@ export default function AddDashboard() {
onError: handleError,
onSuccess() {
router.refresh();
toast({
title: 'Success',
toast('Success', {
description: 'Dashboard created.',
});
popModal();

View File

@@ -27,8 +27,7 @@ export default function AddProject({ organizationId }: AddProjectProps) {
onError: handleError,
onSuccess() {
router.refresh();
toast({
title: 'Success',
toast('Success', {
description: 'Project created! Lets create a client for it 🤘',
});
popModal();

View File

@@ -39,8 +39,7 @@ export default function EditClient({ id, name, cors }: EditClientProps) {
onError: handleError,
onSuccess() {
reset();
toast({
title: 'Success',
toast('Success', {
description: 'Client updated.',
});
popModal();

View File

@@ -37,8 +37,7 @@ export default function EditDashboard({ id, name }: EditDashboardProps) {
onError: handleError,
onSuccess() {
reset();
toast({
title: 'Success',
toast('Success', {
description: 'Dashboard updated.',
});
popModal();

View File

@@ -38,8 +38,7 @@ export default function EditProject({ id, name }: EditProjectProps) {
onSuccess() {
reset();
router.refresh();
toast({
title: 'Success',
toast('Success', {
description: 'Project updated.',
});
popModal();

View File

@@ -38,8 +38,7 @@ export default function SaveReport({ report }: SaveReportProps) {
const save = api.report.save.useMutation({
onError: handleError,
onSuccess(res) {
toast({
title: 'Success',
toast('Success', {
description: 'Report saved.',
});
popModal();
@@ -65,8 +64,7 @@ export default function SaveReport({ report }: SaveReportProps) {
onSuccess(res) {
setValue('dashboardId', res.id);
dashboardQuery.refetch();
toast({
title: 'Success',
toast('Success', {
description: 'Dashboard created.',
});
},

View File

@@ -2,8 +2,7 @@ import { toast } from 'sonner';
export function clipboard(value: string | number) {
navigator.clipboard.writeText(value.toString());
toast({
title: 'Copied to clipboard',
toast('Copied to clipboard', {
description: value.toString(),
});
}

View File

@@ -1,12 +1,9 @@
const parse = (connectionString: string) => {
const match = connectionString.match(/redis:\/\/(.+?):(.+?)@(.+?):(.+)/);
if (!match) {
throw new Error('Invalid connection string');
}
const url = new URL(connectionString);
return {
host: match[3]!,
port: Number(match[4]),
password: match[2]!,
host: url.hostname,
port: Number(url.port),
password: url.password,
} as const;
};