mirror of
https://github.com/makeplane/plane
synced 2024-06-14 14:31:34 +00:00
120 lines
3.7 KiB
TypeScript
120 lines
3.7 KiB
TypeScript
import React, { FC, useEffect, useState } from "react";
|
|
import { observer } from "mobx-react-lite";
|
|
import { Controller, useForm } from "react-hook-form";
|
|
import { IWebhook, TWebhookEventTypes } from "@plane/types";
|
|
// hooks
|
|
import { Button } from "@plane/ui";
|
|
import {
|
|
WebhookIndividualEventOptions,
|
|
WebhookInput,
|
|
WebhookOptions,
|
|
WebhookSecretKey,
|
|
WebhookToggle,
|
|
} from "@/components/web-hooks";
|
|
// constants
|
|
import { WEBHOOK_UPDATED } from "@/constants/event-tracker";
|
|
import { useWebhook, useEventTracker } from "@/hooks/store";
|
|
// components
|
|
// ui
|
|
// types
|
|
|
|
type Props = {
|
|
data?: Partial<IWebhook>;
|
|
onSubmit: (data: IWebhook, webhookEventType: TWebhookEventTypes) => Promise<void>;
|
|
handleClose?: () => void;
|
|
};
|
|
|
|
const initialWebhookPayload: Partial<IWebhook> = {
|
|
cycle: true,
|
|
issue: true,
|
|
issue_comment: true,
|
|
module: true,
|
|
project: true,
|
|
url: "",
|
|
};
|
|
|
|
export const WebhookForm: FC<Props> = observer((props) => {
|
|
const { data, onSubmit, handleClose } = props;
|
|
// states
|
|
const [webhookEventType, setWebhookEventType] = useState<TWebhookEventTypes>("all");
|
|
// store hooks
|
|
const { webhookSecretKey } = useWebhook();
|
|
const { captureEvent } = useEventTracker();
|
|
// use form
|
|
const {
|
|
handleSubmit,
|
|
control,
|
|
formState: { isSubmitting, errors, dirtyFields },
|
|
} = useForm<IWebhook>({
|
|
defaultValues: { ...initialWebhookPayload, ...data },
|
|
});
|
|
|
|
const handleFormSubmit = async (formData: IWebhook) => {
|
|
await onSubmit(formData, webhookEventType).then(() => {
|
|
if (!data) return;
|
|
captureEvent(WEBHOOK_UPDATED, {
|
|
webhook_id: data.id,
|
|
change_details: Object.keys(dirtyFields),
|
|
enabled: formData.is_active,
|
|
});
|
|
});
|
|
};
|
|
|
|
useEffect(() => {
|
|
if (!data) return;
|
|
|
|
if (data.project && data.cycle && data.module && data.issue && data.issue_comment) setWebhookEventType("all");
|
|
else setWebhookEventType("individual");
|
|
}, [data]);
|
|
|
|
return (
|
|
<div className="space-y-6">
|
|
<div className="text-xl font-medium">{data ? "Webhook details" : "Create webhook"}</div>
|
|
<form onSubmit={handleSubmit(handleFormSubmit)}>
|
|
<div className="space-y-8">
|
|
<div>
|
|
<Controller
|
|
control={control}
|
|
name="url"
|
|
rules={{
|
|
required: "URL is required",
|
|
}}
|
|
render={({ field: { onChange, value } }) => (
|
|
<WebhookInput value={value} onChange={onChange} hasError={Boolean(errors.url)} />
|
|
)}
|
|
/>
|
|
{errors.url && <div className="text-xs text-red-500">{errors.url.message}</div>}
|
|
</div>
|
|
{data && <WebhookToggle control={control} />}
|
|
<div className="space-y-3">
|
|
<WebhookOptions value={webhookEventType} onChange={(val) => setWebhookEventType(val)} />
|
|
</div>
|
|
</div>
|
|
<div className="mt-4">
|
|
{webhookEventType === "individual" && <WebhookIndividualEventOptions control={control} />}
|
|
</div>
|
|
{data ? (
|
|
<div className="mt-8 space-y-8">
|
|
<WebhookSecretKey data={data} />
|
|
|
|
<Button type="submit" loading={isSubmitting}>
|
|
{isSubmitting ? "Updating..." : "Update"}
|
|
</Button>
|
|
</div>
|
|
) : (
|
|
<div className="mt-4 flex justify-end gap-2">
|
|
<Button variant="neutral-primary" onClick={handleClose}>
|
|
Discard
|
|
</Button>
|
|
{!webhookSecretKey && (
|
|
<Button type="submit" variant="primary" loading={isSubmitting}>
|
|
{isSubmitting ? "Creating..." : "Create"}
|
|
</Button>
|
|
)}
|
|
</div>
|
|
)}
|
|
</form>
|
|
</div>
|
|
);
|
|
});
|