"use client"; import { Checkbox } from "@/components/ui/checkbox"; import { FormControl, FormDescription, FormField, FormItem, FormLabel, FormMessage, } from "@/components/ui/form"; import { Link } from "@/components/common/link"; import { FormCardContent, FormCardSeparator, } from "@/components/forms/form-card"; import { useFormSheetDirty } from "@/components/forms/form-sheet"; import { Button } from "@/components/ui/button"; import { Form } from "@/components/ui/form"; import { Input } from "@/components/ui/input"; import { Label } from "@/components/ui/label"; import { config } from "@/data/notifications.client"; import { cn } from "@/lib/utils"; import { zodResolver } from "@hookform/resolvers/zod"; import { isTRPCClientError } from "@trpc/client"; import React, { useTransition } from "react"; import { useForm } from "react-hook-form"; import { toast } from "sonner"; import { z } from "zod"; const schema = z.object({ name: z.string(), provider: z.literal("webhook"), data: z.record(z.string(), z.string()), monitors: z.array(z.number()), }); type FormValues = z.infer; export function FormWebhook({ defaultValues, onSubmit, className, monitors, ...props }: Omit, "onSubmit"> & { defaultValues?: FormValues; onSubmit: (values: FormValues) => Promise; monitors: { id: number; name: string }[]; }) { const form = useForm({ resolver: zodResolver(schema), defaultValues: defaultValues ?? { name: "", provider: "webhook", data: { endpoint: "", // headers: [] }, monitors: [], }, }); const [isPending, startTransition] = useTransition(); const { setIsDirty } = useFormSheetDirty(); const formIsDirty = form.formState.isDirty; React.useEffect(() => { setIsDirty(formIsDirty); }, [formIsDirty, setIsDirty]); function submitAction(values: FormValues) { if (isPending) return; startTransition(async () => { try { const promise = onSubmit(values); toast.promise(promise, { loading: "Saving...", success: "Saved", error: (error) => { if (isTRPCClientError(error)) { return error.message; } return "Failed to save"; }, }); await promise; } catch (error) { console.error(error); } }); } function testAction() { if (isPending) return; startTransition(async () => { try { const provider = form.getValues("provider"); const data = form.getValues("data.endpoint"); toast.promise(config[provider].sendTest({ url: data }), { loading: "Sending test...", success: "Test sent", error: "Failed to send test", }); } catch (error) { console.error(error); } }); } return (
( Name Enter a descriptive name for your notifier. )} /> ( Webhook URL Send notifications to a custom webhook URL.{" "} Read more . )} />
( Monitors Select the monitors you want to notify.
{ field.onChange( checked ? monitors.map((m) => m.id) : [], ); }} />
{monitors.map((item) => (
{ const newValue = checked ? [...(field.value || []), item.id] : field.value?.filter((id) => id !== item.id); field.onChange(newValue); }} />
))}
)} />
); }