forked from github/plane
chore: inbox issue update
This commit is contained in:
parent
aeb2c0383b
commit
42f2d346bf
@ -3,15 +3,17 @@ import Router, { useRouter } from "next/router";
|
||||
import { observer } from "mobx-react-lite";
|
||||
import useSWR from "swr";
|
||||
import { useForm } from "react-hook-form";
|
||||
import { AlertTriangle, CheckCircle2, Clock, Copy, ExternalLink, Inbox, XCircle } from "lucide-react";
|
||||
import { AlertTriangle, CheckCircle2, Clock, Copy, ExternalLink, Inbox, RefreshCw, XCircle } from "lucide-react";
|
||||
|
||||
// mobx store
|
||||
import { useMobxStore } from "lib/mobx/store-provider";
|
||||
// components
|
||||
import { IssueDescriptionForm, IssueDetailsSidebar, IssueReaction } from "components/issues";
|
||||
import { InboxIssueActivity } from "components/inbox";
|
||||
// hooks
|
||||
import useReloadConfirmations from "hooks/use-reload-confirmation";
|
||||
// ui
|
||||
import { Loader } from "@plane/ui";
|
||||
import { Loader, StateGroupIcon } from "@plane/ui";
|
||||
// helpers
|
||||
import { renderShortDateWithYearFormat } from "helpers/date-time.helper";
|
||||
// types
|
||||
@ -30,11 +32,18 @@ export const InboxMainContent: React.FC = observer(() => {
|
||||
const router = useRouter();
|
||||
const { workspaceSlug, projectId, inboxId, inboxIssueId } = router.query;
|
||||
|
||||
const { inboxIssues: inboxIssuesStore, inboxIssueDetails: inboxIssueDetailsStore, user: userStore } = useMobxStore();
|
||||
const {
|
||||
inboxIssues: inboxIssuesStore,
|
||||
inboxIssueDetails: inboxIssueDetailsStore,
|
||||
user: userStore,
|
||||
projectState: { states },
|
||||
} = useMobxStore();
|
||||
|
||||
const user = userStore.currentUser;
|
||||
const userRole = userStore.currentProjectRole;
|
||||
|
||||
const { setShowAlert } = useReloadConfirmations();
|
||||
|
||||
const { reset, control, watch } = useForm<IIssue>({
|
||||
defaultValues,
|
||||
});
|
||||
@ -54,6 +63,9 @@ export const InboxMainContent: React.FC = observer(() => {
|
||||
|
||||
const issuesList = inboxId ? inboxIssuesStore.inboxIssues[inboxId.toString()] : undefined;
|
||||
const issueDetails = inboxIssueId ? inboxIssueDetailsStore.issueDetails[inboxIssueId.toString()] : undefined;
|
||||
const currentIssueState = projectId
|
||||
? states[projectId.toString()]?.find((s) => s.id === issueDetails?.state)
|
||||
: undefined;
|
||||
|
||||
const submitChanges = useCallback(
|
||||
async (formData: Partial<IInboxIssue>) => {
|
||||
@ -124,6 +136,17 @@ export const InboxMainContent: React.FC = observer(() => {
|
||||
});
|
||||
}, [issueDetails, reset, inboxIssueId]);
|
||||
|
||||
useEffect(() => {
|
||||
if (inboxIssueDetailsStore.isSubmitting === "submitted") {
|
||||
setShowAlert(false);
|
||||
setTimeout(async () => {
|
||||
inboxIssueDetailsStore.setIsSubmitting("saved");
|
||||
}, 2000);
|
||||
} else if (inboxIssueDetailsStore.isSubmitting === "submitting") {
|
||||
setShowAlert(true);
|
||||
}
|
||||
}, [inboxIssueDetailsStore.isSubmitting, setShowAlert]);
|
||||
|
||||
const issueStatus = issueDetails?.issue_inbox[0].status;
|
||||
|
||||
if (!inboxIssueId)
|
||||
@ -214,8 +237,36 @@ export const InboxMainContent: React.FC = observer(() => {
|
||||
</>
|
||||
) : null}
|
||||
</div>
|
||||
<div className="flex items-center mb-5">
|
||||
{!currentIssueState ? (
|
||||
<StateGroupIcon className="h-4 w-4 mr-3" stateGroup="backlog" color="#ff7700" />
|
||||
) : (
|
||||
<StateGroupIcon
|
||||
className="h-4 w-4 mr-3"
|
||||
stateGroup={currentIssueState.group}
|
||||
color={currentIssueState.color}
|
||||
/>
|
||||
)}
|
||||
<h4 className="text-lg text-custom-text-300 font-medium mr-4">
|
||||
{issueDetails?.project_detail?.identifier}-{issueDetails?.sequence_id}
|
||||
</h4>
|
||||
<div
|
||||
className={`flex transition-all duration-300 items-center gap-x-2 ${
|
||||
inboxIssueDetailsStore.isSubmitting === "saved" ? "fadeOut" : "fadeIn"
|
||||
}`}
|
||||
>
|
||||
{inboxIssueDetailsStore.isSubmitting !== "submitted" &&
|
||||
inboxIssueDetailsStore.isSubmitting !== "saved" && (
|
||||
<RefreshCw className="h-4 w-4 stroke-custom-text-300" />
|
||||
)}
|
||||
<span className="text-sm text-custom-text-300">
|
||||
{inboxIssueDetailsStore.isSubmitting === "submitting" ? "Saving..." : "Saved"}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
<div>
|
||||
<IssueDescriptionForm
|
||||
setShowAlert={setShowAlert}
|
||||
workspaceSlug={workspaceSlug as string}
|
||||
issue={{
|
||||
name: issueDetails.name,
|
||||
|
@ -11,6 +11,7 @@ import { IIssue } from "types";
|
||||
// services
|
||||
import { FileService } from "services/file.service";
|
||||
import useEditorSuggestions from "hooks/use-editor-suggestions";
|
||||
import { useRouter } from "next/router";
|
||||
|
||||
export interface IssueDescriptionFormValues {
|
||||
name: string;
|
||||
@ -32,13 +33,18 @@ export interface IssueDetailsProps {
|
||||
const fileService = new FileService();
|
||||
|
||||
export const IssueDescriptionForm: FC<IssueDetailsProps> = (props) => {
|
||||
const { issue, handleFormSubmit, workspaceSlug, isAllowed,setShowAlert } = props;
|
||||
const { issue, handleFormSubmit, workspaceSlug, isAllowed, setShowAlert } = props;
|
||||
// states
|
||||
const [characterLimit, setCharacterLimit] = useState(false);
|
||||
|
||||
// router
|
||||
const router = useRouter();
|
||||
const { inboxId } = router.query;
|
||||
|
||||
// mobx store
|
||||
const {
|
||||
projectIssues: { setIsSubmitting },
|
||||
projectIssues: { setIsSubmitting: PIsetIsSubmitting },
|
||||
inboxIssueDetails: { setIsSubmitting: IIsetIsSubmitting },
|
||||
} = useMobxStore();
|
||||
|
||||
const editorSuggestion = useEditorSuggestions();
|
||||
@ -89,6 +95,8 @@ export const IssueDescriptionForm: FC<IssueDetailsProps> = (props) => {
|
||||
handleSubmit(handleDescriptionFormSubmit)();
|
||||
}, 1500);
|
||||
|
||||
const setIsSubmitting = inboxId ? IIsetIsSubmitting : PIsetIsSubmitting;
|
||||
|
||||
return (
|
||||
<div className="relative">
|
||||
<div className="relative">
|
||||
|
@ -87,7 +87,7 @@ export const IssueDetailsSidebar: React.FC<Props> = observer((props) => {
|
||||
const userRole = userStore.currentProjectRole;
|
||||
|
||||
const router = useRouter();
|
||||
const { workspaceSlug, projectId, issueId } = router.query;
|
||||
const { workspaceSlug, projectId, issueId, inboxIssueId } = router.query;
|
||||
|
||||
const { isEstimateActive } = useEstimateOption();
|
||||
|
||||
@ -273,13 +273,15 @@ export const IssueDetailsSidebar: React.FC<Props> = observer((props) => {
|
||||
<div className="h-full w-full flex flex-col divide-y-2 divide-custom-border-200 overflow-hidden">
|
||||
<div className="flex items-center justify-between px-5 pb-3">
|
||||
<div className="flex items-center gap-x-2">
|
||||
{currentIssueState && (
|
||||
{currentIssueState ? (
|
||||
<StateGroupIcon
|
||||
className="h-4 w-4"
|
||||
stateGroup={currentIssueState.group}
|
||||
color={currentIssueState.color}
|
||||
/>
|
||||
)}
|
||||
) : inboxIssueId ? (
|
||||
<StateGroupIcon className="h-4 w-4" stateGroup="backlog" color="#ff7700" />
|
||||
) : null}
|
||||
<h4 className="text-lg text-custom-text-300 font-medium">
|
||||
{issueDetail?.project_detail?.identifier}-{issueDetail?.sequence_id}
|
||||
</h4>
|
||||
|
@ -50,6 +50,7 @@ const IssueDetailsPage: NextPageWithLayout = observer(() => {
|
||||
projectIssues: { isSubmitting, setIsSubmitting },
|
||||
} = useMobxStore();
|
||||
|
||||
// hooks
|
||||
const { setShowAlert } = useReloadConfirmations();
|
||||
|
||||
const {
|
||||
|
@ -5,6 +5,7 @@ import { RootStore } from "../root";
|
||||
import { InboxService } from "services/inbox.service";
|
||||
// types
|
||||
import { IInboxIssue, IIssue, TInboxStatus } from "types";
|
||||
import { TIssueUpdateStatus } from "store/issues/types";
|
||||
// constants
|
||||
import { INBOX_ISSUE_SOURCE } from "constants/inbox";
|
||||
|
||||
@ -12,6 +13,7 @@ export interface IInboxIssueDetailsStore {
|
||||
// states
|
||||
loader: boolean;
|
||||
error: any | null;
|
||||
isSubmitting: TIssueUpdateStatus; // inbox issue update status;
|
||||
|
||||
// observables
|
||||
issueDetails: {
|
||||
@ -46,12 +48,14 @@ export interface IInboxIssueDetailsStore {
|
||||
data: TInboxStatus
|
||||
) => Promise<void>;
|
||||
deleteIssue: (workspaceSlug: string, projectId: string, inboxId: string, issueId: string) => Promise<void>;
|
||||
setIsSubmitting: (status: TIssueUpdateStatus) => void;
|
||||
}
|
||||
|
||||
export class InboxIssueDetailsStore implements IInboxIssueDetailsStore {
|
||||
// states
|
||||
loader: boolean = false;
|
||||
error: any | null = null;
|
||||
isSubmitting: TIssueUpdateStatus = "saved";
|
||||
|
||||
// observables
|
||||
issueDetails: { [issueId: string]: IInboxIssue } = {};
|
||||
@ -67,6 +71,7 @@ export class InboxIssueDetailsStore implements IInboxIssueDetailsStore {
|
||||
// states
|
||||
loader: observable.ref,
|
||||
error: observable.ref,
|
||||
isSubmitting: observable.ref,
|
||||
|
||||
// observables
|
||||
issueDetails: observable.ref,
|
||||
@ -76,12 +81,17 @@ export class InboxIssueDetailsStore implements IInboxIssueDetailsStore {
|
||||
createIssue: action,
|
||||
updateIssueStatus: action,
|
||||
deleteIssue: action,
|
||||
setIsSubmitting: action,
|
||||
});
|
||||
|
||||
this.rootStore = _rootStore;
|
||||
this.inboxService = new InboxService();
|
||||
}
|
||||
|
||||
setIsSubmitting = (status: TIssueUpdateStatus) => {
|
||||
this.isSubmitting = status;
|
||||
};
|
||||
|
||||
fetchIssueDetails = async (workspaceSlug: string, projectId: string, inboxId: string, issueId: string) => {
|
||||
try {
|
||||
runInAction(() => {
|
||||
@ -155,6 +165,7 @@ export class InboxIssueDetailsStore implements IInboxIssueDetailsStore {
|
||||
|
||||
try {
|
||||
runInAction(() => {
|
||||
this.isSubmitting = "submitting";
|
||||
this.issueDetails = {
|
||||
...this.issueDetails,
|
||||
[issueId]: updatedIssue,
|
||||
@ -170,6 +181,7 @@ export class InboxIssueDetailsStore implements IInboxIssueDetailsStore {
|
||||
});
|
||||
|
||||
await this.inboxService.patchInboxIssue(workspaceSlug, projectId, inboxId, issueId, { issue: data });
|
||||
this.isSubmitting = "submitted";
|
||||
} catch (error) {
|
||||
runInAction(() => {
|
||||
this.error = error;
|
||||
|
@ -6,13 +6,21 @@ import { IssueService } from "services/issue/issue.service";
|
||||
// types
|
||||
import { TIssueGroupByOptions } from "types";
|
||||
import { IIssue } from "types/issues";
|
||||
import { IIssueResponse, TLoader, IGroupedIssues, ISubGroupedIssues, TUnGroupedIssues, ViewFlags, TIssueUpdateStatus } from "../../types";
|
||||
import {
|
||||
IIssueResponse,
|
||||
TLoader,
|
||||
IGroupedIssues,
|
||||
ISubGroupedIssues,
|
||||
TUnGroupedIssues,
|
||||
ViewFlags,
|
||||
TIssueUpdateStatus,
|
||||
} from "../../types";
|
||||
import { RootStore } from "store/root";
|
||||
|
||||
export interface IProjectIssuesStore {
|
||||
// observable
|
||||
loader: TLoader;
|
||||
isSubmitting: TIssueUpdateStatus;
|
||||
isSubmitting: TIssueUpdateStatus; // update issue status
|
||||
issues: { [project_id: string]: IIssueResponse } | undefined;
|
||||
// computed
|
||||
getIssues: IIssueResponse | undefined;
|
||||
|
Loading…
Reference in New Issue
Block a user