fix:mobx fixes

This commit is contained in:
sriram veeraghanta 2023-08-30 20:33:21 +05:30
parent 0deed77f1d
commit 579347c991
35 changed files with 699 additions and 595 deletions

View File

@ -1,7 +1,7 @@
"use client"; "use client";
// types // types
import { TIssuePriorityKey } from "store/types/issue"; import { TIssuePriorityKey } from "types/issue";
// constants // constants
import { issuePriorityFilter } from "constants/data"; import { issuePriorityFilter } from "constants/data";

View File

@ -10,7 +10,7 @@ import { IssueBlockState } from "components/issues/board-views/block-state";
import { IssueBlockLabels } from "components/issues/board-views/block-labels"; import { IssueBlockLabels } from "components/issues/board-views/block-labels";
import { IssueBlockDueDate } from "components/issues/board-views/block-due-date"; import { IssueBlockDueDate } from "components/issues/board-views/block-due-date";
// interfaces // interfaces
import { IIssue } from "store/types/issue"; import { IIssue } from "types/issue";
import { RootStore } from "store/root"; import { RootStore } from "store/root";
export const IssueListBlock = observer(({ issue }: { issue: IIssue }) => { export const IssueListBlock = observer(({ issue }: { issue: IIssue }) => {

View File

@ -3,7 +3,7 @@
// mobx react lite // mobx react lite
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// interfaces // interfaces
import { IIssueState } from "store/types/issue"; import { IIssueState } from "types/issue";
// constants // constants
import { issueGroupFilter } from "constants/data"; import { issueGroupFilter } from "constants/data";
// mobx hook // mobx hook
@ -24,7 +24,7 @@ export const IssueListHeader = observer(({ state }: { state: IIssueState }) => {
</div> </div>
<div className="font-semibold text-base capitalize ml-2 mr-3">{state?.name}</div> <div className="font-semibold text-base capitalize ml-2 mr-3">{state?.name}</div>
<div className="text-gray-700 w-full max-w-[26px] h-[20px] flex justify-center items-center rounded-full"> <div className="text-gray-700 w-full max-w-[26px] h-[20px] flex justify-center items-center rounded-full">
{store.issue.getCountOfIssuesByState(state.id)} {/* {store.issue.getCountOfIssuesByState(state.id)} */}
</div> </div>
</div> </div>
); );

View File

@ -6,7 +6,7 @@ import { observer } from "mobx-react-lite";
import { IssueListHeader } from "components/issues/board-views/kanban/header"; import { IssueListHeader } from "components/issues/board-views/kanban/header";
import { IssueListBlock } from "components/issues/board-views/kanban/block"; import { IssueListBlock } from "components/issues/board-views/kanban/block";
// interfaces // interfaces
import { IIssueState, IIssue } from "store/types/issue"; import { IIssueState, IIssue } from "types/issue";
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";

View File

@ -10,24 +10,24 @@ import { IssueBlockDueDate } from "components/issues/board-views/block-due-date"
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// interfaces // interfaces
import { IIssue } from "store/types/issue"; import { IIssue } from "types/issue";
import { RootStore } from "store/root"; import { RootStore } from "store/root";
export const IssueListBlock = observer(({ issue }: { issue: IIssue }) => { export const IssueListBlock = observer(({ issue }: { issue: IIssue }) => {
const store: RootStore = useMobxStore(); const { issue: issueStore, project: projectStore }: RootStore = useMobxStore();
return ( return (
<div className="flex items-center px-9 py-3.5 relative gap-10 border-b border-custom-border-200 bg-custom-background-100 last:border-b-0"> <div className="flex items-center px-9 py-3.5 relative gap-10 border-b border-custom-border-200 bg-custom-background-100 last:border-b-0">
<div className="relative flex items-center gap-6 w-full flex-grow overflow-hidden"> <div className="relative flex items-center gap-6 w-full flex-grow overflow-hidden">
{/* id */} {/* id */}
<div className="flex-shrink-0 text-sm w-[60px] text-custom-text-200"> <div className="flex-shrink-0 text-sm w-auto text-custom-text-200">
{store?.project?.project?.identifier}-{issue?.sequence_id} {projectStore?.project?.identifier}-{issue?.sequence_id}
</div> </div>
{/* name */} {/* name */}
<div className="h-full line-clamp-1 w-full overflow-ellipsis cursor-pointer"> <div className="h-full line-clamp-1 w-full overflow-ellipsis cursor-pointer">
<p <p
onClick={() => { onClick={() => {
store.issue.setActivePeekOverviewIssueId(issue.id); issueStore.setPeekId(issue.id);
}} }}
className="text-[0.825rem] font-medium text-sm truncate text-custom-text-100" className="text-[0.825rem] font-medium text-sm truncate text-custom-text-100"
> >

View File

@ -3,7 +3,7 @@
// mobx react lite // mobx react lite
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// interfaces // interfaces
import { IIssueState } from "store/types/issue"; import { IIssueState } from "types/issue";
// constants // constants
import { issueGroupFilter } from "constants/data"; import { issueGroupFilter } from "constants/data";
// mobx hook // mobx hook
@ -23,7 +23,7 @@ export const IssueListHeader = observer(({ state }: { state: IIssueState }) => {
<stateGroup.icon /> <stateGroup.icon />
</div> </div>
<div className="font-semibold leading-6 text-base capitalize ml-2 mr-3">{state?.name}</div> <div className="font-semibold leading-6 text-base capitalize ml-2 mr-3">{state?.name}</div>
<div className="text-gray-700 font-normal text-base">{store.issue.getCountOfIssuesByState(state.id)}</div> {/* <div className="text-gray-700 font-normal text-base">{store.issue.getCountOfIssuesByState(state.id)}</div> */}
</div> </div>
); );
}); });

View File

@ -6,25 +6,25 @@ import { observer } from "mobx-react-lite";
import { IssueListHeader } from "components/issues/board-views/list/header"; import { IssueListHeader } from "components/issues/board-views/list/header";
import { IssueListBlock } from "components/issues/board-views/list/block"; import { IssueListBlock } from "components/issues/board-views/list/block";
// interfaces // interfaces
import { IIssueState, IIssue } from "store/types/issue"; import { IIssueState, IIssue } from "types/issue";
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";
export const IssueListView = observer(() => { export const IssueListView = observer(() => {
const store: RootStore = useMobxStore(); const { issue: issueStore }: RootStore = useMobxStore();
console.log("issueStore", issueStore.states);
return ( return (
<> <>
{store?.issue?.states && {issueStore?.states &&
store?.issue?.states.length > 0 && issueStore?.states.length > 0 &&
store?.issue?.states.map((_state: IIssueState) => ( issueStore?.states.map((_state: IIssueState) => (
<div key={_state.id} className="relative w-full"> <div key={_state.id} className="relative w-full">
<IssueListHeader state={_state} /> <IssueListHeader state={_state} />
{store.issue.getFilteredIssuesByState(_state.id) && {issueStore.getFilteredIssuesByState(_state.id) &&
store.issue.getFilteredIssuesByState(_state.id).length > 0 ? ( issueStore.getFilteredIssuesByState(_state.id).length > 0 ? (
<div className="divide-y"> <div className="divide-y">
{store.issue.getFilteredIssuesByState(_state.id).map((_issue: IIssue) => ( {issueStore.getFilteredIssuesByState(_state.id).map((_issue: IIssue) => (
<IssueListBlock key={_issue.id} issue={_issue} /> <IssueListBlock key={_issue.id} issue={_issue} />
))} ))}
</div> </div>

View File

@ -4,7 +4,7 @@ import { observer } from "mobx-react-lite";
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// interfaces // interfaces
import { IIssueLabel } from "store/types/issue"; import { IIssueLabel } from "types/issue";
export const RenderIssueLabel = observer(({ label }: { label: IIssueLabel }) => { export const RenderIssueLabel = observer(({ label }: { label: IIssueLabel }) => {
const store = useMobxStore(); const store = useMobxStore();

View File

@ -4,7 +4,7 @@ import { observer } from "mobx-react-lite";
// components // components
import { RenderIssueLabel } from "./filter-label-block"; import { RenderIssueLabel } from "./filter-label-block";
// interfaces // interfaces
import { IIssueLabel } from "store/types/issue"; import { IIssueLabel } from "types/issue";
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";

View File

@ -4,7 +4,7 @@ import { observer } from "mobx-react-lite";
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// interfaces // interfaces
import { IIssuePriorityFilters } from "store/types/issue"; import { IIssuePriorityFilters } from "types/issue";
export const RenderIssuePriority = observer(({ priority }: { priority: IIssuePriorityFilters }) => { export const RenderIssuePriority = observer(({ priority }: { priority: IIssuePriorityFilters }) => {
const store = useMobxStore(); const store = useMobxStore();

View File

@ -6,7 +6,7 @@ import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { RenderIssuePriority } from "./filter-priority-block"; import { RenderIssuePriority } from "./filter-priority-block";
// interfaces // interfaces
import { IIssuePriorityFilters } from "store/types/issue"; import { IIssuePriorityFilters } from "types/issue";
// constants // constants
import { issuePriorityFilters } from "constants/data"; import { issuePriorityFilters } from "constants/data";

View File

@ -4,7 +4,7 @@ import { observer } from "mobx-react-lite";
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// interfaces // interfaces
import { IIssueState } from "store/types/issue"; import { IIssueState } from "types/issue";
// constants // constants
import { issueGroupFilter } from "constants/data"; import { issueGroupFilter } from "constants/data";

View File

@ -4,7 +4,7 @@ import { observer } from "mobx-react-lite";
// components // components
import { RenderIssueState } from "./filter-state-block"; import { RenderIssueState } from "./filter-state-block";
// interfaces // interfaces
import { IIssueState } from "store/types/issue"; import { IIssueState } from "types/issue";
// mobx hook // mobx hook
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";

View File

@ -39,9 +39,11 @@ const IssueNavbar = observer(() => {
useEffect(() => { useEffect(() => {
if (workspace_slug && projectStore) { if (workspace_slug && projectStore) {
if (board) { if (board) {
console.log("setting");
projectStore.setActiveBoard(board.toString()); projectStore.setActiveBoard(board.toString());
} else { } else {
router.push(`/${workspace_slug}/${project_slug}?board=list`); router.push(`/${workspace_slug}/${project_slug}?board=list`);
projectStore.setActiveBoard("list");
} }
} }
}, [board, router, projectStore, workspace_slug, project_slug]); }, [board, router, projectStore, workspace_slug, project_slug]);

View File

@ -3,7 +3,7 @@ import { observer } from "mobx-react-lite";
// constants // constants
import { issueViews } from "constants/data"; import { issueViews } from "constants/data";
// interfaces // interfaces
import { TIssueBoardKeys } from "store/types"; import { TIssueBoardKeys } from "types";
// mobx // mobx
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";
@ -16,18 +16,18 @@ export const NavbarIssueBoardView = observer(() => {
const handleCurrentBoardView = (boardView: string) => { const handleCurrentBoardView = (boardView: string) => {
projectStore.setActiveBoard(boardView); projectStore.setActiveBoard(boardView);
router.replace( router.push(
`/${workspace_slug}/${project_slug}?board=${boardView}${ `/${workspace_slug}/${project_slug}?board=${boardView}${
issueStore?.userSelectedLabels && issueStore?.userSelectedLabels.length > 0 issueStore?.filteredLabels && issueStore?.filteredLabels.length > 0
? `&labels=${issueStore?.userSelectedLabels.join(",")}` ? `&labels=${issueStore?.filteredLabels.join(",")}`
: "" : ""
}${ }${
issueStore?.userSelectedPriorities && issueStore?.userSelectedPriorities.length > 0 issueStore?.filteredPriorities && issueStore?.filteredPriorities.length > 0
? `&priorities=${issueStore?.userSelectedPriorities.join(",")}` ? `&priorities=${issueStore?.filteredPriorities.join(",")}`
: "" : ""
}${ }${
issueStore?.userSelectedStates && issueStore?.userSelectedStates.length > 0 issueStore?.filteredStates && issueStore?.filteredStates.length > 0
? `&states=${issueStore?.userSelectedStates.join(",")}` ? `&states=${issueStore?.filteredStates.join(",")}`
: "" : ""
}` }`
); );
@ -37,8 +37,6 @@ export const NavbarIssueBoardView = observer(() => {
<> <>
{projectStore?.viewOptions && {projectStore?.viewOptions &&
Object.keys(projectStore?.viewOptions).map((viewKey: string) => { Object.keys(projectStore?.viewOptions).map((viewKey: string) => {
console.log("projectStore?.activeBoard", projectStore?.activeBoard);
console.log("viewKey", viewKey);
if (projectStore?.viewOptions[viewKey]) { if (projectStore?.viewOptions[viewKey]) {
return ( return (
<div <div

View File

@ -19,27 +19,25 @@ export const NavbarIssueFilter = observer(() => {
const pathName = router.asPath; const pathName = router.asPath;
const handleOnSelect = (key: "states" | "labels" | "priorities", value: string) => { const handleOnSelect = (key: "states" | "labels" | "priorities", value: string) => {
if (key === "states") { // if (key === "states") {
store.issue.userSelectedStates = store.issue.userSelectedStates.includes(value) // store.issue.userSelectedStates = store.issue.userSelectedStates.includes(value)
? store.issue.userSelectedStates.filter((s) => s !== value) // ? store.issue.userSelectedStates.filter((s) => s !== value)
: [...store.issue.userSelectedStates, value]; // : [...store.issue.userSelectedStates, value];
} else if (key === "labels") { // } else if (key === "labels") {
store.issue.userSelectedLabels = store.issue.userSelectedLabels.includes(value) // store.issue.userSelectedLabels = store.issue.userSelectedLabels.includes(value)
? store.issue.userSelectedLabels.filter((l) => l !== value) // ? store.issue.userSelectedLabels.filter((l) => l !== value)
: [...store.issue.userSelectedLabels, value]; // : [...store.issue.userSelectedLabels, value];
} else if (key === "priorities") { // } else if (key === "priorities") {
store.issue.userSelectedPriorities = store.issue.userSelectedPriorities.includes(value) // store.issue.userSelectedPriorities = store.issue.userSelectedPriorities.includes(value)
? store.issue.userSelectedPriorities.filter((p) => p !== value) // ? store.issue.userSelectedPriorities.filter((p) => p !== value)
: [...store.issue.userSelectedPriorities, value]; // : [...store.issue.userSelectedPriorities, value];
} // }
// const paramsCommaSeparated = `${`board=${store.issue.currentIssueBoardView || "list"}`}${
const paramsCommaSeparated = `${`board=${store.issue.currentIssueBoardView || "list"}`}${ // store.issue.userSelectedPriorities.length > 0 ? `&priorities=${store.issue.userSelectedPriorities.join(",")}` : ""
store.issue.userSelectedPriorities.length > 0 ? `&priorities=${store.issue.userSelectedPriorities.join(",")}` : "" // }${store.issue.userSelectedStates.length > 0 ? `&states=${store.issue.userSelectedStates.join(",")}` : ""}${
}${store.issue.userSelectedStates.length > 0 ? `&states=${store.issue.userSelectedStates.join(",")}` : ""}${ // store.issue.userSelectedLabels.length > 0 ? `&labels=${store.issue.userSelectedLabels.join(",")}` : ""
store.issue.userSelectedLabels.length > 0 ? `&labels=${store.issue.userSelectedLabels.join(",")}` : "" // }`;
}`; // router.replace(`${pathName}?${paramsCommaSeparated}`);
router.replace(`${pathName}?${paramsCommaSeparated}`);
}; };
return ( return (
@ -69,7 +67,7 @@ export const NavbarIssueFilter = observer(() => {
</span> </span>
), ),
onClick: () => handleOnSelect("priorities", priority), onClick: () => handleOnSelect("priorities", priority),
isSelected: store.issue.userSelectedPriorities.includes(priority), isSelected: store.issue.filteredPriorities.includes(priority),
})), })),
}, },
{ {
@ -85,7 +83,7 @@ export const NavbarIssueFilter = observer(() => {
</span> </span>
), ),
onClick: () => handleOnSelect("states", state.id), onClick: () => handleOnSelect("states", state.id),
isSelected: store.issue.userSelectedStates.includes(state.id), isSelected: store.issue.filteredStates.includes(state.id),
}; };
}), }),
}, },
@ -104,7 +102,7 @@ export const NavbarIssueFilter = observer(() => {
</span> </span>
), ),
onClick: () => handleOnSelect("labels", label.id), onClick: () => handleOnSelect("labels", label.id),
isSelected: store.issue.userSelectedLabels.includes(label.id), isSelected: store.issue.filteredLabels.includes(label.id),
})), })),
}, },
]} ]}

View File

@ -9,7 +9,7 @@ import useToast from "hooks/use-toast";
// ui // ui
import { SecondaryButton } from "components/ui"; import { SecondaryButton } from "components/ui";
// types // types
import { Comment } from "store/types"; import { Comment } from "types";
// components // components
import Tiptap, { ITiptapRichTextEditor } from "components/tiptap"; import Tiptap, { ITiptapRichTextEditor } from "components/tiptap";

View File

@ -16,7 +16,7 @@ import { ChatBubbleLeftEllipsisIcon, CheckIcon, XMarkIcon, EllipsisVerticalIcon
// helpers // helpers
import { timeAgo } from "helpers/date-time.helper"; import { timeAgo } from "helpers/date-time.helper";
// types // types
import { Comment } from "store/types"; import { Comment } from "types";
import Tiptap, { ITiptapRichTextEditor } from "components/tiptap"; import Tiptap, { ITiptapRichTextEditor } from "components/tiptap";
const TiptapEditor = React.forwardRef<ITiptapRichTextEditor, ITiptapRichTextEditor>((props, ref) => ( const TiptapEditor = React.forwardRef<ITiptapRichTextEditor, ITiptapRichTextEditor>((props, ref) => (

View File

@ -1,7 +1,7 @@
// components // components
import { IssueReactions } from "components/issues/peek-overview"; import { IssueReactions } from "components/issues/peek-overview";
// types // types
import { IIssue } from "store/types"; import { IIssue } from "types";
type Props = { type Props = {
issue: IIssue; issue: IIssue;

View File

@ -10,7 +10,7 @@ import { Icon } from "components/ui";
import { copyTextToClipboard, addSpaceIfCamelCase } from "helpers/string.helper"; import { copyTextToClipboard, addSpaceIfCamelCase } from "helpers/string.helper";
// types // types
import { IIssue } from "store/types"; import { IIssue } from "types";
// constants // constants
import { issueGroupFilter, issuePriorityFilter } from "constants/data"; import { issueGroupFilter, issuePriorityFilter } from "constants/data";

View File

@ -5,7 +5,7 @@ import { Dialog, Transition } from "@headlessui/react";
import { FullScreenPeekView, SidePeekView } from "components/issues/peek-overview"; import { FullScreenPeekView, SidePeekView } from "components/issues/peek-overview";
// types // types
import type { IIssue } from "store/types"; import type { IIssue } from "types";
type Props = { type Props = {
issue: IIssue | null; issue: IIssue | null;
@ -16,12 +16,7 @@ type Props = {
export type TPeekOverviewModes = "side" | "modal" | "full"; export type TPeekOverviewModes = "side" | "modal" | "full";
export const IssuePeekOverview: React.FC<Props> = ({ export const IssuePeekOverview: React.FC<Props> = ({ issue, isOpen, onClose, workspaceSlug }) => {
issue,
isOpen,
onClose,
workspaceSlug,
}) => {
const [peekOverviewMode, setPeekOverviewMode] = useState<TPeekOverviewModes>("side"); const [peekOverviewMode, setPeekOverviewMode] = useState<TPeekOverviewModes>("side");
const handleClose = () => { const handleClose = () => {

View File

@ -1,3 +1,4 @@
import { useEffect } from "react";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
@ -10,15 +11,16 @@ import { IssuePeekOverview } from "components/issues/peek-overview";
// mobx store // mobx store
import { RootStore } from "store/root"; import { RootStore } from "store/root";
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { useEffect } from "react";
export const ProjectDetailsView = () => { export const ProjectDetailsView = observer(() => {
const router = useRouter(); const router = useRouter();
const { workspace_slug, project_slug, states, labels, priorities } = router.query; const { workspace_slug, project_slug, states, labels, priorities } = router.query;
const { issue: issueStore }: RootStore = useMobxStore(); const { issue: issueStore, project: projectStore }: RootStore = useMobxStore();
const activeIssueId = issueStore.activePeekOverviewIssueId; console.log("projectStore?.activeBoard", projectStore?.activeBoard);
const activeIssueId = issueStore.peekId;
useEffect(() => { useEffect(() => {
if (workspace_slug && project_slug) { if (workspace_slug && project_slug) {
@ -33,14 +35,14 @@ export const ProjectDetailsView = () => {
return ( return (
<div className="relative w-full h-full overflow-hidden"> <div className="relative w-full h-full overflow-hidden">
{workspace_slug && ( {/* {workspace_slug && (
<IssuePeekOverview <IssuePeekOverview
isOpen={Boolean(activeIssueId)} isOpen={Boolean(activeIssueId)}
onClose={() => issueStore.setActivePeekOverviewIssueId(null)} onClose={() => issueStore.setActivePeekOverviewIssueId(null)}
issue={issueStore?.issues?.find((_issue) => _issue.id === activeIssueId) || null} issue={issueStore?.issues?.find((_issue) => _issue.id === activeIssueId) || null}
workspaceSlug={workspace_slug.toString()} workspaceSlug={workspace_slug.toString()}
/> />
)} )} */}
{issueStore?.loader && !issueStore.issues ? ( {issueStore?.loader && !issueStore.issues ? (
<div className="text-sm text-center py-10 text-custom-text-100">Loading...</div> <div className="text-sm text-center py-10 text-custom-text-100">Loading...</div>
@ -51,23 +53,23 @@ export const ProjectDetailsView = () => {
Something went wrong. Something went wrong.
</div> </div>
) : ( ) : (
issueStore?.currentIssueBoardView && ( projectStore?.activeBoard && (
<> <>
{issueStore?.currentIssueBoardView === "list" && ( {projectStore?.activeBoard === "list" && (
<div className="relative w-full h-full overflow-y-auto"> <div className="relative w-full h-full overflow-y-auto">
<div className="mx-auto px-4"> <div className="mx-auto px-4">
<IssueListView /> <IssueListView />
</div> </div>
</div> </div>
)} )}
{issueStore?.currentIssueBoardView === "kanban" && ( {projectStore?.activeBoard === "kanban" && (
<div className="relative w-full h-full mx-auto px-9 py-5"> <div className="relative w-full h-full mx-auto px-9 py-5">
<IssueKanbanView /> <IssueKanbanView />
</div> </div>
)} )}
{issueStore?.currentIssueBoardView === "calendar" && <IssueCalendarView />} {projectStore?.activeBoard === "calendar" && <IssueCalendarView />}
{issueStore?.currentIssueBoardView === "spreadsheet" && <IssueSpreadsheetView />} {projectStore?.activeBoard === "spreadsheet" && <IssueSpreadsheetView />}
{issueStore?.currentIssueBoardView === "gantt" && <IssueGanttView />} {projectStore?.activeBoard === "gantt" && <IssueGanttView />}
</> </>
) )
)} )}
@ -75,4 +77,4 @@ export const ProjectDetailsView = () => {
)} )}
</div> </div>
); );
}; });

View File

@ -7,7 +7,7 @@ import {
TIssueGroupKey, TIssueGroupKey,
IIssuePriorityFilters, IIssuePriorityFilters,
IIssueGroup, IIssueGroup,
} from "store/types/issue"; } from "types/issue";
// icons // icons
import { import {
BacklogStateIcon, BacklogStateIcon,

View File

@ -3,7 +3,7 @@ import type { GetServerSideProps } from "next";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import Head from "next/head"; import Head from "next/head";
// types // types
import { TIssueBoardKeys } from "store/types"; import { TIssueBoardKeys } from "types";
import ProjectLayout from "layouts/project-layout"; import ProjectLayout from "layouts/project-layout";
import { ProjectDetailsView } from "components/views/project-details"; import { ProjectDetailsView } from "components/views/project-details";

View File

@ -1,204 +1,79 @@
// mobx
import { observable, action, computed, makeObservable, runInAction, reaction } from "mobx"; import { observable, action, computed, makeObservable, runInAction, reaction } from "mobx";
// service // services
import IssueService from "services/issue.service"; import IssueService from "services/issue.service";
// store
import { RootStore } from "./root";
// types // types
import { IssueDetailType, TIssueBoardKeys } from "store/types/issue"; // import { IssueDetailType, TIssueBoardKeys } from "types/issue";
import { IIssue, IIssueState, IIssueLabel } from "./types"; import { IIssue, IIssueState, IIssueLabel } from "types/issue";
export interface IIssueStore { export interface IIssueStore {
currentIssueBoardView: TIssueBoardKeys | null;
loader: boolean; loader: boolean;
error: any | null; error: any;
// issue options
issues: IIssue[] | null;
states: IIssueState[] | null; states: IIssueState[] | null;
labels: IIssueLabel[] | null; labels: IIssueLabel[] | null;
issues: IIssue[] | null; // filtering
issue_detail: IssueDetailType; filteredStates: string[];
userSelectedStates: string[]; filteredLabels: string[];
userSelectedLabels: string[]; filteredPriorities: string[];
userSelectedPriorities: string[]; // peek info
activePeekOverviewIssueId: string | null; peekId: string | null;
getCountOfIssuesByState: (state: string) => number; // service
issueService: any;
// actions
fetchPublicIssues: (workspace_slug: string, project_slug: string, params: any) => void;
setPeekId: (issueId: string) => void;
getFilteredIssuesByState: (state: string) => IIssue[]; getFilteredIssuesByState: (state: string) => IIssue[];
getUserSelectedFilter: (key: "state" | "priority" | "label", value: string) => boolean;
checkIfFilterExistsForKey: (key: "state" | "priority" | "label") => boolean;
clearUserSelectedFilter: (key: "state" | "priority" | "label" | "all") => void;
getIfFiltersIsEmpty: () => boolean;
getURLDefinition: (
workspaceSlug: string,
projectId: string,
action?: {
key: "state" | "priority" | "label" | "all";
value?: string;
removeAll?: boolean;
}
) => string;
setActivePeekOverviewIssueId: (value: any) => void;
setCurrentIssueBoardView: (view: TIssueBoardKeys) => void;
fetchPublicIssues: (workspaceSlug: string, projectId: string, params: any) => Promise<void>;
getIssueByIdAsync: (workspaceSlug: string, projectId: string, issueId: string) => Promise<IssueDetailType>;
} }
class IssueStore { class IssueStore implements IIssueStore {
currentIssueBoardView: TIssueBoardKeys | null = null;
loader: boolean = false; loader: boolean = false;
error: any | null = null; error: any | null = null;
states: IIssueState[] | null = null; states: IIssueState[] | null = [];
labels: IIssueLabel[] | null = null; labels: IIssueLabel[] | null = [];
issues: IIssue[] | null = null;
issue_detail: IssueDetailType = {}; filteredStates: string[] = [];
filteredLabels: string[] = [];
filteredPriorities: string[] = [];
activePeekOverviewIssueId: string | null = null; issues: IIssue[] | null = [];
issue_detail: any = {};
userSelectedStates: string[] = []; peekId: string | null = null;
userSelectedLabels: string[] = [];
userSelectedPriorities: string[] = []; rootStore: RootStore;
// root store issueService: any;
rootStore;
// service
issueService;
constructor(_rootStore: any) { constructor(_rootStore: any) {
makeObservable(this, { makeObservable(this, {
// observable // observable
currentIssueBoardView: observable,
loader: observable, loader: observable,
error: observable, error: observable,
// issue options
states: observable.ref, states: observable.ref,
labels: observable.ref, labels: observable.ref,
// filtering
filteredStates: observable.ref,
filteredLabels: observable.ref,
filteredPriorities: observable.ref,
// issues
issues: observable.ref, issues: observable.ref,
issue_detail: observable.ref, issue_detail: observable.ref,
// peek
activePeekOverviewIssueId: observable.ref, peekId: observable.ref,
// actions
userSelectedStates: observable.ref,
userSelectedLabels: observable.ref,
userSelectedPriorities: observable.ref,
// action
setCurrentIssueBoardView: action,
fetchPublicIssues: action, fetchPublicIssues: action,
// computed setPeekId: action,
getFilteredIssuesByState: action,
}); });
this.rootStore = _rootStore; this.rootStore = _rootStore;
this.issueService = new IssueService(); this.issueService = new IssueService();
} }
// computed
getCountOfIssuesByState(state_id: string): number {
return this.issues?.filter((issue) => issue.state == state_id).length || 0;
}
getFilteredIssuesByState(state_id: string): IIssue[] | [] {
return this.issues?.filter((issue) => issue.state == state_id) || [];
}
setActivePeekOverviewIssueId = (issueId: string | null) => (this.activePeekOverviewIssueId = issueId);
/**
*
* @param key Is the key of the filter, i.e. state, label, priority
* @param value Is the value of the filter, i.e. state_id, label_id, priority
* @returns boolean
*/
getUserSelectedFilter(key: "state" | "priority" | "label", value: string): boolean {
if (key == "state") {
return this.userSelectedStates.includes(value);
} else if (key == "label") {
return this.userSelectedLabels.includes(value);
} else if (key == "priority") {
return this.userSelectedPriorities.includes(value);
} else {
return false;
}
}
checkIfFilterExistsForKey: (key: "state" | "priority" | "label") => boolean = (key) => {
if (key == "state") {
return this.userSelectedStates.length > 0;
} else if (key == "label") {
return this.userSelectedLabels.length > 0;
} else if (key == "priority") {
return this.userSelectedPriorities.length > 0;
} else {
return false;
}
};
clearUserSelectedFilter(key: "state" | "priority" | "label" | "all") {
if (key == "state") {
this.userSelectedStates = [];
} else if (key == "label") {
this.userSelectedLabels = [];
} else if (key == "priority") {
this.userSelectedPriorities = [];
} else if (key == "all") {
this.userSelectedStates = [];
this.userSelectedLabels = [];
this.userSelectedPriorities = [];
}
}
getIfFiltersIsEmpty: () => boolean = () =>
this.userSelectedStates.length === 0 &&
this.userSelectedLabels.length === 0 &&
this.userSelectedPriorities.length === 0;
getURLDefinition = (
workspaceSlug: string,
projectId: string,
action?: {
key: "state" | "priority" | "label" | "all";
value?: string;
removeAll?: boolean;
}
) => {
let url = `/${workspaceSlug}/${projectId}?board=${this.currentIssueBoardView}`;
if (action) {
if (action.key === "state")
this.userSelectedStates = action.removeAll
? []
: [...this.userSelectedStates].filter((state) => state !== action.value);
if (action.key === "label")
this.userSelectedLabels = action.removeAll
? []
: [...this.userSelectedLabels].filter((label) => label !== action.value);
if (action.key === "priority")
this.userSelectedPriorities = action.removeAll
? []
: [...this.userSelectedPriorities].filter((priority) => priority !== action.value);
if (action.key === "all") {
this.userSelectedStates = [];
this.userSelectedLabels = [];
this.userSelectedPriorities = [];
}
}
if (this.checkIfFilterExistsForKey("state")) {
url += `&states=${this.userSelectedStates.join(",")}`;
}
if (this.checkIfFilterExistsForKey("label")) {
url += `&labels=${this.userSelectedLabels.join(",")}`;
}
if (this.checkIfFilterExistsForKey("priority")) {
url += `&priorities=${this.userSelectedPriorities.join(",")}`;
}
return url;
};
// action
setCurrentIssueBoardView = async (view: TIssueBoardKeys) => {
this.currentIssueBoardView = view;
};
fetchPublicIssues = async (workspaceSlug: string, projectId: string, params: any) => { fetchPublicIssues = async (workspaceSlug: string, projectId: string, params: any) => {
try { try {
this.loader = true; this.loader = true;
@ -216,352 +91,19 @@ class IssueStore {
this.issues = _issues; this.issues = _issues;
this.loader = false; this.loader = false;
}); });
return response;
} }
} catch (error) { } catch (error) {
this.loader = false; this.loader = false;
this.error = error; this.error = error;
return error;
} }
}; };
getIssueByIdAsync = async (workspaceSlug: string, projectId: string, issueId: string): Promise<IssueDetailType> => { setPeekId = (issueId: string) => {
try { this.peekId = issueId;
const response = this.issues?.find((issue) => issue.id === issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
issue: response,
comments: [],
reactions: [],
votes: [],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
this.getIssueReactionsAsync(workspaceSlug, projectId, issueId);
this.getIssueVotesAsync(workspaceSlug, projectId, issueId);
this.getIssueCommentsAsync(workspaceSlug, projectId, issueId);
}
return this.issue_detail[issueId] as any;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
}; };
getIssueVotesAsync = async (workspaceSlug: string, projectId: string, issueId: string) => { getFilteredIssuesByState = (state_id: string): IIssue[] | [] =>
try { this.issues?.filter((issue) => issue.state == state_id) || [];
const response = await this.issueService.getIssueVotes(workspaceSlug, projectId, issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
votes: response,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
createIssueVoteAsync = async (
workspaceSlug: string,
projectId: string,
issueId: string,
data: {
vote: 1 | -1;
}
) => {
try {
const response = await this.issueService.createIssueVote(workspaceSlug, projectId, issueId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
votes: [
...{ ...this.issue_detail }[issueId].votes.filter(
(vote) => vote.actor !== this.rootStore?.user?.currentUser?.id
),
response,
],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
deleteIssueVoteAsync = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const _votes = (this.issue_detail[issueId].votes = this.issue_detail[issueId].votes.filter(
(vote) => vote.actor !== this.rootStore?.user?.user?.id
));
runInAction(() => {
this.issue_detail[issueId].votes = _votes;
});
const response = await this.issueService.deleteIssueVote(workspaceSlug, projectId, issueId);
const votesAfterCall = await this.issueService.getIssueVotes(workspaceSlug, projectId, issueId);
if (votesAfterCall)
runInAction(() => {
this.issue_detail[issueId].votes = votesAfterCall;
});
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
getIssueReactionsAsync = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const response = await this.issueService.getIssueReactions(workspaceSlug, projectId, issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: response,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
createIssueReactionAsync = async (workspaceSlug: string, projectId: string, issueId: string, data: any) => {
try {
const response = await this.issueService.createIssueReaction(workspaceSlug, projectId, issueId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: [...this.issue_detail[issueId].reactions, response],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
deleteIssueReactionAsync = async (workspaceSlug: string, projectId: string, issueId: string, reactionHex: string) => {
try {
const newReactionsList = this.issue_detail[issueId].reactions.filter(
(reaction) => reaction.reaction !== reactionHex
);
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: newReactionsList,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
const response = await this.issueService.deleteIssueReaction(workspaceSlug, projectId, issueId, reactionHex);
const reactionsAfterCall = await this.issueService.getIssueReactions(workspaceSlug, projectId, issueId);
if (reactionsAfterCall) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: reactionsAfterCall,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
getIssueCommentsAsync = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const response = await this.issueService.getIssueComments(workspaceSlug, projectId, issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: response,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
createIssueCommentAsync = async (workspaceSlug: string, projectId: string, issueId: string, data: any) => {
try {
const response = await this.issueService.createIssueComment(workspaceSlug, projectId, issueId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: [...this.issue_detail[issueId].comments, response],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
updateIssueCommentAsync = async (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
data: any
) => {
try {
const response = await this.issueService.updateIssueComment(workspaceSlug, projectId, issueId, commentId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: [
...this.issue_detail[issueId].comments.filter((comment) => comment.id !== response.id),
response,
],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
deleteIssueCommentAsync = async (workspaceSlug: string, projectId: string, issueId: string, commentId: string) => {
try {
const newCommentsList = this.issue_detail[issueId].comments.filter((comment) => comment.id !== commentId);
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: newCommentsList,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
const response = await this.issueService.deleteIssueComment(workspaceSlug, projectId, issueId, commentId);
const commentsAfterCall = await this.issueService.getIssueComments(workspaceSlug, projectId, issueId);
if (commentsAfterCall) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: commentsAfterCall,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
} }
export default IssueStore; export default IssueStore;

View File

@ -0,0 +1,567 @@
// mobx
import { observable, action, computed, makeObservable, runInAction, reaction } from "mobx";
// service
import IssueService from "services/issue.service";
// types
import { IssueDetailType, TIssueBoardKeys } from "types/issue";
import { IIssue, IIssueState, IIssueLabel } from "../types";
export interface IIssueStore {
currentIssueBoardView: TIssueBoardKeys | null;
loader: boolean;
error: any | null;
states: IIssueState[] | null;
labels: IIssueLabel[] | null;
issues: IIssue[] | null;
issue_detail: IssueDetailType;
userSelectedStates: string[];
userSelectedLabels: string[];
userSelectedPriorities: string[];
activePeekOverviewIssueId: string | null;
getCountOfIssuesByState: (state: string) => number;
getFilteredIssuesByState: (state: string) => IIssue[];
getUserSelectedFilter: (key: "state" | "priority" | "label", value: string) => boolean;
checkIfFilterExistsForKey: (key: "state" | "priority" | "label") => boolean;
clearUserSelectedFilter: (key: "state" | "priority" | "label" | "all") => void;
getIfFiltersIsEmpty: () => boolean;
getURLDefinition: (
workspaceSlug: string,
projectId: string,
action?: {
key: "state" | "priority" | "label" | "all";
value?: string;
removeAll?: boolean;
}
) => string;
setActivePeekOverviewIssueId: (value: any) => void;
setCurrentIssueBoardView: (view: TIssueBoardKeys) => void;
fetchPublicIssues: (workspaceSlug: string, projectId: string, params: any) => Promise<void>;
getIssueByIdAsync: (workspaceSlug: string, projectId: string, issueId: string) => Promise<IssueDetailType>;
}
class IssueLegacyStore {
currentIssueBoardView: TIssueBoardKeys | null = null;
loader: boolean = false;
error: any | null = null;
states: IIssueState[] | null = null;
labels: IIssueLabel[] | null = null;
issues: IIssue[] | null = null;
issue_detail: IssueDetailType = {};
activePeekOverviewIssueId: string | null = null;
userSelectedStates: string[] = [];
userSelectedLabels: string[] = [];
userSelectedPriorities: string[] = [];
// root store
rootStore;
// service
issueService;
constructor(_rootStore: any) {
makeObservable(this, {
// observable
currentIssueBoardView: observable,
loader: observable,
error: observable,
states: observable.ref,
labels: observable.ref,
issues: observable.ref,
issue_detail: observable.ref,
activePeekOverviewIssueId: observable.ref,
userSelectedStates: observable.ref,
userSelectedLabels: observable.ref,
userSelectedPriorities: observable.ref,
// action
setCurrentIssueBoardView: action,
fetchPublicIssues: action,
// computed
});
this.rootStore = _rootStore;
this.issueService = new IssueService();
}
// computed
getCountOfIssuesByState(state_id: string): number {
return this.issues?.filter((issue) => issue.state == state_id).length || 0;
}
getFilteredIssuesByState(state_id: string): IIssue[] | [] {
return this.issues?.filter((issue) => issue.state == state_id) || [];
}
setActivePeekOverviewIssueId = (issueId: string | null) => (this.activePeekOverviewIssueId = issueId);
/**
*
* @param key Is the key of the filter, i.e. state, label, priority
* @param value Is the value of the filter, i.e. state_id, label_id, priority
* @returns boolean
*/
getUserSelectedFilter(key: "state" | "priority" | "label", value: string): boolean {
if (key == "state") {
return this.userSelectedStates.includes(value);
} else if (key == "label") {
return this.userSelectedLabels.includes(value);
} else if (key == "priority") {
return this.userSelectedPriorities.includes(value);
} else {
return false;
}
}
checkIfFilterExistsForKey: (key: "state" | "priority" | "label") => boolean = (key) => {
if (key == "state") {
return this.userSelectedStates.length > 0;
} else if (key == "label") {
return this.userSelectedLabels.length > 0;
} else if (key == "priority") {
return this.userSelectedPriorities.length > 0;
} else {
return false;
}
};
clearUserSelectedFilter(key: "state" | "priority" | "label" | "all") {
if (key == "state") {
this.userSelectedStates = [];
} else if (key == "label") {
this.userSelectedLabels = [];
} else if (key == "priority") {
this.userSelectedPriorities = [];
} else if (key == "all") {
this.userSelectedStates = [];
this.userSelectedLabels = [];
this.userSelectedPriorities = [];
}
}
getIfFiltersIsEmpty: () => boolean = () =>
this.userSelectedStates.length === 0 &&
this.userSelectedLabels.length === 0 &&
this.userSelectedPriorities.length === 0;
getURLDefinition = (
workspaceSlug: string,
projectId: string,
action?: {
key: "state" | "priority" | "label" | "all";
value?: string;
removeAll?: boolean;
}
) => {
let url = `/${workspaceSlug}/${projectId}?board=${this.currentIssueBoardView}`;
if (action) {
if (action.key === "state")
this.userSelectedStates = action.removeAll
? []
: [...this.userSelectedStates].filter((state) => state !== action.value);
if (action.key === "label")
this.userSelectedLabels = action.removeAll
? []
: [...this.userSelectedLabels].filter((label) => label !== action.value);
if (action.key === "priority")
this.userSelectedPriorities = action.removeAll
? []
: [...this.userSelectedPriorities].filter((priority) => priority !== action.value);
if (action.key === "all") {
this.userSelectedStates = [];
this.userSelectedLabels = [];
this.userSelectedPriorities = [];
}
}
if (this.checkIfFilterExistsForKey("state")) {
url += `&states=${this.userSelectedStates.join(",")}`;
}
if (this.checkIfFilterExistsForKey("label")) {
url += `&labels=${this.userSelectedLabels.join(",")}`;
}
if (this.checkIfFilterExistsForKey("priority")) {
url += `&priorities=${this.userSelectedPriorities.join(",")}`;
}
return url;
};
// action
setCurrentIssueBoardView = async (view: TIssueBoardKeys) => {
this.currentIssueBoardView = view;
};
fetchPublicIssues = async (workspaceSlug: string, projectId: string, params: any) => {
try {
this.loader = true;
this.error = null;
const response = await this.issueService.getPublicIssues(workspaceSlug, projectId, params);
if (response) {
const _states: IIssueState[] = [...response?.states];
const _labels: IIssueLabel[] = [...response?.labels];
const _issues: IIssue[] = [...response?.issues];
runInAction(() => {
this.states = _states;
this.labels = _labels;
this.issues = _issues;
this.loader = false;
});
return response;
}
} catch (error) {
this.loader = false;
this.error = error;
return error;
}
};
getIssueByIdAsync = async (workspaceSlug: string, projectId: string, issueId: string): Promise<IssueDetailType> => {
try {
const response = this.issues?.find((issue) => issue.id === issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
issue: response,
comments: [],
reactions: [],
votes: [],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
this.getIssueReactionsAsync(workspaceSlug, projectId, issueId);
this.getIssueVotesAsync(workspaceSlug, projectId, issueId);
this.getIssueCommentsAsync(workspaceSlug, projectId, issueId);
}
return this.issue_detail[issueId] as any;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
getIssueVotesAsync = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const response = await this.issueService.getIssueVotes(workspaceSlug, projectId, issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
votes: response,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
createIssueVoteAsync = async (
workspaceSlug: string,
projectId: string,
issueId: string,
data: {
vote: 1 | -1;
}
) => {
try {
const response = await this.issueService.createIssueVote(workspaceSlug, projectId, issueId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
votes: [
...{ ...this.issue_detail }[issueId].votes.filter(
(vote) => vote.actor !== this.rootStore?.user?.currentUser?.id
),
response,
],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
deleteIssueVoteAsync = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const _votes = (this.issue_detail[issueId].votes = this.issue_detail[issueId].votes.filter(
(vote) => vote.actor !== this.rootStore?.user?.user?.id
));
runInAction(() => {
this.issue_detail[issueId].votes = _votes;
});
const response = await this.issueService.deleteIssueVote(workspaceSlug, projectId, issueId);
const votesAfterCall = await this.issueService.getIssueVotes(workspaceSlug, projectId, issueId);
if (votesAfterCall)
runInAction(() => {
this.issue_detail[issueId].votes = votesAfterCall;
});
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
getIssueReactionsAsync = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const response = await this.issueService.getIssueReactions(workspaceSlug, projectId, issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: response,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
createIssueReactionAsync = async (workspaceSlug: string, projectId: string, issueId: string, data: any) => {
try {
const response = await this.issueService.createIssueReaction(workspaceSlug, projectId, issueId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: [...this.issue_detail[issueId].reactions, response],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
deleteIssueReactionAsync = async (workspaceSlug: string, projectId: string, issueId: string, reactionHex: string) => {
try {
const newReactionsList = this.issue_detail[issueId].reactions.filter(
(reaction) => reaction.reaction !== reactionHex
);
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: newReactionsList,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
const response = await this.issueService.deleteIssueReaction(workspaceSlug, projectId, issueId, reactionHex);
const reactionsAfterCall = await this.issueService.getIssueReactions(workspaceSlug, projectId, issueId);
if (reactionsAfterCall) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
reactions: reactionsAfterCall,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
getIssueCommentsAsync = async (workspaceSlug: string, projectId: string, issueId: string) => {
try {
const response = await this.issueService.getIssueComments(workspaceSlug, projectId, issueId);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: response,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
createIssueCommentAsync = async (workspaceSlug: string, projectId: string, issueId: string, data: any) => {
try {
const response = await this.issueService.createIssueComment(workspaceSlug, projectId, issueId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: [...this.issue_detail[issueId].comments, response],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
updateIssueCommentAsync = async (
workspaceSlug: string,
projectId: string,
issueId: string,
commentId: string,
data: any
) => {
try {
const response = await this.issueService.updateIssueComment(workspaceSlug, projectId, issueId, commentId, data);
if (response) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: [
...this.issue_detail[issueId].comments.filter((comment) => comment.id !== response.id),
response,
],
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
deleteIssueCommentAsync = async (workspaceSlug: string, projectId: string, issueId: string, commentId: string) => {
try {
const newCommentsList = this.issue_detail[issueId].comments.filter((comment) => comment.id !== commentId);
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: newCommentsList,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
const response = await this.issueService.deleteIssueComment(workspaceSlug, projectId, issueId, commentId);
const commentsAfterCall = await this.issueService.getIssueComments(workspaceSlug, projectId, issueId);
if (commentsAfterCall) {
const _issue_detail = {
...this.issue_detail,
[issueId]: {
...this.issue_detail[issueId],
comments: commentsAfterCall,
},
};
runInAction(() => {
this.issue_detail = _issue_detail;
});
}
return response;
} catch (error) {
this.loader = false;
this.error = error;
throw error;
}
};
}
export default IssueLegacyStore;

View File

@ -3,7 +3,7 @@ import { observable, action, makeObservable, runInAction } from "mobx";
// service // service
import ProjectService from "services/project.service"; import ProjectService from "services/project.service";
// types // types
import { IWorkspace, IProject, IProjectSettings } from "./types"; import { IWorkspace, IProject, IProjectSettings } from "../types";
export interface IProjectStore { export interface IProjectStore {
loader: boolean; loader: boolean;

View File

@ -6,7 +6,7 @@ import ThemeStore from "./theme";
import IssueStore, { IIssueStore } from "./issue"; import IssueStore, { IIssueStore } from "./issue";
import ProjectStore, { IProjectStore } from "./project"; import ProjectStore, { IProjectStore } from "./project";
// types // types
import { IThemeStore } from "./types"; import { IThemeStore } from "../types";
enableStaticRendering(typeof window === "undefined"); enableStaticRendering(typeof window === "undefined");

View File

@ -1,7 +1,7 @@
// mobx // mobx
import { observable, action, computed, makeObservable, runInAction } from "mobx"; import { observable, action, computed, makeObservable, runInAction } from "mobx";
// types // types
import { IThemeStore } from "./types"; import { IThemeStore } from "../types";
class ThemeStore implements IThemeStore { class ThemeStore implements IThemeStore {
theme: "light" | "dark" = "light"; theme: "light" | "dark" = "light";

View File

@ -3,7 +3,7 @@ import { observable, action, computed, makeObservable, runInAction } from "mobx"
// service // service
import UserService from "services/user.service"; import UserService from "services/user.service";
// types // types
import { IUserStore } from "./types"; import { IUserStore } from "../types";
class UserStore implements IUserStore { class UserStore implements IUserStore {
currentUser: any | null = null; currentUser: any | null = null;