import { Fragment, useEffect, useState } from "react"; import { useRouter } from "next/router"; import { observer } from "mobx-react-lite"; import { Controller, useForm } from "react-hook-form"; import { Dialog, Transition } from "@headlessui/react"; import { Check, CircleDot, Globe2 } from "lucide-react"; // hooks import { useProjectPublish } from "hooks/store"; import useToast from "hooks/use-toast"; // ui import { Button, Loader, ToggleSwitch } from "@plane/ui"; import { CustomPopover } from "./popover"; // types import { IProject } from "@plane/types"; import { IProjectPublishSettings, TProjectPublishViews } from "store/project/project-publish.store"; type Props = { isOpen: boolean; project: IProject; onClose: () => void; }; type FormData = { id: string | null; comments: boolean; reactions: boolean; votes: boolean; inbox: string | null; views: TProjectPublishViews[]; }; const defaultValues: FormData = { id: null, comments: false, reactions: false, votes: false, inbox: null, views: ["list", "kanban"], }; const viewOptions: { key: TProjectPublishViews; label: string; }[] = [ { key: "list", label: "List" }, { key: "kanban", label: "Kanban" }, // { key: "calendar", label: "Calendar" }, // { key: "gantt", label: "Gantt" }, // { key: "spreadsheet", label: "Spreadsheet" }, ]; export const PublishProjectModal: React.FC<Props> = observer((props) => { const { isOpen, project, onClose } = props; // states const [isUnPublishing, setIsUnPublishing] = useState(false); const [isUpdateRequired, setIsUpdateRequired] = useState(false); let plane_deploy_url = process.env.NEXT_PUBLIC_DEPLOY_URL; if (typeof window !== "undefined" && !plane_deploy_url) plane_deploy_url = window.location.protocol + "//" + window.location.host + "/spaces"; // router const router = useRouter(); const { workspaceSlug } = router.query; // store hooks const { projectPublishSettings, getProjectSettingsAsync, publishProject, updateProjectSettingsAsync, unPublishProject, fetchSettingsLoader, } = useProjectPublish(); // toast alert const { setToastAlert } = useToast(); // form info const { control, formState: { isSubmitting }, getValues, handleSubmit, reset, watch, } = useForm({ defaultValues, }); const handleClose = () => { onClose(); setIsUpdateRequired(false); reset({ ...defaultValues }); }; // prefill form with the saved settings if the project is already published useEffect(() => { if (projectPublishSettings && projectPublishSettings !== "not-initialized") { let userBoards: TProjectPublishViews[] = []; if (projectPublishSettings?.views) { const savedViews = projectPublishSettings?.views; if (!savedViews) return; if (savedViews.list) userBoards.push("list"); if (savedViews.kanban) userBoards.push("kanban"); if (savedViews.calendar) userBoards.push("calendar"); if (savedViews.gantt) userBoards.push("gantt"); if (savedViews.spreadsheet) userBoards.push("spreadsheet"); userBoards = userBoards && userBoards.length > 0 ? userBoards : ["list"]; } const updatedData = { id: projectPublishSettings?.id || null, comments: projectPublishSettings?.comments || false, reactions: projectPublishSettings?.reactions || false, votes: projectPublishSettings?.votes || false, inbox: projectPublishSettings?.inbox || null, views: userBoards, }; reset({ ...updatedData }); } }, [reset, projectPublishSettings, isOpen]); // fetch publish settings useEffect(() => { if (!workspaceSlug || !isOpen) return; if (projectPublishSettings === "not-initialized") { getProjectSettingsAsync(workspaceSlug.toString(), project.id); } }, [isOpen, workspaceSlug, project, projectPublishSettings, getProjectSettingsAsync]); const handlePublishProject = async (payload: IProjectPublishSettings) => { if (!workspaceSlug) return; return publishProject(workspaceSlug.toString(), project.id, payload) .then((res) => { handleClose(); // window.open(`${plane_deploy_url}/${workspaceSlug}/${project.id}`, "_blank"); return res; }) .catch((err) => err); }; const handleUpdatePublishSettings = async (payload: IProjectPublishSettings) => { if (!workspaceSlug) return; await updateProjectSettingsAsync(workspaceSlug.toString(), project.id, payload.id ?? "", payload) .then((res) => { setToastAlert({ type: "success", title: "Success!", message: "Publish settings updated successfully!", }); handleClose(); return res; }) .catch((error) => { console.error("error", error); return error; }); }; const handleUnPublishProject = async (publishId: string) => { if (!workspaceSlug || !publishId) return; setIsUnPublishing(true); await unPublishProject(workspaceSlug.toString(), project.id, publishId) .then((res) => { handleClose(); return res; }) .catch(() => setToastAlert({ type: "error", title: "Error!", message: "Something went wrong while un-publishing the project.", }) ) .finally(() => setIsUnPublishing(false)); }; const CopyLinkToClipboard = ({ copy_link }: { copy_link: string }) => { const [status, setStatus] = useState(false); const copyText = () => { navigator.clipboard.writeText(copy_link); setStatus(true); setTimeout(() => { setStatus(false); }, 1000); }; return ( <div className="flex h-[30px] min-w-[30px] cursor-pointer items-center justify-center rounded border border-custom-border-100 bg-custom-background-100 px-2 text-xs hover:bg-custom-background-90" onClick={() => copyText()} > {status ? "Copied" : "Copy Link"} </div> ); }; const handleFormSubmit = async (formData: FormData) => { if (!formData.views || formData.views.length === 0) { setToastAlert({ type: "error", title: "Error!", message: "Please select at least one view layout to publish the project.", }); return; } const payload = { comments: formData.comments, reactions: formData.reactions, votes: formData.votes, inbox: formData.inbox, views: { list: formData.views.includes("list"), kanban: formData.views.includes("kanban"), calendar: formData.views.includes("calendar"), gantt: formData.views.includes("gantt"), spreadsheet: formData.views.includes("spreadsheet"), }, }; if (project.is_deployed) await handleUpdatePublishSettings({ id: watch("id") ?? "", ...payload }); else await handlePublishProject(payload); }; // check if an update is required or not const checkIfUpdateIsRequired = () => { if (!projectPublishSettings || projectPublishSettings === "not-initialized") return; const currentSettings = projectPublishSettings; const newSettings = getValues(); if ( currentSettings.comments !== newSettings.comments || currentSettings.reactions !== newSettings.reactions || currentSettings.votes !== newSettings.votes ) { setIsUpdateRequired(true); return; } let viewCheckFlag = 0; viewOptions.forEach((option) => { if (currentSettings.views[option.key] !== newSettings.views.includes(option.key)) viewCheckFlag++; }); if (viewCheckFlag !== 0) { setIsUpdateRequired(true); return; } setIsUpdateRequired(false); }; return ( <Transition.Root show={isOpen} as={Fragment}> <Dialog as="div" className="relative z-20" onClose={handleClose}> <Transition.Child as={Fragment} enter="ease-out duration-200" enterFrom="opacity-0" enterTo="opacity-100" leave="ease-in duration-100" leaveFrom="opacity-100" leaveTo="opacity-0" > <div className="fixed inset-0 bg-custom-backdrop transition-opacity" /> </Transition.Child> <div className="fixed inset-0 z-20 overflow-y-auto"> <div className="flex min-h-full items-center justify-center p-4 text-center sm:p-0"> <Transition.Child as={Fragment} enter="ease-out duration-200" enterFrom="opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95" enterTo="opacity-100 translate-y-0 sm:scale-100" leave="ease-in duration-100" leaveFrom="opacity-100 translate-y-0 sm:scale-100" leaveTo="opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95" > <Dialog.Panel className="w-full transform rounded-lg bg-custom-background-100 text-left shadow-custom-shadow-md transition-all sm:w-3/5 lg:w-1/2 xl:w-2/5"> <form onSubmit={handleSubmit(handleFormSubmit)} className="space-y-4"> {/* heading */} <div className="flex items-center justify-between gap-2 px-6 pt-4"> <h5 className="inline-block text-xl font-semibold">Publish</h5> {project.is_deployed && ( <Button variant="danger" onClick={() => handleUnPublishProject(watch("id") ?? "")} loading={isUnPublishing} > {isUnPublishing ? "Un-publishing..." : "Un-publish"} </Button> )} </div> {/* content */} {fetchSettingsLoader ? ( <Loader className="space-y-4 px-6"> <Loader.Item height="30px" /> <Loader.Item height="30px" /> <Loader.Item height="30px" /> <Loader.Item height="30px" /> </Loader> ) : ( <div className="px-6"> {project.is_deployed && ( <> <div className="relative flex items-center gap-2 rounded-md border border-custom-border-100 bg-custom-background-80 px-3 py-2"> <div className="flex-grow truncate text-sm"> {`${plane_deploy_url}/${workspaceSlug}/${project.id}`} </div> <div className="relative flex flex-shrink-0 items-center gap-1"> <CopyLinkToClipboard copy_link={`${plane_deploy_url}/${workspaceSlug}/${project.id}`} /> </div> </div> <div className="mt-3 flex items-center gap-1 text-custom-primary-100"> <div className="flex h-5 w-5 items-center overflow-hidden"> <CircleDot className="h-5 w-5" /> </div> <div className="text-sm">This project is live on web</div> </div> </> )} <div className="mt-6 space-y-4"> <div className="relative flex items-center justify-between gap-2"> <div className="text-sm">Views</div> <Controller control={control} name="views" render={({ field: { onChange, value } }) => ( <CustomPopover label={ value.length > 0 ? viewOptions .filter((v) => value.includes(v.key)) .map((v) => v.label) .join(", ") : `` } placeholder="Select views" > <> {viewOptions.map((option) => ( <div key={option.key} className={`relative m-1 flex cursor-pointer items-center justify-between gap-2 rounded-sm p-1 px-2 text-custom-text-200 ${ value.includes(option.key) ? "bg-custom-background-80 text-custom-text-100" : "hover:bg-custom-background-80 hover:text-custom-text-100" }`} onClick={() => { const _views = value.length > 0 ? value.includes(option.key) ? value.filter((_o: string) => _o !== option.key) : [...value, option.key] : [option.key]; if (_views.length === 0) return; onChange(_views); checkIfUpdateIsRequired(); }} > <div className="text-sm">{option.label}</div> <div className={`relative flex h-4 w-4 items-center justify-center`}> {value.length > 0 && value.includes(option.key) && ( <Check className="h-5 w-5" /> )} </div> </div> ))} </> </CustomPopover> )} /> </div> <div className="relative flex items-center justify-between gap-2"> <div className="text-sm">Allow comments</div> <Controller control={control} name="comments" render={({ field: { onChange, value } }) => ( <ToggleSwitch value={value} onChange={(val) => { onChange(val); checkIfUpdateIsRequired(); }} size="sm" /> )} /> </div> <div className="relative flex items-center justify-between gap-2"> <div className="text-sm">Allow reactions</div> <Controller control={control} name="reactions" render={({ field: { onChange, value } }) => ( <ToggleSwitch value={value} onChange={(val) => { onChange(val); checkIfUpdateIsRequired(); }} size="sm" /> )} /> </div> <div className="relative flex items-center justify-between gap-2"> <div className="text-sm">Allow voting</div> <Controller control={control} name="votes" render={({ field: { onChange, value } }) => ( <ToggleSwitch value={value} onChange={(val) => { onChange(val); checkIfUpdateIsRequired(); }} size="sm" /> )} /> </div> {/* toggle inbox */} {/* <div className="relative flex justify-between items-center gap-2"> <div className="text-sm">Allow issue proposals</div> <Controller control={control} name="inbox" render={({ field: { onChange, value } }) => ( <ToggleSwitch value={value} onChange={onChange} size="sm" /> )} /> </div> */} </div> </div> )} {/* modal handlers */} <div className="relative flex items-center justify-between border-t border-custom-border-200 px-6 py-5"> <div className="flex items-center gap-1 text-sm text-custom-text-400"> <Globe2 className="h-4 w-4" /> <div className="text-sm">Anyone with the link can access</div> </div> {!fetchSettingsLoader && ( <div className="relative flex items-center gap-2"> <Button variant="neutral-primary" size="sm" onClick={handleClose}> Cancel </Button> {project.is_deployed ? ( <> {isUpdateRequired && ( <Button variant="primary" size="sm" type="submit" loading={isSubmitting}> {isSubmitting ? "Updating..." : "Update settings"} </Button> )} </> ) : ( <Button variant="primary" size="sm" type="submit" loading={isSubmitting}> {isSubmitting ? "Publishing..." : "Publish"} </Button> )} </div> )} </div> </form> </Dialog.Panel> </Transition.Child> </div> </div> </Dialog> </Transition.Root> ); });