refactor: publish project modal (#2514)

* chore: add publish badge to the header

* refactor: project oublish components

* chore: remove link tag
This commit is contained in:
Aaryan Khandelwal 2023-10-23 12:12:42 +05:30 committed by GitHub
parent 914657334d
commit c739b7235d
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
7 changed files with 194 additions and 163 deletions

View File

@ -1,9 +1,9 @@
import { useCallback, useState, FC } from "react"; import { useCallback, useState } from "react";
import Link from "next/link"; import Link from "next/link";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
import { ArrowLeft, Plus } from "lucide-react"; import { ArrowLeft, Circle, ExternalLink, Plus } from "lucide-react";
// hooks // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues"; import { DisplayFiltersSelection, FiltersDropdown, FilterSelection, LayoutSelection } from "components/issues";
@ -17,7 +17,7 @@ import { ISSUE_DISPLAY_FILTERS_BY_LAYOUT } from "constants/issue";
// helper // helper
import { truncateText } from "helpers/string.helper"; import { truncateText } from "helpers/string.helper";
export const ProjectIssuesHeader: FC = observer(() => { export const ProjectIssuesHeader: React.FC = observer(() => {
const [analyticsModal, setAnalyticsModal] = useState(false); const [analyticsModal, setAnalyticsModal] = useState(false);
const router = useRouter(); const router = useRouter();
@ -93,6 +93,8 @@ export const ProjectIssuesHeader: FC = observer(() => {
const inboxDetails = projectId ? inboxStore.inboxesList?.[projectId.toString()]?.[0] : undefined; const inboxDetails = projectId ? inboxStore.inboxesList?.[projectId.toString()]?.[0] : undefined;
const deployUrl = process.env.NEXT_PUBLIC_DEPLOY_URL;
return ( return (
<> <>
<ProjectAnalyticsModal <ProjectAnalyticsModal
@ -125,6 +127,18 @@ export const ProjectIssuesHeader: FC = observer(() => {
<BreadcrumbItem title={`${truncateText(projectDetails?.name ?? "Project", 32)} Issues`} /> <BreadcrumbItem title={`${truncateText(projectDetails?.name ?? "Project", 32)} Issues`} />
</Breadcrumbs> </Breadcrumbs>
</div> </div>
{projectDetails?.is_deployed && deployUrl && (
<a
href={`${deployUrl}/${workspaceSlug}/${projectDetails?.id}`}
className="group bg-custom-primary-100/20 text-custom-primary-100 px-2.5 py-1 text-xs flex items-center gap-1.5 rounded font-medium"
target="_blank"
rel="noopener noreferrer"
>
<Circle className="h-1.5 w-1.5 fill-custom-primary-100" strokeWidth={2} />
Public
<ExternalLink className="h-3 w-3 hidden group-hover:block" strokeWidth={2} />
</a>
)}
</div> </div>
<div className="flex items-center gap-2"> <div className="flex items-center gap-2">
<LayoutSelection <LayoutSelection

View File

@ -40,14 +40,15 @@ export const CalendarChart: React.FC<Props> = observer((props) => {
<CalendarHeader /> <CalendarHeader />
<CalendarWeekHeader isLoading={!issues} showWeekends={showWeekends} /> <CalendarWeekHeader isLoading={!issues} showWeekends={showWeekends} />
<div className="h-full w-full overflow-y-auto"> <div className="h-full w-full overflow-y-auto">
{layout === "month" ? ( {layout === "month" && (
<div className="h-full w-full grid grid-cols-1 divide-y-[0.5px] divide-custom-border-200"> <div className="h-full w-full grid grid-cols-1 divide-y-[0.5px] divide-custom-border-200">
{allWeeksOfActiveMonth && {allWeeksOfActiveMonth &&
Object.values(allWeeksOfActiveMonth).map((week: ICalendarWeek, weekIndex) => ( Object.values(allWeeksOfActiveMonth).map((week: ICalendarWeek, weekIndex) => (
<CalendarWeekDays key={weekIndex} week={week} issues={issues} quickActions={quickActions} /> <CalendarWeekDays key={weekIndex} week={week} issues={issues} quickActions={quickActions} />
))} ))}
</div> </div>
) : ( )}
{layout === "week" && (
<CalendarWeekDays week={calendarStore.allDaysOfActiveWeek} issues={issues} quickActions={quickActions} /> <CalendarWeekDays week={calendarStore.allDaysOfActiveWeek} issues={issues} quickActions={quickActions} />
)} )}
</div> </div>

View File

@ -15,3 +15,4 @@ export * from "./join-project-modal";
export * from "./form"; export * from "./form";
export * from "./form-loader"; export * from "./form-loader";
export * from "./delete-project-section"; export * from "./delete-project-section";
export * from "./publish-project";

View File

@ -1,27 +1,24 @@
import React, { useEffect, useState } from "react"; import React, { useEffect, useState } from "react";
// next imports
import { useRouter } from "next/router"; import { useRouter } from "next/router";
// react-hook-form import { observer } from "mobx-react-lite";
import { Controller, useForm } from "react-hook-form"; import { Controller, useForm } from "react-hook-form";
// headless ui
import { Dialog, Transition } from "@headlessui/react"; import { Dialog, Transition } from "@headlessui/react";
// mobx store
import { useMobxStore } from "lib/mobx/store-provider";
// ui components // ui components
import { Button, Loader, ToggleSwitch } from "@plane/ui"; import { Button, Loader, ToggleSwitch } from "@plane/ui";
import { Check, CircleDot, Globe2 } from "lucide-react"; import { Check, CircleDot, Globe2 } from "lucide-react";
import { CustomPopover } from "./popover"; import { CustomPopover } from "./popover";
// mobx react lite
import { observer } from "mobx-react-lite";
// mobx store
import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root";
import { IProjectPublishSettings, TProjectPublishViews } from "store/project"; import { IProjectPublishSettings, TProjectPublishViews } from "store/project";
// hooks // hooks
import useToast from "hooks/use-toast"; import useToast from "hooks/use-toast";
import useProjectDetails from "hooks/use-project-details"; // types
import useUser from "hooks/use-user"; import { IProject } from "types";
type Props = { type Props = {
// user: IUser | undefined; isOpen: boolean;
project: IProject;
onClose: () => void;
}; };
type FormData = { type FormData = {
@ -53,26 +50,24 @@ const viewOptions: {
// { key: "spreadsheet", label: "Spreadsheet" }, // { key: "spreadsheet", label: "Spreadsheet" },
]; ];
export const PublishProjectModal: React.FC<Props> = observer(() => { export const PublishProjectModal: React.FC<Props> = observer((props) => {
const { isOpen, project, onClose } = props;
const [isUnpublishing, setIsUnpublishing] = useState(false); const [isUnpublishing, setIsUnpublishing] = useState(false);
const [isUpdateRequired, setIsUpdateRequired] = useState(false); const [isUpdateRequired, setIsUpdateRequired] = useState(false);
let plane_deploy_url = process.env.NEXT_PUBLIC_DEPLOY_URL; let plane_deploy_url = process.env.NEXT_PUBLIC_DEPLOY_URL;
if (typeof window !== "undefined" && !plane_deploy_url) { if (typeof window !== "undefined" && !plane_deploy_url)
plane_deploy_url = window.location.protocol + "//" + window.location.host + "/spaces"; plane_deploy_url = window.location.protocol + "//" + window.location.host + "/spaces";
}
// router
const router = useRouter(); const router = useRouter();
const { workspaceSlug } = router.query; const { workspaceSlug } = router.query;
// store
const store: RootStore = useMobxStore(); const { projectPublish: projectPublishStore } = useMobxStore();
const { projectPublish } = store;
// hooks
const { user } = useUser();
const { mutateProjectDetails } = useProjectDetails();
const { setToastAlert } = useToast(); const { setToastAlert } = useToast();
// form info
const { const {
control, control,
formState: { isSubmitting }, formState: { isSubmitting },
@ -80,23 +75,27 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
handleSubmit, handleSubmit,
reset, reset,
watch, watch,
} = useForm<FormData>({ } = useForm({
defaultValues, defaultValues,
}); });
const handleClose = () => { const handleClose = () => {
projectPublish.handleProjectModal(null); onClose();
setIsUpdateRequired(false); setIsUpdateRequired(false);
reset({ ...defaultValues }); reset({ ...defaultValues });
}; };
// prefill form with the saved settings if the project is already published // prefill form with the saved settings if the project is already published
useEffect(() => { useEffect(() => {
if (projectPublish.projectPublishSettings && projectPublish.projectPublishSettings !== "not-initialized") { if (
projectPublishStore.projectPublishSettings &&
projectPublishStore.projectPublishSettings !== "not-initialized"
) {
let userBoards: TProjectPublishViews[] = []; let userBoards: TProjectPublishViews[] = [];
if (projectPublish.projectPublishSettings?.views) { if (projectPublishStore.projectPublishSettings?.views) {
const savedViews = projectPublish.projectPublishSettings?.views; const savedViews = projectPublishStore.projectPublishSettings?.views;
if (!savedViews) return; if (!savedViews) return;
@ -110,61 +109,46 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
} }
const updatedData = { const updatedData = {
id: projectPublish.projectPublishSettings?.id || null, id: projectPublishStore.projectPublishSettings?.id || null,
comments: projectPublish.projectPublishSettings?.comments || false, comments: projectPublishStore.projectPublishSettings?.comments || false,
reactions: projectPublish.projectPublishSettings?.reactions || false, reactions: projectPublishStore.projectPublishSettings?.reactions || false,
votes: projectPublish.projectPublishSettings?.votes || false, votes: projectPublishStore.projectPublishSettings?.votes || false,
inbox: projectPublish.projectPublishSettings?.inbox || null, inbox: projectPublishStore.projectPublishSettings?.inbox || null,
views: userBoards, views: userBoards,
}; };
reset({ ...updatedData }); reset({ ...updatedData });
} }
}, [reset, projectPublish.projectPublishSettings]); }, [reset, projectPublishStore.projectPublishSettings]);
// fetch publish settings // fetch publish settings
useEffect(() => { useEffect(() => {
if (!workspaceSlug) return; if (!workspaceSlug || !isOpen) return;
if ( if (projectPublishStore.projectPublishSettings === "not-initialized") {
projectPublish.projectPublishModal && projectPublishStore.getProjectSettingsAsync(workspaceSlug.toString(), project.id);
projectPublish.project_id !== null &&
projectPublish?.projectPublishSettings === "not-initialized"
) {
projectPublish.getProjectSettingsAsync(workspaceSlug.toString(), projectPublish.project_id, null);
} }
}, [workspaceSlug, projectPublish, projectPublish.projectPublishModal]); }, [isOpen, workspaceSlug, project, projectPublishStore]);
const handlePublishProject = async (payload: IProjectPublishSettings) => { const handlePublishProject = async (payload: IProjectPublishSettings) => {
if (!workspaceSlug || !user) return; if (!workspaceSlug) return;
const projectId = projectPublish.project_id; return projectPublishStore
.publishProject(workspaceSlug.toString(), project.id, payload)
return projectPublish
.publishProject(workspaceSlug.toString(), projectId?.toString() ?? "", payload, user)
.then((res) => { .then((res) => {
mutateProjectDetails();
handleClose(); handleClose();
if (projectId) window.open(`${plane_deploy_url}/${workspaceSlug}/${projectId}`, "_blank"); // window.open(`${plane_deploy_url}/${workspaceSlug}/${project.id}`, "_blank");
return res; return res;
}) })
.catch((err) => err); .catch((err) => err);
}; };
const handleUpdatePublishSettings = async (payload: IProjectPublishSettings) => { const handleUpdatePublishSettings = async (payload: IProjectPublishSettings) => {
if (!workspaceSlug || !user) return; if (!workspaceSlug) return;
await projectPublish await projectPublishStore
.updateProjectSettingsAsync( .updateProjectSettingsAsync(workspaceSlug.toString(), project.id, payload.id ?? "", payload)
workspaceSlug.toString(),
projectPublish.project_id?.toString() ?? "",
payload.id ?? "",
payload,
user
)
.then((res) => { .then((res) => {
mutateProjectDetails();
setToastAlert({ setToastAlert({
type: "success", type: "success",
title: "Success!", title: "Success!",
@ -185,15 +169,19 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
setIsUnpublishing(true); setIsUnpublishing(true);
projectPublish await projectPublishStore
.unPublishProject(workspaceSlug.toString(), projectPublish.project_id as string, publishId, null) .unPublishProject(workspaceSlug.toString(), project.id, publishId)
.then((res) => { .then((res) => {
mutateProjectDetails();
handleClose(); handleClose();
return res; return res;
}) })
.catch((err) => err) .catch(() =>
setToastAlert({
type: "error",
title: "Error!",
message: "Something went wrong while unpublishing the project.",
})
)
.finally(() => setIsUnpublishing(false)); .finally(() => setIsUnpublishing(false));
}; };
@ -242,15 +230,16 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
}, },
}; };
if (watch("id")) await handleUpdatePublishSettings({ id: watch("id") ?? "", ...payload }); if (project.is_deployed) await handleUpdatePublishSettings({ id: watch("id") ?? "", ...payload });
else await handlePublishProject(payload); else await handlePublishProject(payload);
}; };
// check if an update is required or not // check if an update is required or not
const checkIfUpdateIsRequired = () => { const checkIfUpdateIsRequired = () => {
if (!projectPublish.projectPublishSettings || projectPublish.projectPublishSettings === "not-initialized") return; if (!projectPublishStore.projectPublishSettings || projectPublishStore.projectPublishSettings === "not-initialized")
return;
const currentSettings = projectPublish.projectPublishSettings as IProjectPublishSettings; const currentSettings = projectPublishStore.projectPublishSettings as IProjectPublishSettings;
const newSettings = getValues(); const newSettings = getValues();
if ( if (
@ -276,7 +265,7 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
}; };
return ( return (
<Transition.Root show={projectPublish.projectPublishModal} as={React.Fragment}> <Transition.Root show={isOpen} as={React.Fragment}>
<Dialog as="div" className="relative z-20" onClose={handleClose}> <Dialog as="div" className="relative z-20" onClose={handleClose}>
<Transition.Child <Transition.Child
as={React.Fragment} as={React.Fragment}
@ -301,12 +290,12 @@ 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 "> <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"> <form onSubmit={handleSubmit(handleFormSubmit)} className="space-y-4">
{/* heading */} {/* heading */}
<div className="px-6 pt-4 flex items-center justify-between gap-2"> <div className="px-6 pt-4 flex items-center justify-between gap-2">
<h5 className="font-semibold text-xl inline-block">Publish</h5> <h5 className="font-semibold text-xl inline-block">Publish</h5>
{projectPublish.projectPublishSettings !== "not-initialized" && ( {project.is_deployed && (
<Button <Button
variant="danger" variant="danger"
onClick={() => handleUnpublishProject(watch("id") ?? "")} onClick={() => handleUnpublishProject(watch("id") ?? "")}
@ -318,7 +307,7 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
</div> </div>
{/* content */} {/* content */}
{projectPublish.fetchSettingsLoader ? ( {projectPublishStore.fetchSettingsLoader ? (
<Loader className="px-6 space-y-4"> <Loader className="px-6 space-y-4">
<Loader.Item height="30px" /> <Loader.Item height="30px" />
<Loader.Item height="30px" /> <Loader.Item height="30px" />
@ -327,16 +316,14 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
</Loader> </Loader>
) : ( ) : (
<div className="px-6"> <div className="px-6">
{watch("id") && ( {project.is_deployed && (
<> <>
<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="border border-custom-border-100 bg-custom-background-80 rounded-md px-3 py-2 relative flex gap-2 items-center">
<div className="truncate flex-grow text-sm"> <div className="truncate flex-grow text-sm">
{`${plane_deploy_url}/${workspaceSlug}/${projectPublish.project_id}`} {`${plane_deploy_url}/${workspaceSlug}/${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">
<CopyLinkToClipboard <CopyLinkToClipboard copy_link={`${plane_deploy_url}/${workspaceSlug}/${project.id}`} />
copy_link={`${plane_deploy_url}/${workspaceSlug}/${projectPublish.project_id}`}
/>
</div> </div>
</div> </div>
<div className="flex items-center gap-1 text-custom-primary-100 mt-3"> <div className="flex items-center gap-1 text-custom-primary-100 mt-3">
@ -454,6 +441,7 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
/> />
</div> </div>
{/* toggle inbox */}
{/* <div className="relative flex justify-between items-center gap-2"> {/* <div className="relative flex justify-between items-center gap-2">
<div className="text-sm">Allow issue proposals</div> <div className="text-sm">Allow issue proposals</div>
<Controller <Controller
@ -474,12 +462,12 @@ export const PublishProjectModal: React.FC<Props> = observer(() => {
<Globe2 className="h-4 w-4" /> <Globe2 className="h-4 w-4" />
<div className="text-sm">Anyone with the link can access</div> <div className="text-sm">Anyone with the link can access</div>
</div> </div>
{!projectPublish.fetchSettingsLoader && ( {!projectPublishStore.fetchSettingsLoader && (
<div className="relative flex items-center gap-2"> <div className="relative flex items-center gap-2">
<Button variant="neutral-primary" onClick={handleClose}> <Button variant="neutral-primary" onClick={handleClose}>
Cancel Cancel
</Button> </Button>
{watch("id") ? ( {project.is_deployed ? (
<> <>
{isUpdateRequired && ( {isUpdateRequired && (
<Button variant="primary" type="submit" loading={isSubmitting}> <Button variant="primary" type="submit" loading={isSubmitting}>

View File

@ -27,8 +27,7 @@ import { IProject } from "types";
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { CustomMenu, Tooltip, ArchiveIcon, PhotoFilterIcon, DiceIcon, ContrastIcon, LayersIcon } from "@plane/ui"; import { CustomMenu, Tooltip, ArchiveIcon, PhotoFilterIcon, DiceIcon, ContrastIcon, LayersIcon } from "@plane/ui";
import { LeaveProjectModal, DeleteProjectModal } from "components/project"; import { LeaveProjectModal, DeleteProjectModal, PublishProjectModal } from "components/project";
import { PublishProjectModal } from "components/project/publish-project";
type Props = { type Props = {
project: IProject; project: IProject;
@ -74,7 +73,7 @@ const navigation = (workspaceSlug: string, projectId: string) => [
export const ProjectSidebarListItem: React.FC<Props> = observer((props) => { export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
const { project, provided, snapshot, handleCopyText, shortContextMenu = false } = props; const { project, provided, snapshot, handleCopyText, shortContextMenu = false } = props;
// store // store
const { projectPublish, project: projectStore, theme: themeStore } = useMobxStore(); const { project: projectStore, theme: themeStore } = useMobxStore();
// router // router
const router = useRouter(); const router = useRouter();
const { workspaceSlug, projectId } = router.query; const { workspaceSlug, projectId } = router.query;
@ -83,6 +82,7 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
// states // states
const [leaveProjectModalOpen, setLeaveProjectModal] = useState(false); const [leaveProjectModalOpen, setLeaveProjectModal] = useState(false);
const [deleteProjectModalOpen, setDeleteProjectModal] = useState(false); const [deleteProjectModalOpen, setDeleteProjectModal] = useState(false);
const [publishModalOpen, setPublishModal] = useState(false);
const isAdmin = project.member_role === 20; const isAdmin = project.member_role === 20;
const isViewerOrGuest = project.member_role === 10 || project.member_role === 5; const isViewerOrGuest = project.member_role === 10 || project.member_role === 5;
@ -132,7 +132,7 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
return ( return (
<> <>
<PublishProjectModal /> <PublishProjectModal isOpen={publishModalOpen} project={project} onClose={() => setPublishModal(false)} />
<DeleteProjectModal project={project} isOpen={deleteProjectModalOpen} onClose={handleDeleteProjectModalClose} /> <DeleteProjectModal project={project} isOpen={deleteProjectModalOpen} onClose={handleDeleteProjectModalClose} />
<LeaveProjectModal project={project} isOpen={leaveProjectModalOpen} onClose={handleLeaveProjectModalClose} /> <LeaveProjectModal project={project} isOpen={leaveProjectModalOpen} onClose={handleLeaveProjectModalClose} />
<Disclosure key={project.id} defaultOpen={projectId === project.id}> <Disclosure key={project.id} defaultOpen={projectId === project.id}>
@ -239,7 +239,7 @@ export const ProjectSidebarListItem: React.FC<Props> = observer((props) => {
{/* publish project settings */} {/* publish project settings */}
{isAdmin && ( {isAdmin && (
<CustomMenu.MenuItem onClick={() => projectPublish.handleProjectModal(project?.id)}> <CustomMenu.MenuItem onClick={() => setPublishModal(true)}>
<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">
<Share2 className="h-3.5 w-3.5 stroke-[1.5]" /> <Share2 className="h-3.5 w-3.5 stroke-[1.5]" />

View File

@ -273,9 +273,9 @@ export class ProjectStore implements IProjectStore {
}; };
getProjectById = (workspaceSlug: string, projectId: string) => { getProjectById = (workspaceSlug: string, projectId: string) => {
if (!this.projectId) return null;
const projects = this.projects?.[workspaceSlug]; const projects = this.projects?.[workspaceSlug];
if (!projects) return null; if (!projects) return null;
const projectInfo: IProject | null = projects.find((project) => project.id === projectId) || null; const projectInfo: IProject | null = projects.find((project) => project.id === projectId) || null;
return projectInfo; return projectInfo;
}; };

View File

@ -25,88 +25,67 @@ export interface IProjectPublishStore {
fetchSettingsLoader: boolean; fetchSettingsLoader: boolean;
error: any | null; error: any | null;
projectPublishModal: boolean;
project_id: string | null;
projectPublishSettings: IProjectPublishSettings | "not-initialized"; projectPublishSettings: IProjectPublishSettings | "not-initialized";
handleProjectModal: (project_id: string | null) => void; getProjectSettingsAsync: (workspaceSlug: string, projectId: string) => Promise<void>;
publishProject: (workspaceSlug: string, projectId: string, data: IProjectPublishSettings) => Promise<void>;
getProjectSettingsAsync: (workspace_slug: string, project_slug: string, user: any) => Promise<void>;
publishProject: (
workspace_slug: string,
project_slug: string,
data: IProjectPublishSettings,
user: any
) => Promise<void>;
updateProjectSettingsAsync: ( updateProjectSettingsAsync: (
workspace_slug: string, workspaceSlug: string,
project_slug: string, projectId: string,
project_publish_id: string, projectPublishId: string,
data: IProjectPublishSettings, data: IProjectPublishSettings
user: any
) => Promise<void>;
unPublishProject: (
workspace_slug: string,
project_slug: string,
project_publish_id: string,
user: any
) => Promise<void>; ) => Promise<void>;
unPublishProject: (workspaceSlug: string, projectId: string, projectPublishId: string) => Promise<void>;
} }
export class ProjectPublishStore implements IProjectPublishStore { export class ProjectPublishStore implements IProjectPublishStore {
// states
generalLoader: boolean = false; generalLoader: boolean = false;
fetchSettingsLoader: boolean = false; fetchSettingsLoader: boolean = false;
error: any | null = null; error: any | null = null;
projectPublishModal: boolean = false; // actions
project_id: string | null = null; project_id: string | null = null;
projectPublishSettings: IProjectPublishSettings | "not-initialized" = "not-initialized"; projectPublishSettings: IProjectPublishSettings | "not-initialized" = "not-initialized";
// root store // root store
rootStore; rootStore;
// service
// services
projectPublishService; projectPublishService;
constructor(_rootStore: RootStore) { constructor(_rootStore: RootStore) {
makeObservable(this, { makeObservable(this, {
// observable // states
generalLoader: observable, generalLoader: observable,
fetchSettingsLoader: observable, fetchSettingsLoader: observable,
error: observable, error: observable,
projectPublishModal: observable, // observables
project_id: observable, project_id: observable,
projectPublishSettings: observable.ref, projectPublishSettings: observable.ref,
// action
handleProjectModal: action, // actions
getProjectSettingsAsync: action, getProjectSettingsAsync: action,
publishProject: action, publishProject: action,
updateProjectSettingsAsync: action, updateProjectSettingsAsync: action,
unPublishProject: action, unPublishProject: action,
// computed
}); });
this.rootStore = _rootStore; this.rootStore = _rootStore;
// services
this.projectPublishService = new ProjectPublishService(); this.projectPublishService = new ProjectPublishService();
} }
handleProjectModal = (project_id: string | null = null) => { getProjectSettingsAsync = async (workspaceSlug: string, projectId: string) => {
if (project_id) {
this.projectPublishModal = !this.projectPublishModal;
this.project_id = project_id;
} else {
this.projectPublishModal = !this.projectPublishModal;
this.project_id = null;
this.projectPublishSettings = "not-initialized";
}
};
getProjectSettingsAsync = async (workspace_slug: string, project_slug: string) => {
try { try {
runInAction(() => {
this.fetchSettingsLoader = true; this.fetchSettingsLoader = true;
this.error = null; this.error = null;
});
const response = await this.projectPublishService.getProjectSettingsAsync(workspace_slug, project_slug); const response = await this.projectPublishService.getProjectSettingsAsync(workspaceSlug, projectId);
if (response && response.length > 0) { if (response && response.length > 0) {
const _projectPublishSettings: IProjectPublishSettings = { const _projectPublishSettings: IProjectPublishSettings = {
@ -131,24 +110,31 @@ export class ProjectPublishStore implements IProjectPublishStore {
this.error = null; this.error = null;
}); });
} else { } else {
runInAction(() => {
this.projectPublishSettings = "not-initialized"; this.projectPublishSettings = "not-initialized";
this.fetchSettingsLoader = false; this.fetchSettingsLoader = false;
this.error = null; this.error = null;
});
} }
return response; return response;
} catch (error) { } catch (error) {
runInAction(() => {
this.fetchSettingsLoader = false; this.fetchSettingsLoader = false;
this.error = error; this.error = error;
});
return error; return error;
} }
}; };
publishProject = async (workspace_slug: string, project_slug: string, data: IProjectPublishSettings) => { publishProject = async (workspaceSlug: string, projectId: string, data: IProjectPublishSettings) => {
try { try {
runInAction(() => {
this.generalLoader = true; this.generalLoader = true;
this.error = null; this.error = null;
});
const response = await this.projectPublishService.createProjectSettingsAsync(workspace_slug, project_slug, data); const response = await this.projectPublishService.createProjectSettingsAsync(workspaceSlug, projectId, data);
if (response) { if (response) {
const _projectPublishSettings: IProjectPublishSettings = { const _projectPublishSettings: IProjectPublishSettings = {
@ -163,6 +149,20 @@ export class ProjectPublishStore implements IProjectPublishStore {
runInAction(() => { runInAction(() => {
this.projectPublishSettings = _projectPublishSettings; this.projectPublishSettings = _projectPublishSettings;
this.rootStore.project.projects = {
...this.rootStore.project.projects,
[workspaceSlug]: this.rootStore.project.projects[workspaceSlug].map((p) => ({
...p,
is_deployed: p.id === projectId ? true : p.is_deployed,
})),
};
this.rootStore.project.project_details = {
...this.rootStore.project.project_details,
[projectId]: {
...this.rootStore.project.project_details[projectId],
is_deployed: true,
},
};
this.generalLoader = false; this.generalLoader = false;
this.error = null; this.error = null;
}); });
@ -170,26 +170,31 @@ export class ProjectPublishStore implements IProjectPublishStore {
return response; return response;
} }
} catch (error) { } catch (error) {
runInAction(() => {
this.generalLoader = false; this.generalLoader = false;
this.error = error; this.error = error;
});
return error; return error;
} }
}; };
updateProjectSettingsAsync = async ( updateProjectSettingsAsync = async (
workspace_slug: string, workspaceSlug: string,
project_slug: string, projectId: string,
project_publish_id: string, projectPublishId: string,
data: IProjectPublishSettings data: IProjectPublishSettings
) => { ) => {
try { try {
runInAction(() => {
this.generalLoader = true; this.generalLoader = true;
this.error = null; this.error = null;
});
const response = await this.projectPublishService.updateProjectSettingsAsync( const response = await this.projectPublishService.updateProjectSettingsAsync(
workspace_slug, workspaceSlug,
project_slug, projectId,
project_publish_id, projectPublishId,
data data
); );
@ -213,33 +218,55 @@ export class ProjectPublishStore implements IProjectPublishStore {
return response; return response;
} }
} catch (error) { } catch (error) {
runInAction(() => {
this.generalLoader = false; this.generalLoader = false;
this.error = error; this.error = error;
});
return error; return error;
} }
}; };
unPublishProject = async (workspace_slug: string, project_slug: string, project_publish_id: string) => { unPublishProject = async (workspaceSlug: string, projectId: string, projectPublishId: string) => {
try { try {
runInAction(() => {
this.generalLoader = true; this.generalLoader = true;
this.error = null; this.error = null;
});
const response = await this.projectPublishService.deleteProjectSettingsAsync( const response = await this.projectPublishService.deleteProjectSettingsAsync(
workspace_slug, workspaceSlug,
project_slug, projectId,
project_publish_id projectPublishId
); );
runInAction(() => { runInAction(() => {
this.projectPublishSettings = "not-initialized"; this.projectPublishSettings = "not-initialized";
this.rootStore.project.projects = {
...this.rootStore.project.projects,
[workspaceSlug]: this.rootStore.project.projects[workspaceSlug].map((p) => ({
...p,
is_deployed: p.id === projectId ? false : p.is_deployed,
})),
};
this.rootStore.project.project_details = {
...this.rootStore.project.project_details,
[projectId]: {
...this.rootStore.project.project_details[projectId],
is_deployed: false,
},
};
this.generalLoader = false; this.generalLoader = false;
this.error = null; this.error = null;
}); });
return response; return response;
} catch (error) { } catch (error) {
runInAction(() => {
this.generalLoader = false; this.generalLoader = false;
this.error = error; this.error = error;
});
return error; return error;
} }
}; };