dev: revamp publish project modal (#2022)

* dev: revamp publish project modal

* chore: sidebar dropdown text
This commit is contained in:
Aaryan Khandelwal 2023-08-30 17:27:49 +05:30 committed by GitHub
parent 6c6b81bea7
commit 54527cc2bb
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 377 additions and 346 deletions

View File

@ -1,28 +1,38 @@
import React, { useEffect } from "react"; import React, { useEffect, useState } from "react";
// next imports // next imports
import { useRouter } from "next/router"; import { useRouter } from "next/router";
// react-hook-form // react-hook-form
import { useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
// headless ui // headless ui
import { Dialog, Transition } from "@headlessui/react"; import { Dialog, Transition } from "@headlessui/react";
// ui components // ui components
import { ToggleSwitch, PrimaryButton, SecondaryButton } from "components/ui"; import { ToggleSwitch, PrimaryButton, SecondaryButton, Icon, DangerButton } from "components/ui";
import { CustomPopover } from "./popover"; import { CustomPopover } from "./popover";
// mobx react lite // mobx react lite
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";
import { IProjectPublishSettingsViews } from "store/project-publish"; import { IProjectPublishSettings, TProjectPublishViews } from "store/project-publish";
// hooks // hooks
import useToast from "hooks/use-toast"; import useToast from "hooks/use-toast";
import useProjectDetails from "hooks/use-project-details"; import useProjectDetails from "hooks/use-project-details";
import useUser from "hooks/use-user";
type Props = { type Props = {
// user: ICurrentUserResponse | undefined; // user: ICurrentUserResponse | undefined;
}; };
const defaultValues: Partial<any> = { type FormData = {
id: string | null;
comments: boolean;
reactions: boolean;
votes: boolean;
inbox: string | null;
views: TProjectPublishViews[];
};
const defaultValues: FormData = {
id: null, id: null,
comments: false, comments: false,
reactions: false, reactions: false,
@ -31,71 +41,74 @@ const defaultValues: Partial<any> = {
views: ["list", "kanban"], views: ["list", "kanban"],
}; };
const viewOptions = [ const viewOptions: {
{ key: "list", value: "List" }, key: TProjectPublishViews;
{ key: "kanban", value: "Kanban" }, label: string;
// { key: "calendar", value: "Calendar" }, }[] = [
// { key: "gantt", value: "Gantt" }, { key: "list", label: "List" },
// { key: "spreadsheet", value: "Spreadsheet" }, { key: "kanban", label: "Kanban" },
// { key: "calendar", label: "Calendar" },
// { key: "gantt", label: "Gantt" },
// { key: "spreadsheet", label: "Spreadsheet" },
]; ];
export const PublishProjectModal: React.FC<Props> = observer(() => { export const PublishProjectModal: React.FC<Props> = observer(() => {
const store: RootStore = useMobxStore(); const [isUnpublishing, setIsUnpublishing] = useState(false);
const { projectPublish } = store; const [isUpdateRequired, setIsUpdateRequired] = useState(false);
const { projectDetails, mutateProjectDetails } = useProjectDetails(); const plane_deploy_url = process.env.NEXT_PUBLIC_DEPLOY_URL ?? "http://localhost:4000";
const { setToastAlert } = useToast();
const handleToastAlert = (title: string, type: string, message: string) => {
setToastAlert({
title: title || "Title",
type: "error" || "warning",
message: message || "Message",
});
};
const { NEXT_PUBLIC_DEPLOY_URL } = process.env;
const plane_deploy_url = NEXT_PUBLIC_DEPLOY_URL
? NEXT_PUBLIC_DEPLOY_URL
: "http://localhost:3001";
const router = useRouter(); const router = useRouter();
const { workspaceSlug } = router.query; const { workspaceSlug } = router.query;
const store: RootStore = useMobxStore();
const { projectPublish } = store;
const { user } = useUser();
const { mutateProjectDetails } = useProjectDetails();
const { setToastAlert } = useToast();
const { const {
formState: { errors, isSubmitting }, control,
formState: { isSubmitting },
getValues,
handleSubmit, handleSubmit,
reset, reset,
watch, watch,
setValue, } = useForm<FormData>({
} = useForm<any>({
defaultValues, defaultValues,
reValidateMode: "onChange",
}); });
const handleClose = () => { const handleClose = () => {
projectPublish.handleProjectModal(null); projectPublish.handleProjectModal(null);
setIsUpdateRequired(false);
reset({ ...defaultValues }); reset({ ...defaultValues });
}; };
// prefill form with the saved settings if the project is already published
useEffect(() => { useEffect(() => {
if ( if (
projectPublish.projectPublishSettings && projectPublish.projectPublishSettings &&
projectPublish.projectPublishSettings != "not-initialized" projectPublish.projectPublishSettings !== "not-initialized"
) { ) {
let userBoards: string[] = []; let userBoards: TProjectPublishViews[] = [];
if (projectPublish.projectPublishSettings?.views) { if (projectPublish.projectPublishSettings?.views) {
const _views: IProjectPublishSettingsViews | null = const savedViews = projectPublish.projectPublishSettings?.views;
projectPublish.projectPublishSettings?.views || null;
if (_views != null) { if (!savedViews) return;
if (_views.list) userBoards.push("list");
if (_views.kanban) userBoards.push("kanban"); if (savedViews.list) userBoards.push("list");
if (_views.calendar) userBoards.push("calendar"); if (savedViews.kanban) userBoards.push("kanban");
if (_views.gantt) userBoards.push("gantt"); if (savedViews.calendar) userBoards.push("calendar");
if (_views.spreadsheet) userBoards.push("spreadsheet"); if (savedViews.gantt) userBoards.push("gantt");
if (savedViews.spreadsheet) userBoards.push("spreadsheet");
userBoards = userBoards && userBoards.length > 0 ? userBoards : ["list"]; userBoards = userBoards && userBoards.length > 0 ? userBoards : ["list"];
} }
}
const updatedData = { const updatedData = {
id: projectPublish.projectPublishSettings?.id || null, id: projectPublish.projectPublishSettings?.id || null,
@ -105,98 +118,74 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
inbox: projectPublish.projectPublishSettings?.inbox || null, inbox: projectPublish.projectPublishSettings?.inbox || null,
views: userBoards, views: userBoards,
}; };
reset({ ...updatedData }); reset({ ...updatedData });
} }
}, [reset, projectPublish.projectPublishSettings]); }, [reset, projectPublish.projectPublishSettings]);
// fetch publish settings
useEffect(() => { useEffect(() => {
if (!workspaceSlug) return;
if ( if (
projectPublish.projectPublishModal && projectPublish.projectPublishModal &&
workspaceSlug && projectPublish.project_id !== null &&
projectPublish.project_id != null &&
projectPublish?.projectPublishSettings === "not-initialized" projectPublish?.projectPublishSettings === "not-initialized"
) { ) {
projectPublish.getProjectSettingsAsync( projectPublish.getProjectSettingsAsync(
workspaceSlug as string, workspaceSlug.toString(),
projectPublish.project_id as string, projectPublish.project_id,
null null
); );
} }
}, [workspaceSlug, projectPublish, projectPublish.projectPublishModal]); }, [workspaceSlug, projectPublish, projectPublish.projectPublishModal]);
const onSettingsPublish = async (formData: any) => { const handlePublishProject = async (payload: IProjectPublishSettings) => {
if (formData.views && formData.views.length > 0) { if (!workspaceSlug || !user) return;
const payload = {
comments: formData.comments || false,
reactions: formData.reactions || false,
votes: formData.votes || false,
inbox: formData.inbox || null,
views: {
list: formData.views.includes("list") || false,
kanban: formData.views.includes("kanban") || false,
calendar: formData.views.includes("calendar") || false,
gantt: formData.views.includes("gantt") || false,
spreadsheet: formData.views.includes("spreadsheet") || false,
},
};
const _workspaceSlug = workspaceSlug; const projectId = projectPublish.project_id;
const _projectId = projectPublish.project_id;
return projectPublish return projectPublish
.createProjectSettingsAsync(_workspaceSlug as string, _projectId as string, payload, null) .createProjectSettingsAsync(
workspaceSlug.toString(),
projectId?.toString() ?? "",
payload,
user
)
.then((response) => { .then((response) => {
mutateProjectDetails(); mutateProjectDetails();
handleClose(); handleClose();
console.log("_projectId", _projectId); if (projectId) window.open(`${plane_deploy_url}/${workspaceSlug}/${projectId}`, "_blank");
if (_projectId)
window.open(`${plane_deploy_url}/${_workspaceSlug}/${_projectId}`, "_blank");
return response; return response;
}) })
.catch((error) => { .catch((error) => {
console.error("error", error); console.error("error", error);
return error; return error;
}); });
} else {
handleToastAlert("Missing fields", "warning", "Please select at least one view to publish");
}
}; };
const onSettingsUpdate = async (key: string, value: any) => { const handleUpdatePublishSettings = async (payload: IProjectPublishSettings) => {
const payload = { if (!workspaceSlug || !user) return;
comments: key === "comments" ? value : watch("comments"),
reactions: key === "reactions" ? value : watch("reactions"),
votes: key === "votes" ? value : watch("votes"),
inbox: key === "inbox" ? value : watch("inbox"),
views:
key === "views"
? {
list: value.includes("list") ? true : false,
kanban: value.includes("kanban") ? true : false,
calendar: value.includes("calendar") ? true : false,
gantt: value.includes("gantt") ? true : false,
spreadsheet: value.includes("spreadsheet") ? true : false,
}
: {
list: watch("views").includes("list") ? true : false,
kanban: watch("views").includes("kanban") ? true : false,
calendar: watch("views").includes("calendar") ? true : false,
gantt: watch("views").includes("gantt") ? true : false,
spreadsheet: watch("views").includes("spreadsheet") ? true : false,
},
};
return projectPublish await projectPublish
.updateProjectSettingsAsync( .updateProjectSettingsAsync(
workspaceSlug as string, workspaceSlug.toString(),
projectPublish.project_id as string, projectPublish.project_id?.toString() ?? "",
watch("id"), payload.id ?? "",
payload, payload,
null user
) )
.then((response) => { .then((res) => {
mutateProjectDetails(); mutateProjectDetails();
return response;
setToastAlert({
type: "success",
title: "Success!",
message: "Publish settings updated successfully!",
});
handleClose();
return res;
}) })
.catch((error) => { .catch((error) => {
console.log("error", error); console.log("error", error);
@ -204,27 +193,30 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
}); });
}; };
const onSettingsUnPublish = async (formData: any) => const handleUnpublishProject = async (publishId: string) => {
if (!workspaceSlug || !publishId) return;
setIsUnpublishing(true);
projectPublish projectPublish
.deleteProjectSettingsAsync( .deleteProjectSettingsAsync(
workspaceSlug as string, workspaceSlug.toString(),
projectPublish.project_id as string, projectPublish.project_id as string,
formData?.id, publishId,
null null
) )
.then((response) => { .then((res) => {
mutateProjectDetails(); mutateProjectDetails();
reset({ ...defaultValues });
handleClose(); handleClose();
return response; return res;
}) })
.catch((error) => { .catch((err) => err)
console.error("error", error); .finally(() => setIsUnpublishing(false));
return error; };
});
const CopyLinkToClipboard = ({ copy_link }: { copy_link: string }) => { const CopyLinkToClipboard = ({ copy_link }: { copy_link: string }) => {
const [status, setStatus] = React.useState(false); const [status, setStatus] = useState(false);
const copyText = () => { const copyText = () => {
navigator.clipboard.writeText(copy_link); navigator.clipboard.writeText(copy_link);
@ -244,6 +236,68 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
); );
}; };
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 (watch("id")) await handleUpdatePublishSettings({ id: watch("id") ?? "", ...payload });
else await handlePublishProject(payload);
};
// check if an update is required or not
const checkIfUpdateIsRequired = () => {
if (
!projectPublish.projectPublishSettings ||
projectPublish.projectPublishSettings === "not-initialized"
)
return;
const currentSettings = projectPublish.projectPublishSettings as IProjectPublishSettings;
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 ( return (
<Transition.Root show={projectPublish.projectPublishModal} as={React.Fragment}> <Transition.Root show={projectPublish.projectPublishModal} as={React.Fragment}>
<Dialog as="div" className="relative z-20" onClose={handleClose}> <Dialog as="div" className="relative z-20" onClose={handleClose}>
@ -270,200 +324,190 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
leaveFrom="opacity-100 translate-y-0 sm:scale-100" leaveFrom="opacity-100 translate-y-0 sm:scale-100"
leaveTo="opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95" leaveTo="opacity-0 translate-y-4 sm:translate-y-0 sm:scale-95"
> >
<Dialog.Panel className="transform rounded-lg bg-custom-background-100 border border-custom-border-100 text-left shadow-xl transition-all w-full sm:w-3/5 lg:w-1/2 xl:w-2/5 space-y-4"> <Dialog.Panel className="transform rounded-lg bg-custom-background-100 border border-custom-border-100 text-left shadow-xl transition-all w-full sm:w-3/5 lg:w-1/2 xl:w-2/5 ">
<form onSubmit={handleSubmit(handleFormSubmit)} className="space-y-4">
{/* heading */} {/* heading */}
<div className="p-3 px-4 pb-0 flex gap-2 justify-between items-center"> <div className="px-6 pt-4 flex items-center justify-between gap-2">
<div className="font-medium text-xl">Publish</div> <h5 className="font-semibold text-xl inline-block">Publish</h5>
{projectPublish.loader && ( {watch("id") && (
<div className="text-xs text-custom-text-400">Changes saved</div> <DangerButton
)} onClick={() => handleUnpublishProject(watch("id") ?? "")}
<div className="!px-2 !py-1.5"
className="hover:bg-custom-background-90 w-[30px] h-[30px] rounded flex justify-center items-center cursor-pointer transition-all" loading={isUnpublishing}
onClick={handleClose}
> >
<span className="material-symbols-rounded text-[16px]">close</span> {isUnpublishing ? "Unpublishing..." : "Unpublish"}
</div> </DangerButton>
)}
</div> </div>
{/* content */} {/* content */}
<div className="space-y-3"> <div className="space-y-3 px-6">
{watch("id") && ( <div className="border border-custom-border-100 bg-custom-background-80 rounded-md px-3 py-2 relative flex gap-2 items-center">
<div className="flex items-center gap-1 px-4 text-custom-primary-100"> <div className="truncate flex-grow text-sm">
<div className="w-[20px] h-[20px] overflow-hidden flex items-center">
<span className="material-symbols-rounded text-[18px]">
radio_button_checked
</span>
</div>
<div className="text-sm">This project is live on web</div>
</div>
)}
<div className="mx-4 border border-custom-border-100 bg-custom-background-90 rounded p-3 py-2 relative flex gap-2 items-center">
<div className="relative line-clamp-1 overflow-hidden w-full text-sm">
{`${plane_deploy_url}/${workspaceSlug}/${projectPublish.project_id}`} {`${plane_deploy_url}/${workspaceSlug}/${projectPublish.project_id}`}
</div> </div>
<div className="flex-shrink-0 relative flex items-center gap-1"> <div className="flex-shrink-0 relative flex items-center gap-1">
<a
href={`${plane_deploy_url}/${workspaceSlug}/${projectPublish.project_id}`}
target="_blank"
rel="noreferrer"
>
<div className="border border-custom-border-100 bg-custom-background-100 w-[30px] h-[30px] rounded flex justify-center items-center hover:bg-custom-background-90 cursor-pointer">
<span className="material-symbols-rounded text-[16px]">open_in_new</span>
</div>
</a>
<CopyLinkToClipboard <CopyLinkToClipboard
copy_link={`${plane_deploy_url}/${workspaceSlug}/${projectPublish.project_id}`} copy_link={`${plane_deploy_url}/${workspaceSlug}/${projectPublish.project_id}`}
/> />
</div> </div>
</div> </div>
<div className="space-y-3 px-4"> {watch("id") && (
<div className="flex items-center gap-1 text-custom-primary-100">
<div className="w-5 h-5 overflow-hidden flex items-center">
<Icon iconName="radio_button_checked" className="!text-lg" />
</div>
<div className="text-sm">This project is live on web</div>
</div>
)}
<div className="space-y-4">
<div className="relative flex justify-between items-center gap-2"> <div className="relative flex justify-between items-center gap-2">
<div className="text-custom-text-100">Views</div> <div className="text-sm">Views</div>
<div> <Controller
control={control}
name="views"
render={({ field: { onChange, value } }) => (
<CustomPopover <CustomPopover
label={ label={
watch("views") && watch("views").length > 0 value.length > 0
? viewOptions ? viewOptions
.filter( .filter((v) => value.includes(v.key))
(_view) => watch("views").includes(_view.key) && _view.value .map((v) => v.label)
)
.map((_view) => _view.value)
.join(", ") .join(", ")
: `` : ``
} }
placeholder="Select views" placeholder="Select views"
> >
<> <>
{viewOptions && {viewOptions.map((option) => (
viewOptions.length > 0 &&
viewOptions.map((_view) => (
<div <div
key={_view.value} key={option.key}
className={`relative flex items-center gap-2 justify-between p-1 m-1 px-2 cursor-pointer rounded-sm text-custom-text-200 ${ className={`relative flex items-center gap-2 justify-between p-1 m-1 px-2 cursor-pointer rounded-sm text-custom-text-200 ${
watch("views").includes(_view.key) value.includes(option.key)
? `bg-custom-background-80 text-custom-text-100` ? "bg-custom-background-80 text-custom-text-100"
: `hover:bg-custom-background-80 hover:text-custom-text-100` : "hover:bg-custom-background-80 hover:text-custom-text-100"
}`} }`}
onClick={() => { onClick={() => {
const _views = const _views =
watch("views") && watch("views").length > 0 value.length > 0
? watch("views").includes(_view?.key) ? value.includes(option.key)
? watch("views").filter((_o: string) => _o !== _view?.key) ? value.filter((_o: string) => _o !== option.key)
: [...watch("views"), _view?.key] : [...value, option.key]
: [_view?.key]; : [option.key];
setValue("views", _views);
if (watch("id") != null) onSettingsUpdate("views", _views); if (_views.length === 0) return;
onChange(_views);
checkIfUpdateIsRequired();
}} }}
> >
<div className="text-sm">{_view.value}</div> <div className="text-sm">{option.label}</div>
<div <div
className={`w-[18px] h-[18px] relative flex justify-center items-center`} className={`w-[18px] h-[18px] relative flex justify-center items-center`}
> >
{watch("views") && {value.length > 0 && value.includes(option.key) && (
watch("views").length > 0 && <Icon iconName="done" className="!text-lg" />
watch("views").includes(_view.key) && (
<span className="material-symbols-rounded text-[18px]">
done
</span>
)} )}
</div> </div>
</div> </div>
))} ))}
</> </>
</CustomPopover> </CustomPopover>
</div> )}
/>
</div> </div>
{/* <div className="relative flex justify-between items-center gap-2"> <div className="relative flex justify-between items-center gap-2">
<div className="text-custom-text-100">Allow comments</div> <div className="text-sm">Allow comments</div>
<div> <Controller
control={control}
name="comments"
render={({ field: { onChange, value } }) => (
<ToggleSwitch <ToggleSwitch
value={watch("comments") ?? false} value={value}
onChange={() => { onChange={(val) => {
const _comments = !watch("comments"); onChange(val);
setValue("comments", _comments); checkIfUpdateIsRequired();
if (watch("id") != null) onSettingsUpdate("comments", _comments);
}} }}
size="sm" size="sm"
/> />
)}
/>
</div> </div>
</div> */} <div className="relative flex justify-between items-center gap-2">
<div className="text-sm">Allow reactions</div>
{/* <div className="relative flex justify-between items-center gap-2"> <Controller
<div className="text-custom-text-100">Allow reactions</div> control={control}
<div> name="reactions"
render={({ field: { onChange, value } }) => (
<ToggleSwitch <ToggleSwitch
value={watch("reactions") ?? false} value={value}
onChange={() => { onChange={(val) => {
const _reactions = !watch("reactions"); onChange(val);
setValue("reactions", _reactions); checkIfUpdateIsRequired();
if (watch("id") != null) onSettingsUpdate("reactions", _reactions);
}} }}
size="sm" size="sm"
/> />
)}
/>
</div> </div>
</div> */} <div className="relative flex justify-between items-center gap-2">
<div className="text-sm">Allow voting</div>
{/* <div className="relative flex justify-between items-center gap-2"> <Controller
<div className="text-custom-text-100">Allow Voting</div> control={control}
<div> name="votes"
render={({ field: { onChange, value } }) => (
<ToggleSwitch <ToggleSwitch
value={watch("votes") ?? false} value={value}
onChange={() => { onChange={(val) => {
const _votes = !watch("votes"); onChange(val);
setValue("votes", _votes); checkIfUpdateIsRequired();
if (watch("id") != null) onSettingsUpdate("votes", _votes);
}} }}
size="sm" size="sm"
/> />
</div> )}
</div> */}
{/* <div className="relative flex justify-between items-center gap-2">
<div className="text-custom-text-100">Allow issue proposals</div>
<div>
<ToggleSwitch
value={watch("inbox") ?? false}
onChange={() => {
setValue("inbox", !watch("inbox"));
}}
size="sm"
/> />
</div> </div>
{/* <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> </div>
</div> </div>
{/* modal handlers */} {/* modal handlers */}
<div className="border-t border-custom-border-300 p-3 px-4 relative flex justify-between items-center"> <div className="border-t border-custom-border-200 px-6 py-5 relative flex justify-between items-center">
<div className="flex items-center gap-1 text-custom-text-300"> <div className="flex items-center gap-1 text-custom-text-400 text-sm">
<div className="w-[20px] h-[20px] overflow-hidden flex items-center"> <Icon iconName="public" className="!text-base" />
<span className="material-symbols-rounded text-[18px]">public</span>
</div>
<div className="text-sm">Anyone with the link can access</div> <div className="text-sm">Anyone with the link can access</div>
</div> </div>
<div className="relative flex items-center gap-2"> <div className="relative flex items-center gap-2">
<SecondaryButton onClick={handleClose}>Cancel</SecondaryButton> <SecondaryButton onClick={handleClose}>Cancel</SecondaryButton>
{watch("id") != null ? ( {watch("id") ? (
<PrimaryButton <>
outline {isUpdateRequired && (
onClick={handleSubmit(onSettingsUnPublish)} <PrimaryButton type="submit" loading={isSubmitting}>
disabled={isSubmitting} {isSubmitting ? "Updating..." : "Update settings"}
>
{isSubmitting ? "Unpublishing..." : "Unpublish"}
</PrimaryButton> </PrimaryButton>
)}
</>
) : ( ) : (
<PrimaryButton <PrimaryButton type="submit" loading={isSubmitting}>
onClick={handleSubmit(onSettingsPublish)}
disabled={isSubmitting}
>
{isSubmitting ? "Publishing..." : "Publish"} {isSubmitting ? "Publishing..." : "Publish"}
</PrimaryButton> </PrimaryButton>
)} )}
</div> </div>
</div> </div>
</form>
</Dialog.Panel> </Dialog.Panel>
</Transition.Child> </Transition.Child>
</div> </div>

View File

@ -1,6 +1,9 @@
import React, { Fragment } from "react"; import React, { Fragment } from "react";
// headless ui // headless ui
import { Popover, Transition } from "@headlessui/react"; import { Popover, Transition } from "@headlessui/react";
// icons
import { Icon } from "components/ui";
export const CustomPopover = ({ export const CustomPopover = ({
children, children,
@ -16,18 +19,14 @@ export const CustomPopover = ({
{({ open }) => ( {({ open }) => (
<> <>
<Popover.Button <Popover.Button
className={`${ className={`${open ? "" : ""} relative flex items-center gap-1 ring-0 outline-none`}
open ? "" : ""
} relative flex items-center gap-1 border border-custom-border-300 shadow-sm p-1 px-2 ring-0 outline-none`}
> >
<div className="text-sm font-medium"> <div className="text-sm">{label ?? placeholder}</div>
{label ? label : placeholder ? placeholder : "Select"} <div className="w-5 h-5 grid place-items-center">
</div>
<div className="w-[20px] h-[20px] relative flex justify-center items-center">
{!open ? ( {!open ? (
<span className="material-symbols-rounded text-[20px]">expand_more</span> <Icon iconName="expand_more" className="!text-base" />
) : ( ) : (
<span className="material-symbols-rounded text-[20px]">expand_less</span> <Icon iconName="expand_less" className="!text-base" />
)} )}
</div> </div>
</Popover.Button> </Popover.Button>
@ -41,8 +40,8 @@ export const CustomPopover = ({
leaveFrom="opacity-100 translate-y-0" leaveFrom="opacity-100 translate-y-0"
leaveTo="opacity-0 translate-y-1" leaveTo="opacity-0 translate-y-1"
> >
<Popover.Panel className="absolute right-0 z-[9999]"> <Popover.Panel className="absolute right-0 z-10 mt-1 min-w-[150px]">
<div className="overflow-hidden rounded-sm border border-custom-border-300 mt-1 overflow-y-auto bg-custom-background-90 shadow-lg focus:outline-none"> <div className="overflow-hidden rounded border border-custom-border-300 mt-1 overflow-y-auto bg-custom-background-90 shadow-custom-shadow-2xs focus:outline-none">
{children} {children}
</div> </div>
</Popover.Panel> </Popover.Panel>

View File

@ -26,7 +26,6 @@ import {
SettingsOutlined, SettingsOutlined,
} from "@mui/icons-material"; } from "@mui/icons-material";
// helpers // helpers
import { truncateText } from "helpers/string.helper";
import { renderEmoji } from "helpers/emoji.helper"; import { renderEmoji } from "helpers/emoji.helper";
// types // types
import { IProject } from "types"; import { IProject } from "types";
@ -265,11 +264,10 @@ export const SingleSidebarProject: React.FC<Props> = observer(
> >
<div className="flex-shrink-0 relative flex items-center justify-start gap-2"> <div className="flex-shrink-0 relative flex items-center justify-start gap-2">
<div className="rounded transition-all w-4 h-4 flex justify-center items-center text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 duration-300 cursor-pointer"> <div className="rounded transition-all w-4 h-4 flex justify-center items-center text-custom-sidebar-text-200 hover:bg-custom-sidebar-background-80 duration-300 cursor-pointer">
<span className="material-symbols-rounded text-[16px]">ios_share</span> <Icon iconName="ios_share" className="!text-base" />
</div> </div>
<div>Publish</div> <div>{project.is_deployed ? "Publish settings" : "Publish"}</div>
</div> </div>
{/* <PublishProjectModal /> */}
</CustomMenu.MenuItem> </CustomMenu.MenuItem>
)} )}

View File

@ -4,21 +4,11 @@ import { RootStore } from "./root";
// services // services
import ProjectServices from "services/project-publish.service"; import ProjectServices from "services/project-publish.service";
export type IProjectPublishSettingsViewKeys = export type TProjectPublishViews = "list" | "gantt" | "kanban" | "calendar" | "spreadsheet";
| "list"
| "gantt"
| "kanban"
| "calendar"
| "spreadsheet"
| string;
export interface IProjectPublishSettingsViews { export type TProjectPublishViewsSettings = {
list: boolean; [key in TProjectPublishViews]: boolean;
gantt: boolean; };
kanban: boolean;
calendar: boolean;
spreadsheet: boolean;
}
export interface IProjectPublishSettings { export interface IProjectPublishSettings {
id?: string; id?: string;
@ -26,8 +16,8 @@ export interface IProjectPublishSettings {
comments: boolean; comments: boolean;
reactions: boolean; reactions: boolean;
votes: boolean; votes: boolean;
views: IProjectPublishSettingsViews; views: TProjectPublishViewsSettings;
inbox: null; inbox: string | null;
} }
export interface IProjectPublishStore { export interface IProjectPublishStore {