dashboard: fix toaster
This commit is contained in:
@@ -24,25 +24,20 @@ export function ListDashboards({ dashboards }: ListDashboardsProps) {
|
|||||||
const deletion = api.dashboard.delete.useMutation({
|
const deletion = api.dashboard.delete.useMutation({
|
||||||
onError: (error, variables) => {
|
onError: (error, variables) => {
|
||||||
return handleErrorToastOptions({
|
return handleErrorToastOptions({
|
||||||
action: (
|
action: {
|
||||||
<ToastAction
|
label: 'Force delete',
|
||||||
altText="Force delete"
|
onClick: () => {
|
||||||
onClick={() => {
|
deletion.mutate({
|
||||||
deletion.mutate({
|
forceDelete: true,
|
||||||
forceDelete: true,
|
id: variables.id,
|
||||||
id: variables.id,
|
});
|
||||||
});
|
},
|
||||||
}}
|
},
|
||||||
>
|
|
||||||
Force delete
|
|
||||||
</ToastAction>
|
|
||||||
),
|
|
||||||
})(error);
|
})(error);
|
||||||
},
|
},
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
router.refresh();
|
router.refresh();
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Dashboard deleted.',
|
description: 'Dashboard deleted.',
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -25,13 +25,12 @@ export default function EditOrganization({
|
|||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
|
|
||||||
const { register, handleSubmit, formState, reset } = useForm<IForm>({
|
const { register, handleSubmit, formState, reset } = useForm<IForm>({
|
||||||
defaultValues: organization,
|
defaultValues: organization ?? undefined,
|
||||||
});
|
});
|
||||||
|
|
||||||
const mutation = api.organization.update.useMutation({
|
const mutation = api.organization.update.useMutation({
|
||||||
onSuccess(res) {
|
onSuccess(res) {
|
||||||
toast({
|
toast('Organization updated', {
|
||||||
title: 'Organization updated',
|
|
||||||
description: 'Your organization has been updated.',
|
description: 'Your organization has been updated.',
|
||||||
});
|
});
|
||||||
reset(res);
|
reset(res);
|
||||||
@@ -57,7 +56,7 @@ export default function EditOrganization({
|
|||||||
<InputWithLabel
|
<InputWithLabel
|
||||||
label="Name"
|
label="Name"
|
||||||
{...register('name')}
|
{...register('name')}
|
||||||
defaultValue={organization.name}
|
defaultValue={organization?.name}
|
||||||
/>
|
/>
|
||||||
</WidgetBody>
|
</WidgetBody>
|
||||||
</Widget>
|
</Widget>
|
||||||
|
|||||||
@@ -27,8 +27,7 @@ export function InviteUser() {
|
|||||||
|
|
||||||
const mutation = api.organization.inviteUser.useMutation({
|
const mutation = api.organization.inviteUser.useMutation({
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
toast({
|
toast('User invited!', {
|
||||||
title: 'User invited!',
|
|
||||||
description: 'The user has been invited to the organization.',
|
description: 'The user has been invited to the organization.',
|
||||||
});
|
});
|
||||||
reset();
|
reset();
|
||||||
|
|||||||
@@ -35,8 +35,7 @@ export default function EditProfile({ profile }: EditProfileProps) {
|
|||||||
|
|
||||||
const mutation = api.user.update.useMutation({
|
const mutation = api.user.update.useMutation({
|
||||||
onSuccess(res) {
|
onSuccess(res) {
|
||||||
toast({
|
toast('Profile updated', {
|
||||||
title: 'Profile updated',
|
|
||||||
description: 'Your profile has been updated.',
|
description: 'Your profile has been updated.',
|
||||||
});
|
});
|
||||||
reset(res);
|
reset(res);
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
import type { Toast } from '@/components/ui/use-toast';
|
|
||||||
import type { AppRouter } from '@/server/api/root';
|
import type { AppRouter } from '@/server/api/root';
|
||||||
import type { TRPCClientErrorBase } from '@trpc/react-query';
|
import type { TRPCClientErrorBase } from '@trpc/react-query';
|
||||||
import { createTRPCReact } from '@trpc/react-query';
|
import { createTRPCReact } from '@trpc/react-query';
|
||||||
import type { inferRouterInputs, inferRouterOutputs } from '@trpc/server';
|
import type { inferRouterInputs, inferRouterOutputs } from '@trpc/server';
|
||||||
|
import type { ExternalToast } from 'sonner';
|
||||||
import { toast } from 'sonner';
|
import { toast } from 'sonner';
|
||||||
|
|
||||||
export const api = createTRPCReact<AppRouter>({});
|
export const api = createTRPCReact<AppRouter>({});
|
||||||
@@ -24,16 +24,14 @@ export type IChartData = RouterOutputs['chart']['chart'];
|
|||||||
export type IChartSerieDataItem = IChartData['series'][number]['data'][number];
|
export type IChartSerieDataItem = IChartData['series'][number]['data'][number];
|
||||||
|
|
||||||
export function handleError(error: TRPCClientErrorBase<any>) {
|
export function handleError(error: TRPCClientErrorBase<any>) {
|
||||||
toast({
|
toast('Error', {
|
||||||
title: 'Error',
|
|
||||||
description: error.message,
|
description: error.message,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
export function handleErrorToastOptions(options: Toast) {
|
export function handleErrorToastOptions(options: ExternalToast) {
|
||||||
return function (error: TRPCClientErrorBase<any>) {
|
return function (error: TRPCClientErrorBase<any>) {
|
||||||
toast({
|
toast('Error', {
|
||||||
title: 'Error',
|
|
||||||
description: error.message,
|
description: error.message,
|
||||||
...options,
|
...options,
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
import React, { useRef, useState } from 'react';
|
import React, { useRef, useState } from 'react';
|
||||||
import { api } from '@/app/_trpc/client';
|
import { api } from '@/app/_trpc/client';
|
||||||
import { Toaster } from '@/components/ui/toaster';
|
|
||||||
import { TooltipProvider } from '@/components/ui/tooltip';
|
import { TooltipProvider } from '@/components/ui/tooltip';
|
||||||
import { ModalProvider } from '@/modals';
|
import { ModalProvider } from '@/modals';
|
||||||
import type { AppStore } from '@/redux';
|
import type { AppStore } from '@/redux';
|
||||||
@@ -50,7 +49,6 @@ export default function Providers({ children }: { children: React.ReactNode }) {
|
|||||||
<QueryClientProvider client={queryClient}>
|
<QueryClientProvider client={queryClient}>
|
||||||
<TooltipProvider delayDuration={200}>
|
<TooltipProvider delayDuration={200}>
|
||||||
{children}
|
{children}
|
||||||
<Toaster />
|
|
||||||
<ModalProvider />
|
<ModalProvider />
|
||||||
</TooltipProvider>
|
</TooltipProvider>
|
||||||
</QueryClientProvider>
|
</QueryClientProvider>
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import type { IClientWithProject } from '@/types';
|
|||||||
import { clipboard } from '@/utils/clipboard';
|
import { clipboard } from '@/utils/clipboard';
|
||||||
import { MoreHorizontal } from 'lucide-react';
|
import { MoreHorizontal } from 'lucide-react';
|
||||||
import { useRouter } from 'next/navigation';
|
import { useRouter } from 'next/navigation';
|
||||||
|
import { toast } from 'sonner';
|
||||||
|
|
||||||
import { Button } from '../ui/button';
|
import { Button } from '../ui/button';
|
||||||
import {
|
import {
|
||||||
@@ -16,15 +17,13 @@ import {
|
|||||||
DropdownMenuSeparator,
|
DropdownMenuSeparator,
|
||||||
DropdownMenuTrigger,
|
DropdownMenuTrigger,
|
||||||
} from '../ui/dropdown-menu';
|
} from '../ui/dropdown-menu';
|
||||||
import { toast } from '../ui/use-toast';
|
|
||||||
|
|
||||||
export function ClientActions(client: IClientWithProject) {
|
export function ClientActions(client: IClientWithProject) {
|
||||||
const { id } = client;
|
const { id } = client;
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const deletion = api.client.remove.useMutation({
|
const deletion = api.client.remove.useMutation({
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Client revoked, incoming requests will be rejected.',
|
description: 'Client revoked, incoming requests will be rejected.',
|
||||||
});
|
});
|
||||||
router.refresh();
|
router.refresh();
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import type { IProject } from '@/types';
|
|||||||
import { clipboard } from '@/utils/clipboard';
|
import { clipboard } from '@/utils/clipboard';
|
||||||
import { MoreHorizontal } from 'lucide-react';
|
import { MoreHorizontal } from 'lucide-react';
|
||||||
import { useRouter } from 'next/navigation';
|
import { useRouter } from 'next/navigation';
|
||||||
|
import { toast } from 'sonner';
|
||||||
|
|
||||||
import { Button } from '../ui/button';
|
import { Button } from '../ui/button';
|
||||||
import {
|
import {
|
||||||
@@ -16,15 +17,13 @@ import {
|
|||||||
DropdownMenuSeparator,
|
DropdownMenuSeparator,
|
||||||
DropdownMenuTrigger,
|
DropdownMenuTrigger,
|
||||||
} from '../ui/dropdown-menu';
|
} from '../ui/dropdown-menu';
|
||||||
import { toast } from '../ui/use-toast';
|
|
||||||
|
|
||||||
export function ProjectActions(project: IProject) {
|
export function ProjectActions(project: IProject) {
|
||||||
const { id } = project;
|
const { id } = project;
|
||||||
const router = useRouter();
|
const router = useRouter();
|
||||||
const deletion = api.project.remove.useMutation({
|
const deletion = api.project.remove.useMutation({
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Project deleted successfully.',
|
description: 'Project deleted successfully.',
|
||||||
});
|
});
|
||||||
router.refresh();
|
router.refresh();
|
||||||
|
|||||||
@@ -19,8 +19,7 @@ export function ReportSaveButton({ className }: ReportSaveButtonProps) {
|
|||||||
const update = api.report.update.useMutation({
|
const update = api.report.update.useMutation({
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
dispatch(resetDirty());
|
dispatch(resetDirty());
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Report updated.',
|
description: 'Report updated.',
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -1,29 +1,29 @@
|
|||||||
import { useTheme } from "next-themes"
|
import { useTheme } from 'next-themes';
|
||||||
import { Toaster as Sonner } from "sonner"
|
import { Toaster as Sonner } from 'sonner';
|
||||||
|
|
||||||
type ToasterProps = React.ComponentProps<typeof Sonner>
|
type ToasterProps = React.ComponentProps<typeof Sonner>;
|
||||||
|
|
||||||
const Toaster = ({ ...props }: ToasterProps) => {
|
const Toaster = ({ ...props }: ToasterProps) => {
|
||||||
const { theme = "system" } = useTheme()
|
const { theme = 'system' } = useTheme();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Sonner
|
<Sonner
|
||||||
theme={theme as ToasterProps["theme"]}
|
theme={theme as ToasterProps['theme']}
|
||||||
className="toaster group"
|
className="toaster group"
|
||||||
toastOptions={{
|
toastOptions={{
|
||||||
classNames: {
|
classNames: {
|
||||||
toast:
|
toast:
|
||||||
"group toast group-[.toaster]:bg-background group-[.toaster]:text-foreground group-[.toaster]:border-border group-[.toaster]:shadow-lg",
|
'group toast group-[.toaster]:bg-background group-[.toaster]:text-foreground group-[.toaster]:border-border group-[.toaster]:shadow-lg',
|
||||||
description: "group-[.toast]:text-muted-foreground",
|
description: 'group-[.toast]:text-muted-foreground',
|
||||||
actionButton:
|
actionButton:
|
||||||
"group-[.toast]:bg-primary group-[.toast]:text-primary-foreground",
|
'group-[.toast]:bg-primary group-[.toast]:text-primary-foreground',
|
||||||
cancelButton:
|
cancelButton:
|
||||||
"group-[.toast]:bg-muted group-[.toast]:text-muted-foreground",
|
'group-[.toast]:bg-muted group-[.toast]:text-muted-foreground',
|
||||||
},
|
},
|
||||||
}}
|
}}
|
||||||
{...props}
|
{...props}
|
||||||
/>
|
/>
|
||||||
)
|
);
|
||||||
}
|
};
|
||||||
|
|
||||||
export { Toaster }
|
export { Toaster };
|
||||||
|
|||||||
@@ -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 };
|
|
||||||
@@ -40,8 +40,7 @@ export default function AddClient({ organizationId }: AddClientProps) {
|
|||||||
const mutation = api.client.create.useMutation({
|
const mutation = api.client.create.useMutation({
|
||||||
onError: handleError,
|
onError: handleError,
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Client created!',
|
description: 'Client created!',
|
||||||
});
|
});
|
||||||
router.refresh();
|
router.refresh();
|
||||||
|
|||||||
@@ -35,8 +35,7 @@ export default function AddDashboard() {
|
|||||||
onError: handleError,
|
onError: handleError,
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
router.refresh();
|
router.refresh();
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Dashboard created.',
|
description: 'Dashboard created.',
|
||||||
});
|
});
|
||||||
popModal();
|
popModal();
|
||||||
|
|||||||
@@ -27,8 +27,7 @@ export default function AddProject({ organizationId }: AddProjectProps) {
|
|||||||
onError: handleError,
|
onError: handleError,
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
router.refresh();
|
router.refresh();
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Project created! Lets create a client for it 🤘',
|
description: 'Project created! Lets create a client for it 🤘',
|
||||||
});
|
});
|
||||||
popModal();
|
popModal();
|
||||||
|
|||||||
@@ -39,8 +39,7 @@ export default function EditClient({ id, name, cors }: EditClientProps) {
|
|||||||
onError: handleError,
|
onError: handleError,
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
reset();
|
reset();
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Client updated.',
|
description: 'Client updated.',
|
||||||
});
|
});
|
||||||
popModal();
|
popModal();
|
||||||
|
|||||||
@@ -37,8 +37,7 @@ export default function EditDashboard({ id, name }: EditDashboardProps) {
|
|||||||
onError: handleError,
|
onError: handleError,
|
||||||
onSuccess() {
|
onSuccess() {
|
||||||
reset();
|
reset();
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Dashboard updated.',
|
description: 'Dashboard updated.',
|
||||||
});
|
});
|
||||||
popModal();
|
popModal();
|
||||||
|
|||||||
@@ -38,8 +38,7 @@ export default function EditProject({ id, name }: EditProjectProps) {
|
|||||||
onSuccess() {
|
onSuccess() {
|
||||||
reset();
|
reset();
|
||||||
router.refresh();
|
router.refresh();
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Project updated.',
|
description: 'Project updated.',
|
||||||
});
|
});
|
||||||
popModal();
|
popModal();
|
||||||
|
|||||||
@@ -38,8 +38,7 @@ export default function SaveReport({ report }: SaveReportProps) {
|
|||||||
const save = api.report.save.useMutation({
|
const save = api.report.save.useMutation({
|
||||||
onError: handleError,
|
onError: handleError,
|
||||||
onSuccess(res) {
|
onSuccess(res) {
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Report saved.',
|
description: 'Report saved.',
|
||||||
});
|
});
|
||||||
popModal();
|
popModal();
|
||||||
@@ -65,8 +64,7 @@ export default function SaveReport({ report }: SaveReportProps) {
|
|||||||
onSuccess(res) {
|
onSuccess(res) {
|
||||||
setValue('dashboardId', res.id);
|
setValue('dashboardId', res.id);
|
||||||
dashboardQuery.refetch();
|
dashboardQuery.refetch();
|
||||||
toast({
|
toast('Success', {
|
||||||
title: 'Success',
|
|
||||||
description: 'Dashboard created.',
|
description: 'Dashboard created.',
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
|||||||
@@ -2,8 +2,7 @@ import { toast } from 'sonner';
|
|||||||
|
|
||||||
export function clipboard(value: string | number) {
|
export function clipboard(value: string | number) {
|
||||||
navigator.clipboard.writeText(value.toString());
|
navigator.clipboard.writeText(value.toString());
|
||||||
toast({
|
toast('Copied to clipboard', {
|
||||||
title: 'Copied to clipboard',
|
|
||||||
description: value.toString(),
|
description: value.toString(),
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,12 +1,9 @@
|
|||||||
const parse = (connectionString: string) => {
|
const parse = (connectionString: string) => {
|
||||||
const match = connectionString.match(/redis:\/\/(.+?):(.+?)@(.+?):(.+)/);
|
const url = new URL(connectionString);
|
||||||
if (!match) {
|
|
||||||
throw new Error('Invalid connection string');
|
|
||||||
}
|
|
||||||
return {
|
return {
|
||||||
host: match[3]!,
|
host: url.hostname,
|
||||||
port: Number(match[4]),
|
port: Number(url.port),
|
||||||
password: match[2]!,
|
password: url.password,
|
||||||
} as const;
|
} as const;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user