diff --git a/apps/app/.env.example b/apps/app/.env.example new file mode 100644 index 000000000..aff5c6c31 --- /dev/null +++ b/apps/app/.env.example @@ -0,0 +1,4 @@ +NEXT_PUBLIC_API_BASE_URL = "<-- endpoint goes here -->" +NEXT_PUBLIC_GOOGLE_CLIENTID = "<-- google client id goes here -->" +NEXT_PUBLIC_GITHUB_ID = "<-- github id goes here -->" +NEXT_PUBLIC_APP_ENVIRONMENT=development \ No newline at end of file diff --git a/apps/app/components/command-palette/addAsSubIssue.tsx b/apps/app/components/command-palette/addAsSubIssue.tsx index e8c585e45..aadc04574 100644 --- a/apps/app/components/command-palette/addAsSubIssue.tsx +++ b/apps/app/components/command-palette/addAsSubIssue.tsx @@ -7,7 +7,7 @@ import { useForm } from "react-hook-form"; // headless ui import { Combobox, Dialog, Transition } from "@headlessui/react"; // services -import issuesServices from "lib/services/issues.services"; +import issuesServices from "lib/services/issues.service"; // hooks import useUser from "lib/hooks/useUser"; // icons diff --git a/apps/app/components/command-palette/index.tsx b/apps/app/components/command-palette/index.tsx index 63490c97d..32f71a3db 100644 --- a/apps/app/components/command-palette/index.tsx +++ b/apps/app/components/command-palette/index.tsx @@ -8,7 +8,7 @@ import { SubmitHandler, useForm } from "react-hook-form"; // headless ui import { Combobox, Dialog, Transition } from "@headlessui/react"; // services -import issuesServices from "lib/services/issues.services"; +import issuesServices from "lib/services/issues.service"; // hooks import useUser from "lib/hooks/useUser"; import useTheme from "lib/hooks/useTheme"; @@ -22,7 +22,7 @@ import { } from "@heroicons/react/24/outline"; // components import ShortcutsModal from "components/command-palette/shortcuts"; -import CreateProjectModal from "components/project/CreateProjectModal"; +import CreateProjectModal from "components/project/create-project-modal"; import CreateUpdateIssuesModal from "components/project/issues/CreateUpdateIssueModal"; import CreateUpdateCycleModal from "components/project/cycles/CreateUpdateCyclesModal"; // ui diff --git a/apps/app/components/forms/EmailPasswordForm.tsx b/apps/app/components/forms/EmailPasswordForm.tsx index a73c8b040..384b77598 100644 --- a/apps/app/components/forms/EmailPasswordForm.tsx +++ b/apps/app/components/forms/EmailPasswordForm.tsx @@ -88,9 +88,7 @@ const EmailPasswordForm = ({ onSuccess }: any) => {
- - Forgot your password? - + Forgot your password?
diff --git a/apps/app/components/lexical/editor.tsx b/apps/app/components/lexical/editor.tsx index 510c03e94..71fbb03cf 100644 --- a/apps/app/components/lexical/editor.tsx +++ b/apps/app/components/lexical/editor.tsx @@ -27,7 +27,7 @@ import { getValidatedValue } from "./helpers/editor"; import LexicalErrorBoundary from "@lexical/react/LexicalErrorBoundary"; export interface RichTextEditorProps { - onChange: (state: SerializedEditorState) => void; + onChange: (state: string) => void; id: string; value: string; placeholder?: string; @@ -41,8 +41,7 @@ const RichTextEditor: React.FC = ({ }) => { const handleChange = (editorState: EditorState) => { editorState.read(() => { - let editorData = editorState.toJSON(); - if (onChange) onChange(editorData); + onChange(JSON.stringify(editorState.toJSON())); }); }; diff --git a/apps/app/components/lexical/helpers/editor.ts b/apps/app/components/lexical/helpers/editor.ts index 702269cd5..56f2ffe20 100644 --- a/apps/app/components/lexical/helpers/editor.ts +++ b/apps/app/components/lexical/helpers/editor.ts @@ -18,10 +18,12 @@ export const getValidatedValue = (value: string) => { const defaultValue = '{"root":{"children":[{"children":[],"direction":null,"format":"","indent":0,"type":"paragraph","version":1}],"direction":null,"format":"","indent":0,"type":"root","version":1}}'; + console.log("Value: ", value); + if (value) { try { - console.log(value); - return value; + const data = JSON.parse(value); + return JSON.stringify(data); } catch (e) { return defaultValue; } diff --git a/apps/app/components/project/SendProjectInvitationModal.tsx b/apps/app/components/project/SendProjectInvitationModal.tsx index 0c0485803..ef77b65f9 100644 --- a/apps/app/components/project/SendProjectInvitationModal.tsx +++ b/apps/app/components/project/SendProjectInvitationModal.tsx @@ -20,7 +20,7 @@ import { Button, Select, TextArea } from "ui"; import { ChevronDownIcon, CheckIcon } from "@heroicons/react/20/solid"; // types -import { ProjectMember, WorkspaceMember } from "types"; +import { IProjectMemberInvitation } from "types"; type Props = { isOpen: boolean; @@ -28,6 +28,11 @@ type Props = { members: any[]; }; +type ProjectMember = IProjectMemberInvitation & { + member_id: string; + user_id: string; +}; + const defaultValues: Partial = { email: "", message: "", @@ -49,7 +54,7 @@ const SendProjectInvitationModal: React.FC = ({ isOpen, setIsOpen, member const { setToastAlert } = useToast(); - const { data: people } = useSWR( + const { data: people } = useSWR( activeWorkspace ? WORKSPACE_MEMBERS(activeWorkspace.slug) : null, activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null, { @@ -211,7 +216,7 @@ const SendProjectInvitationModal: React.FC = ({ isOpen, setIsOpen, member {selected ? ( >; - data?: IProject; + onClose: () => void; + data: IProject | null; }; -const ConfirmProjectDeletion: React.FC = ({ isOpen, setIsOpen, data }) => { +const ConfirmProjectDeletion: React.FC = ({ isOpen, data, onClose }) => { const [isDeleteLoading, setIsDeleteLoading] = useState(false); + const [selectedProject, setSelectedProject] = useState(null); + + const [confirmProjectName, setConfirmProjectName] = useState(""); + const [confirmDeleteMyProject, setConfirmDeleteMyProject] = useState(false); + + const canDelete = confirmProjectName === data?.name && confirmDeleteMyProject; + const { activeWorkspace, mutateProjects } = useUser(); const { setToastAlert } = useToast(); @@ -29,13 +36,18 @@ const ConfirmProjectDeletion: React.FC = ({ isOpen, setIsOpen, data }) => const cancelButtonRef = useRef(null); const handleClose = () => { - setIsOpen(false); setIsDeleteLoading(false); + const timer = setTimeout(() => { + setConfirmProjectName(""); + setConfirmDeleteMyProject(false); + clearTimeout(timer); + }, 350); + onClose(); }; const handleDeletion = async () => { setIsDeleteLoading(true); - if (!data || !activeWorkspace) return; + if (!data || !activeWorkspace || !canDelete) return; await projectService .deleteProject(activeWorkspace.slug, data.id) .then(() => { @@ -54,8 +66,14 @@ const ConfirmProjectDeletion: React.FC = ({ isOpen, setIsOpen, data }) => }; useEffect(() => { - data && setIsOpen(true); - }, [data, setIsOpen]); + if (data) setSelectedProject(data); + else { + const timer = setTimeout(() => { + setSelectedProject(null); + clearTimeout(timer); + }, 300); + } + }, [data]); return ( @@ -104,11 +122,48 @@ const ConfirmProjectDeletion: React.FC = ({ isOpen, setIsOpen, data }) =>

Are you sure you want to delete project - {`"`} - {data?.name} + {selectedProject?.name} {`"`} ? All of the data related to the project will be permanently removed. This action cannot be undone.

+
+
+

+ Enter the project name{" "} + {selectedProject?.name} to + continue: +

+ { + setConfirmProjectName(e.target.value); + }} + name="projectName" + /> +
+
+

+ To confirm, type delete my project{" "} + below: +

+ { + if (e.target.value === "delete my project") { + setConfirmDeleteMyProject(true); + } else { + setConfirmDeleteMyProject(false); + } + }} + name="typeDelete" + /> +
@@ -117,7 +172,7 @@ const ConfirmProjectDeletion: React.FC = ({ isOpen, setIsOpen, data }) => type="button" onClick={handleDeletion} theme="danger" - disabled={isDeleteLoading} + disabled={isDeleteLoading || !canDelete} className="inline-flex sm:ml-3" > {isDeleteLoading ? "Deleting..." : "Delete"} diff --git a/apps/app/components/project/CreateProjectModal.tsx b/apps/app/components/project/create-project-modal.tsx similarity index 78% rename from apps/app/components/project/CreateProjectModal.tsx rename to apps/app/components/project/create-project-modal.tsx index c93f9123b..5d4d9edb8 100644 --- a/apps/app/components/project/CreateProjectModal.tsx +++ b/apps/app/components/project/create-project-modal.tsx @@ -1,4 +1,4 @@ -import React, { useState, useEffect, useCallback } from "react"; +import React, { useState, useEffect } from "react"; // swr import useSWR, { mutate } from "swr"; // react hook form @@ -8,6 +8,10 @@ import { Dialog, Transition } from "@headlessui/react"; // services import projectServices from "lib/services/project.service"; import workspaceService from "lib/services/workspace.service"; +// common +import { createSimilarString } from "constants/common"; +// constants +import { NETWORK_CHOICES } from "constants/"; // fetch keys import { PROJECTS_LIST, WORKSPACE_MEMBERS } from "constants/fetch-keys"; // hooks @@ -15,21 +19,19 @@ import useUser from "lib/hooks/useUser"; import useToast from "lib/hooks/useToast"; // ui import { Button, Input, TextArea, Select } from "ui"; -// common -import { debounce } from "constants/common"; // types -import { IProject, WorkspaceMember } from "types"; +import { IProject } from "types"; type Props = { isOpen: boolean; setIsOpen: React.Dispatch>; }; -const NETWORK_CHOICES = { "0": "Secret", "2": "Public" }; - const defaultValues: Partial = { name: "", + identifier: "", description: "", + network: 0, }; const IsGuestCondition: React.FC<{ @@ -60,11 +62,16 @@ const CreateProjectModal: React.FC = ({ isOpen, setIsOpen }) => { const { activeWorkspace, user } = useUser(); - const { data: workspaceMembers } = useSWR( + const { data: workspaceMembers } = useSWR( activeWorkspace ? WORKSPACE_MEMBERS(activeWorkspace.slug) : null, - activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null + activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null, + { + shouldRetryOnError: false, + } ); + const [recommendedIdentifier, setRecommendedIdentifier] = useState([]); + const { setToastAlert } = useToast(); const [isChangeIdentifierRequired, setIsChangeIdentifierRequired] = useState(true); @@ -75,12 +82,11 @@ const CreateProjectModal: React.FC = ({ isOpen, setIsOpen }) => { handleSubmit, reset, setError, + clearErrors, watch, setValue, } = useForm({ defaultValues, - reValidateMode: "onChange", - mode: "all", }); const onSubmit = async (formData: IProject) => { @@ -111,6 +117,7 @@ const CreateProjectModal: React.FC = ({ isOpen, setIsOpen }) => { handleClose(); return; } + err = err.data; Object.keys(err).map((key) => { const errorMessages = err[key]; setError(key as keyof IProject, { @@ -123,22 +130,30 @@ const CreateProjectModal: React.FC = ({ isOpen, setIsOpen }) => { const projectName = watch("name") ?? ""; const projectIdentifier = watch("identifier") ?? ""; - const checkIdentifier = (slug: string, value: string) => { - projectServices.checkProjectIdentifierAvailability(slug, value).then((response) => { - console.log(response); - if (response.exists) setError("identifier", { message: "Identifier already exists" }); - }); - }; - - // eslint-disable-next-line react-hooks/exhaustive-deps - const checkIdentifierAvailability = useCallback(debounce(checkIdentifier, 1500), []); - useEffect(() => { if (projectName && isChangeIdentifierRequired) { - setValue("identifier", projectName.replace(/ /g, "-").toUpperCase().substring(0, 3)); + setValue("identifier", projectName.replace(/ /g, "").toUpperCase().substring(0, 3)); } }, [projectName, projectIdentifier, setValue, isChangeIdentifierRequired]); + useEffect(() => { + if (!projectName) return; + const suggestedIdentifier = createSimilarString( + projectName.replace(/ /g, "").toUpperCase().substring(0, 3) + ); + + setRecommendedIdentifier([ + suggestedIdentifier + Math.floor(Math.random() * 101), + suggestedIdentifier + Math.floor(Math.random() * 101), + projectIdentifier.toUpperCase().substring(0, 3) + Math.floor(Math.random() * 101), + projectIdentifier.toUpperCase().substring(0, 3) + Math.floor(Math.random() * 101), + ]); + }, [errors.identifier, projectIdentifier, projectName]); + + useEffect(() => { + return () => setIsChangeIdentifierRequired(true); + }, [isOpen]); + if (workspaceMembers) { const isMember = workspaceMembers.find((member) => member.member.id === user?.id); const isGuest = workspaceMembers.find( @@ -234,11 +249,7 @@ const CreateProjectModal: React.FC = ({ isOpen, setIsOpen }) => { placeholder="Enter Project Identifier" error={errors.identifier} register={register} - onChange={(e: any) => { - setIsChangeIdentifierRequired(false); - if (!activeWorkspace || !e.target.value) return; - checkIdentifierAvailability(activeWorkspace.slug, e.target.value); - }} + onChange={() => setIsChangeIdentifierRequired(false)} validations={{ required: "Identifier is required", minLength: { @@ -251,6 +262,27 @@ const CreateProjectModal: React.FC = ({ isOpen, setIsOpen }) => { }, }} /> + {errors.identifier && ( +
+

Ops! Identifier is already taken. Try one of the following:

+
+ {recommendedIdentifier.map((identifier) => ( + + ))} +
+
+ )} diff --git a/apps/app/components/project/cycles/BoardView/index.tsx b/apps/app/components/project/cycles/BoardView/index.tsx new file mode 100644 index 000000000..8026786c4 --- /dev/null +++ b/apps/app/components/project/cycles/BoardView/index.tsx @@ -0,0 +1,79 @@ +// components +import SingleBoard from "components/project/cycles/BoardView/single-board"; +// ui +import { Spinner } from "ui"; +// types +import { IIssue, IProjectMember, NestedKeyOf, Properties } from "types"; +import useUser from "lib/hooks/useUser"; + +type Props = { + groupedByIssues: { + [key: string]: IIssue[]; + }; + properties: Properties; + selectedGroup: NestedKeyOf | null; + members: IProjectMember[] | undefined; + openCreateIssueModal: ( + sprintId: string, + issue?: IIssue, + actionType?: "create" | "edit" | "delete" + ) => void; + openIssuesListModal: (cycleId: string) => void; + removeIssueFromCycle: (cycleId: string, bridgeId: string) => void; +}; + +const CyclesBoardView: React.FC = ({ + groupedByIssues, + properties, + selectedGroup, + members, + openCreateIssueModal, + openIssuesListModal, + removeIssueFromCycle, +}) => { + const { states } = useUser(); + + return ( + <> + {groupedByIssues ? ( +
+
+
+
+ {Object.keys(groupedByIssues).map((singleGroup) => ( + m.member.id === singleGroup)?.member.first_name ?? + "loading..." + : null + } + groupedByIssues={groupedByIssues} + bgColor={ + selectedGroup === "state_detail.name" + ? states?.find((s) => s.name === singleGroup)?.color + : undefined + } + properties={properties} + removeIssueFromCycle={removeIssueFromCycle} + openIssuesListModal={openIssuesListModal} + openCreateIssueModal={openCreateIssueModal} + /> + ))} +
+
+
+
+ ) : ( +
+ +
+ )} + + ); +}; + +export default CyclesBoardView; diff --git a/apps/app/components/project/cycles/BoardView/single-board.tsx b/apps/app/components/project/cycles/BoardView/single-board.tsx new file mode 100644 index 000000000..92ff37110 --- /dev/null +++ b/apps/app/components/project/cycles/BoardView/single-board.tsx @@ -0,0 +1,662 @@ +// react +import React, { useState } from "react"; +// next +import Link from "next/link"; +import Image from "next/image"; +// swr +import useSWR from "swr"; +// services +import cycleServices from "lib/services/cycles.service"; +// hooks +import useUser from "lib/hooks/useUser"; +// ui +import { Spinner } from "ui"; +// icons +import { + ArrowsPointingInIcon, + ArrowsPointingOutIcon, + CalendarDaysIcon, + PlusIcon, + EllipsisHorizontalIcon, + TrashIcon, +} from "@heroicons/react/24/outline"; +import User from "public/user.png"; +// types +import { + CycleIssueResponse, + ICycle, + IIssue, + IWorkspaceMember, + NestedKeyOf, + Properties, +} from "types"; +// constants +import { CYCLE_ISSUES, WORKSPACE_MEMBERS } from "constants/fetch-keys"; +import { + addSpaceIfCamelCase, + findHowManyDaysLeft, + renderShortNumericDateFormat, +} from "constants/common"; +import { Menu, Transition } from "@headlessui/react"; +import workspaceService from "lib/services/workspace.service"; + +type Props = { + properties: Properties; + groupedByIssues: { + [key: string]: IIssue[]; + }; + selectedGroup: NestedKeyOf | null; + groupTitle: string; + createdBy: string | null; + bgColor?: string; + openCreateIssueModal: ( + sprintId: string, + issue?: IIssue, + actionType?: "create" | "edit" | "delete" + ) => void; + openIssuesListModal: (cycleId: string) => void; + removeIssueFromCycle: (cycleId: string, bridgeId: string) => void; +}; + +const SingleCycleBoard: React.FC = ({ + properties, + groupedByIssues, + selectedGroup, + groupTitle, + createdBy, + bgColor, + openCreateIssueModal, + openIssuesListModal, + removeIssueFromCycle, +}) => { + // Collapse/Expand + const [show, setState] = useState(true); + + const { activeWorkspace, activeProject } = useUser(); + + if (selectedGroup === "priority") + groupTitle === "high" + ? (bgColor = "#dc2626") + : groupTitle === "medium" + ? (bgColor = "#f97316") + : groupTitle === "low" + ? (bgColor = "#22c55e") + : (bgColor = "#ff0000"); + + const { data: people } = useSWR( + activeWorkspace ? WORKSPACE_MEMBERS : null, + activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null + ); + + return ( +
+
+
+
+
+ +

+ {groupTitle === null || groupTitle === "null" + ? "None" + : createdBy + ? createdBy + : addSpaceIfCamelCase(groupTitle)} +

+ + {groupedByIssues[groupTitle].length} + +
+
+
+
+ {groupedByIssues[groupTitle].map((childIssue, index: number) => { + const assignees = [ + ...(childIssue?.assignees_list ?? []), + ...(childIssue?.assignees ?? []), + ]?.map((assignee) => { + const tempPerson = people?.find((p) => p.member.id === assignee)?.member; + + return { + avatar: tempPerson?.avatar, + first_name: tempPerson?.first_name, + email: tempPerson?.email, + }; + }); + + return ( +
+
+
+ +
+ + + {properties.key && ( +
+ {activeProject?.identifier}-{childIssue.sequence_id} +
+ )} +
+ {childIssue.name} +
+
+ +
+ {properties.priority && ( +
+ {/* {getPriorityIcon(childIssue.priority ?? "")} */} + {childIssue.priority ?? "None"} +
+
Priority
+
+ {childIssue.priority ?? "None"} +
+
+
+ )} + {properties.state && ( +
+ + {addSpaceIfCamelCase(childIssue.state_detail.name)} +
+
State
+
{childIssue.state_detail.name}
+
+
+ )} + {properties.start_date && ( +
+ + {childIssue.start_date + ? renderShortNumericDateFormat(childIssue.start_date) + : "N/A"} +
+
Started at
+
{renderShortNumericDateFormat(childIssue.start_date ?? "")}
+
+
+ )} + {properties.target_date && ( +
+ + {childIssue.target_date + ? renderShortNumericDateFormat(childIssue.target_date) + : "N/A"} +
+
Target date
+
{renderShortNumericDateFormat(childIssue.target_date ?? "")}
+
+ {childIssue.target_date && + (childIssue.target_date < new Date().toISOString() + ? `Target date has passed by ${findHowManyDaysLeft( + childIssue.target_date + )} days` + : findHowManyDaysLeft(childIssue.target_date) <= 3 + ? `Target date is in ${findHowManyDaysLeft( + childIssue.target_date + )} days` + : "Target date")} +
+
+
+ )} + {properties.assignee && ( +
+ {childIssue.assignee_details?.length > 0 ? ( + childIssue.assignee_details?.map((assignee, index: number) => ( +
+ {assignee.avatar && assignee.avatar !== "" ? ( +
+ {assignee.name} +
+ ) : ( +
+ {assignee.first_name.charAt(0)} +
+ )} +
+ )) + ) : ( +
+ No user +
+ )} +
+
Assigned to
+
+ {childIssue.assignee_details?.length > 0 + ? childIssue.assignee_details + .map((assignee) => assignee.first_name) + .join(", ") + : "No one"} +
+
+
+ )} +
+
+
+ ); + })} + +
+
+
+ ); + + // return ( + //
+ //
+ //
+ //
+ //
+ //

+ // {cycle.name} + //

+ // {cycleIssues?.length} + //
+ //
+ + //
+ // + // + // + // + // + + // + // + //
+ // + // {(active) => ( + // + // )} + // + // + // {(active) => ( + // + // )} + // + //
+ //
+ //
+ //
+ //
+ //
+ //
+ // {cycleIssues ? ( + // cycleIssues.map((issue, index: number) => ( + //
+ //
+ //
+ // + // + // + // + // + // + //
+ // + //
+ //
+ // + //
+ // + //
+ //
+ //
+ //
+ //
+ // + // + // {properties.key && ( + //
+ // {activeProject?.identifier}-{childIssue.sequence_id} + //
+ // )} + //
+ // {childIssue.name} + //
+ //
+ // + //
+ // {properties.priority && ( + //
+ // {/* {getPriorityIcon(childIssue.priority ?? "")} */} + // {childIssue.priority ?? "None"} + //
+ //
Priority
+ //
+ // {childIssue.priority ?? "None"} + //
+ //
+ //
+ // )} + // {properties.state && ( + //
+ // + // {addSpaceIfCamelCase(childIssue.state_detail.name)} + //
+ //
State
+ //
{childIssue.state_detail.name}
+ //
+ //
+ // )} + // {properties.start_date && ( + //
+ // + // {childIssue.start_date + // ? renderShortNumericDateFormat(childIssue.start_date) + // : "N/A"} + //
+ //
Started at
+ //
+ // {renderShortNumericDateFormat(childIssue.start_date ?? "")} + //
+ //
+ //
+ // )} + // {properties.target_date && ( + //
+ // + // {childIssue.target_date + // ? renderShortNumericDateFormat(childIssue.target_date) + // : "N/A"} + //
+ //
Target date
+ //
+ // {renderShortNumericDateFormat(childIssue.target_date ?? "")} + //
+ //
+ // {childIssue.target_date && + // (childIssue.target_date < new Date().toISOString() + // ? `Target date has passed by ${findHowManyDaysLeft( + // childIssue.target_date + // )} days` + // : findHowManyDaysLeft(childIssue.target_date) <= 3 + // ? `Target date is in ${findHowManyDaysLeft( + // childIssue.target_date + // )} days` + // : "Target date")} + //
+ //
+ //
+ // )} + // {properties.assignee && ( + //
+ // {childIssue.assignee_details?.length > 0 ? ( + // childIssue.assignee_details?.map((assignee, index: number) => ( + //
+ // {assignee.avatar && assignee.avatar !== "" ? ( + //
+ // {assignee.name} + //
+ // ) : ( + //
+ // {assignee.first_name.charAt(0)} + //
+ // )} + //
+ // )) + // ) : ( + //
+ // No user + //
+ // )} + //
+ //
Assigned to
+ //
+ // {childIssue.assignee_details?.length > 0 + // ? childIssue.assignee_details + // .map((assignee) => assignee.first_name) + // .join(", ") + // : "No one"} + //
+ //
+ //
+ // )} + //
+ //
+ //
+ // )) + // ) : ( + //
+ // + //
+ // )} + // + //
+ //
+ //
+ // ); +}; + +export default SingleCycleBoard; diff --git a/apps/app/components/project/cycles/ConfirmCycleDeletion.tsx b/apps/app/components/project/cycles/ConfirmCycleDeletion.tsx index fb5cf565b..b3571831a 100644 --- a/apps/app/components/project/cycles/ConfirmCycleDeletion.tsx +++ b/apps/app/components/project/cycles/ConfirmCycleDeletion.tsx @@ -4,7 +4,7 @@ import { mutate } from "swr"; // headless ui import { Dialog, Transition } from "@headlessui/react"; // services -import cycleService from "lib/services/cycles.services"; +import cycleService from "lib/services/cycles.service"; // fetch api import { CYCLE_LIST } from "constants/fetch-keys"; // hooks diff --git a/apps/app/components/project/cycles/CreateUpdateCyclesModal.tsx b/apps/app/components/project/cycles/CreateUpdateCyclesModal.tsx index 0fd25f481..3f3127082 100644 --- a/apps/app/components/project/cycles/CreateUpdateCyclesModal.tsx +++ b/apps/app/components/project/cycles/CreateUpdateCyclesModal.tsx @@ -6,7 +6,7 @@ import { useForm } from "react-hook-form"; // headless import { Dialog, Transition } from "@headlessui/react"; // services -import cycleService from "lib/services/cycles.services"; +import cycleService from "lib/services/cycles.service"; // fetch keys import { CYCLE_LIST } from "constants/fetch-keys"; // hooks diff --git a/apps/app/components/project/cycles/CycleIssuesListModal.tsx b/apps/app/components/project/cycles/CycleIssuesListModal.tsx index 5f29a5bfd..d9f3226c0 100644 --- a/apps/app/components/project/cycles/CycleIssuesListModal.tsx +++ b/apps/app/components/project/cycles/CycleIssuesListModal.tsx @@ -7,7 +7,7 @@ import { Combobox, Dialog, Transition } from "@headlessui/react"; // ui import { Button } from "ui"; // services -import issuesServices from "lib/services/issues.services"; +import issuesServices from "lib/services/issues.service"; // hooks import useUser from "lib/hooks/useUser"; import useToast from "lib/hooks/useToast"; @@ -47,7 +47,12 @@ const CycleIssuesListModal: React.FC = ({ reset(); }; - const { handleSubmit, reset, control } = useForm({ + const { + handleSubmit, + reset, + control, + formState: { isSubmitting }, + } = useForm({ defaultValues: { issue_ids: [], }, @@ -68,6 +73,7 @@ const CycleIssuesListModal: React.FC = ({ .bulkAddIssuesToCycle(activeWorkspace.slug, activeProject.id, cycleId, data) .then((res) => { console.log(res); + handleClose(); }) .catch((e) => { console.log(e); @@ -138,36 +144,39 @@ const CycleIssuesListModal: React.FC = ({ )}
    - {filteredIssues.map((issue) => ( - - classNames( - "flex items-center gap-2 cursor-pointer select-none w-full rounded-md px-3 py-2", - active ? "bg-gray-900 bg-opacity-5 text-gray-900" : "" - ) - } - > - {({ selected }) => ( - <> - - - - {activeProject?.identifier}-{issue.sequence_id} - - {issue.name} - - )} - - ))} + {filteredIssues.map((issue) => { + // if (issue.cycle !== cycleId) + return ( + + classNames( + "flex items-center gap-2 cursor-pointer select-none w-full rounded-md px-3 py-2", + active ? "bg-gray-900 bg-opacity-5 text-gray-900" : "" + ) + } + > + {({ selected }) => ( + <> + + + + {activeProject?.identifier}-{issue.sequence_id} + + {issue.name} + + )} + + ); + })}
)} @@ -191,8 +200,13 @@ const CycleIssuesListModal: React.FC = ({ - diff --git a/apps/app/components/project/cycles/CycleView.tsx b/apps/app/components/project/cycles/CycleView.tsx deleted file mode 100644 index b88883ae0..000000000 --- a/apps/app/components/project/cycles/CycleView.tsx +++ /dev/null @@ -1,332 +0,0 @@ -// react -import React, { useState } from "react"; -// next -import Link from "next/link"; -// swr -import useSWR, { mutate } from "swr"; -// headless ui -import { Disclosure, Transition, Menu } from "@headlessui/react"; -// services -import cycleServices from "lib/services/cycles.services"; -// hooks -import useUser from "lib/hooks/useUser"; -// components -import CycleIssuesListModal from "./CycleIssuesListModal"; -// ui -import { Spinner } from "ui"; -// icons -import { PlusIcon, EllipsisHorizontalIcon, ChevronDownIcon } from "@heroicons/react/20/solid"; -// types -import type { CycleViewProps as Props, CycleIssueResponse, IssueResponse } from "types"; -// fetch keys -import { CYCLE_ISSUES } from "constants/fetch-keys"; -// constants -import { addSpaceIfCamelCase, renderShortNumericDateFormat } from "constants/common"; -import issuesServices from "lib/services/issues.services"; -import StrictModeDroppable from "components/dnd/StrictModeDroppable"; -import { Draggable } from "react-beautiful-dnd"; -import { CalendarDaysIcon } from "@heroicons/react/24/outline"; - -const CycleView: React.FC = ({ - cycle, - selectSprint, - workspaceSlug, - projectId, - openIssueModal, -}) => { - const [cycleIssuesListModal, setCycleIssuesListModal] = useState(false); - - const { activeWorkspace, activeProject, issues } = useUser(); - - const { data: cycleIssues } = useSWR(CYCLE_ISSUES(cycle.id), () => - cycleServices.getCycleIssues(workspaceSlug, projectId, cycle.id) - ); - - const removeIssueFromCycle = (cycleId: string, bridgeId: string) => { - if (activeWorkspace && activeProject && cycleIssues) { - mutate( - CYCLE_ISSUES(cycleId), - (prevData) => prevData?.filter((p) => p.id !== bridgeId), - false - ); - - issuesServices - .removeIssueFromCycle(activeWorkspace.slug, activeProject.id, cycleId, bridgeId) - .then((res) => { - console.log(res); - }) - .catch((e) => { - console.log(e); - }); - } - }; - - return ( - <> - setCycleIssuesListModal(false)} - issues={issues} - cycleId={cycle.id} - /> - - {({ open }) => ( -
-
- -
- - - -

{cycle.name}

-

- - {cycle.status === "started" - ? cycle.start_date - ? `${renderShortNumericDateFormat(cycle.start_date)} - ` - : "" - : cycle.status} - - - {cycle.end_date ? renderShortNumericDateFormat(cycle.end_date) : ""} - -

-

{cycleIssues?.length}

-
-
- - - - - - - - - - - - - - -
- - - - {(provided) => ( -
- {cycleIssues ? ( - cycleIssues.length > 0 ? ( - cycleIssues.map((issue, index) => ( - - {(provided, snapshot) => ( -
- -
-
- - {issue.issue_details.start_date - ? renderShortNumericDateFormat( - issue.issue_details.start_date - ) - : "N/A"} -
-
- - {addSpaceIfCamelCase(issue.issue_details.state_detail.name)} -
- - - - - - - - - -
- -
-
- -
- -
-
-
-
-
-
- )} -
- )) - ) : ( -

This cycle has no issue.

- ) - ) : ( -
- -
- )} - {provided.placeholder} -
- )} -
-
-
-
- - - - Add issue - - - - -
- - {(active) => ( - - )} - - - {(active) => ( - - )} - -
-
-
-
-
-
- )} -
- - ); -}; - -export default CycleView; diff --git a/apps/app/components/project/cycles/ListView/index.tsx b/apps/app/components/project/cycles/ListView/index.tsx new file mode 100644 index 000000000..5add9687a --- /dev/null +++ b/apps/app/components/project/cycles/ListView/index.tsx @@ -0,0 +1,714 @@ +// react +import React from "react"; +// next +import Link from "next/link"; +// swr +import useSWR from "swr"; +// headless ui +import { Disclosure, Transition, Menu } from "@headlessui/react"; +// services +import cycleServices from "lib/services/cycles.service"; +// hooks +import useUser from "lib/hooks/useUser"; +// ui +import { Spinner } from "ui"; +// icons +import { PlusIcon, EllipsisHorizontalIcon, ChevronDownIcon } from "@heroicons/react/20/solid"; +import { CalendarDaysIcon } from "@heroicons/react/24/outline"; +// types +import { IIssue, IWorkspaceMember, NestedKeyOf, Properties, SelectSprintType } from "types"; +// fetch keys +import { CYCLE_ISSUES, WORKSPACE_MEMBERS } from "constants/fetch-keys"; +// constants +import { + addSpaceIfCamelCase, + findHowManyDaysLeft, + renderShortNumericDateFormat, +} from "constants/common"; +import workspaceService from "lib/services/workspace.service"; + +type Props = { + groupedByIssues: { + [key: string]: IIssue[]; + }; + properties: Properties; + selectedGroup: NestedKeyOf | null; + openCreateIssueModal: ( + sprintId: string, + issue?: IIssue, + actionType?: "create" | "edit" | "delete" + ) => void; + openIssuesListModal: (cycleId: string) => void; + removeIssueFromCycle: (cycleId: string, bridgeId: string) => void; +}; + +const CyclesListView: React.FC = ({ + groupedByIssues, + selectedGroup, + openCreateIssueModal, + openIssuesListModal, + properties, + removeIssueFromCycle, +}) => { + const { activeWorkspace, activeProject } = useUser(); + + const { data: people } = useSWR( + activeWorkspace ? WORKSPACE_MEMBERS : null, + activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null + ); + + return ( +
+ {Object.keys(groupedByIssues).map((singleGroup) => ( + + {({ open }) => ( +
+
+ +
+ + + + {selectedGroup !== null ? ( +

+ {singleGroup === null || singleGroup === "null" + ? selectedGroup === "priority" && "No priority" + : addSpaceIfCamelCase(singleGroup)} +

+ ) : ( +

All Issues

+ )} +

+ {groupedByIssues[singleGroup as keyof IIssue].length} +

+
+
+
+ + +
+ {groupedByIssues[singleGroup] ? ( + groupedByIssues[singleGroup].length > 0 ? ( + groupedByIssues[singleGroup].map((issue: IIssue) => { + const assignees = [ + ...(issue?.assignees_list ?? []), + ...(issue?.assignees ?? []), + ]?.map((assignee) => { + const tempPerson = people?.find( + (p) => p.member.id === assignee + )?.member; + + return { + avatar: tempPerson?.avatar, + first_name: tempPerson?.first_name, + email: tempPerson?.email, + }; + }); + + return ( +
+ +
+ {properties.priority && ( +
+ {/* {getPriorityIcon(issue.priority ?? "")} */} + {issue.priority ?? "None"} +
+
Priority
+
+ {issue.priority ?? "None"} +
+
+
+ )} + {properties.state && ( +
+ + {addSpaceIfCamelCase(issue?.state_detail.name)} +
+
State
+
{issue?.state_detail.name}
+
+
+ )} + {properties.start_date && ( +
+ + {issue.start_date + ? renderShortNumericDateFormat(issue.start_date) + : "N/A"} +
+
Started at
+
+ {renderShortNumericDateFormat(issue.start_date ?? "")} +
+
+
+ )} + {properties.target_date && ( +
+ + {issue.target_date + ? renderShortNumericDateFormat(issue.target_date) + : "N/A"} +
+
+ Target date +
+
+ {renderShortNumericDateFormat(issue.target_date ?? "")} +
+
+ {issue.target_date && + (issue.target_date < new Date().toISOString() + ? `Target date has passed by ${findHowManyDaysLeft( + issue.target_date + )} days` + : findHowManyDaysLeft(issue.target_date) <= 3 + ? `Target date is in ${findHowManyDaysLeft( + issue.target_date + )} days` + : "Target date")} +
+
+
+ )} + + + + + + + + + +
+ +
+
+ +
+ +
+
+
+
+
+
+ ); + }) + ) : ( +

No issues.

+ ) + ) : ( +
+ +
+ )} +
+
+
+
+ +
+
+ )} +
+ ))} +
+ ); + + // return ( + // <> + // + // {({ open }) => ( + //
+ //
+ // + + // + // + // + // + // + // + // + // + // + // + // + // + // + //
+ // + // + // + // {(provided) => ( + //
+ // {cycleIssues ? ( + // cycleIssues.length > 0 ? ( + // cycleIssues.map((issue, index) => ( + // + // {(provided, snapshot) => ( + //
+ // + //
+ // {properties.priority && ( + //
+ // {/* {getPriorityIcon(issue.priority ?? "")} */} + // {issue.priority ?? "None"} + //
+ //
+ // Priority + //
+ //
+ // {issue.priority ?? "None"} + //
+ //
+ //
+ // )} + // {properties.state && ( + //
+ // + // {addSpaceIfCamelCase( + // issue?.state_detail.name + // )} + //
+ //
State
+ //
{issue?.state_detail.name}
+ //
+ //
+ // )} + // {properties.start_date && ( + //
+ // + // {issue.start_date + // ? renderShortNumericDateFormat( + // issue.start_date + // ) + // : "N/A"} + //
+ //
Started at
+ //
+ // {renderShortNumericDateFormat( + // issue.start_date ?? "" + // )} + //
+ //
+ //
+ // )} + // {properties.target_date && ( + //
+ // + // {issue.target_date + // ? renderShortNumericDateFormat( + // issue.target_date + // ) + // : "N/A"} + //
+ //
+ // Target date + //
+ //
+ // {renderShortNumericDateFormat( + // issue.target_date ?? "" + // )} + //
+ //
+ // {issue.target_date && + // (issue.target_date < + // new Date().toISOString() + // ? `Target date has passed by ${findHowManyDaysLeft( + // issue.target_date + // )} days` + // : findHowManyDaysLeft( + // issue.target_date + // ) <= 3 + // ? `Target date is in ${findHowManyDaysLeft( + // issue.target_date + // )} days` + // : "Target date")} + //
+ //
+ //
+ // )} + // + // + // + // + // + // + // + // + // + //
+ // + //
+ //
+ // + //
+ // + //
+ //
+ //
+ //
+ //
+ //
+ // )} + //
+ // )) + // ) : ( + //

+ // This cycle has no issue. + //

+ // ) + // ) : ( + //
+ // + //
+ // )} + // {provided.placeholder} + //
+ // )} + //
+ //
+ //
+ //
+ // + // + // + // Add issue + // + + // + // + //
+ // + // {(active) => ( + // + // )} + // + // + // {(active) => ( + // + // )} + // + //
+ //
+ //
+ //
+ //
+ //
+ // )} + //
+ // + // ); +}; + +export default CyclesListView; diff --git a/apps/app/components/project/issues/BoardView/SingleBoard.tsx b/apps/app/components/project/issues/BoardView/SingleBoard.tsx deleted file mode 100644 index 08efcdac3..000000000 --- a/apps/app/components/project/issues/BoardView/SingleBoard.tsx +++ /dev/null @@ -1,338 +0,0 @@ -import React, { useState } from "react"; -// Next imports -import Link from "next/link"; -import Image from "next/image"; -// React beautiful dnd -import { Draggable } from "react-beautiful-dnd"; -import StrictModeDroppable from "components/dnd/StrictModeDroppable"; -// types -import { IIssue, Properties, NestedKeyOf } from "types"; -// icons -import { - ArrowsPointingInIcon, - ArrowsPointingOutIcon, - CalendarDaysIcon, - EllipsisHorizontalIcon, - PlusIcon, -} from "@heroicons/react/24/outline"; -import User from "public/user.png"; -// common -import { - addSpaceIfCamelCase, - findHowManyDaysLeft, - renderShortNumericDateFormat, -} from "constants/common"; -import { getPriorityIcon } from "constants/global"; - -type Props = { - selectedGroup: NestedKeyOf | null; - groupTitle: string; - groupedByIssues: { - [key: string]: IIssue[]; - }; - index: number; - setIsIssueOpen: React.Dispatch>; - properties: Properties; - setPreloadedData: React.Dispatch< - React.SetStateAction< - | (Partial & { - actionType: "createIssue" | "edit" | "delete"; - }) - | undefined - > - >; - bgColor?: string; - stateId: string | null; - createdBy: string | null; -}; - -const SingleBoard: React.FC = ({ - selectedGroup, - groupTitle, - groupedByIssues, - index, - setIsIssueOpen, - properties, - setPreloadedData, - bgColor = "#0f2b16", - stateId, - createdBy, -}) => { - // Collapse/Expand - const [show, setState] = useState(true); - - if (selectedGroup === "priority") - groupTitle === "high" - ? (bgColor = "#dc2626") - : groupTitle === "medium" - ? (bgColor = "#f97316") - : groupTitle === "low" - ? (bgColor = "#22c55e") - : (bgColor = "#ff0000"); - - return ( - - {(provided, snapshot) => ( -
-
-
-
- -
- -

- {groupTitle === null || groupTitle === "null" - ? "None" - : createdBy - ? createdBy - : addSpaceIfCamelCase(groupTitle)} -

- - {groupedByIssues[groupTitle].length} - -
-
- -
- - -
-
- - {(provided, snapshot) => ( -
- {groupedByIssues[groupTitle].map((childIssue, index: number) => ( - - {(provided, snapshot) => ( - - -
- {properties.key && ( -
- {childIssue.project_detail?.identifier}-{childIssue.sequence_id} -
- )} -
- {childIssue.name} -
-
- {properties.priority && ( -
- {/* {getPriorityIcon(childIssue.priority ?? "")} */} - {childIssue.priority ?? "None"} -
- )} - {properties.state && ( -
- - {addSpaceIfCamelCase(childIssue.state_detail.name)} -
- )} - {properties.start_date && ( -
- - {childIssue.start_date - ? renderShortNumericDateFormat(childIssue.start_date) - : "N/A"} -
- )} - {properties.target_date && ( -
- - {childIssue.target_date - ? renderShortNumericDateFormat(childIssue.target_date) - : "N/A"} - {childIssue.target_date && ( - - {childIssue.target_date < new Date().toISOString() - ? `Target date has passed by ${findHowManyDaysLeft( - childIssue.target_date - )} days` - : findHowManyDaysLeft(childIssue.target_date) <= 3 - ? `Target date is in ${findHowManyDaysLeft( - childIssue.target_date - )} days` - : "Target date"} - - )} -
- )} - {properties.assignee && ( -
- {childIssue?.assignee_details?.length > 0 ? ( - childIssue?.assignee_details?.map( - (assignee, index: number) => ( -
- {assignee.avatar && assignee.avatar !== "" ? ( -
- {assignee.name} -
- ) : ( -
- {assignee.first_name.charAt(0)} -
- )} -
- ) - ) - ) : ( -
- No user -
- )} -
- )} -
-
-
- - )} -
- ))} - {provided.placeholder} - -
- )} -
-
-
- )} -
- ); -}; - -export default SingleBoard; diff --git a/apps/app/components/project/issues/BoardView/index.tsx b/apps/app/components/project/issues/BoardView/index.tsx index fe69a091e..86d54c526 100644 --- a/apps/app/components/project/issues/BoardView/index.tsx +++ b/apps/app/components/project/issues/BoardView/index.tsx @@ -7,22 +7,21 @@ import useSWR from "swr"; import type { DropResult } from "react-beautiful-dnd"; import { DragDropContext } from "react-beautiful-dnd"; // services -import stateServices from "lib/services/state.services"; -import issuesServices from "lib/services/issues.services"; +import stateServices from "lib/services/state.service"; +import issuesServices from "lib/services/issues.service"; // hooks import useUser from "lib/hooks/useUser"; // fetching keys import { STATE_LIST } from "constants/fetch-keys"; // components -import SingleBoard from "components/project/issues/BoardView/SingleBoard"; +import SingleBoard from "components/project/issues/BoardView/single-board"; import StrictModeDroppable from "components/dnd/StrictModeDroppable"; import CreateUpdateIssuesModal from "components/project/issues/CreateUpdateIssueModal"; // ui import { Spinner } from "ui"; // types -import type { IState, IIssue, Properties, NestedKeyOf, ProjectMember } from "types"; -import ConfirmIssueDeletion from "../ConfirmIssueDeletion"; -import { TrashIcon } from "@heroicons/react/24/outline"; +import type { IState, IIssue, Properties, NestedKeyOf, IProjectMember } from "types"; +import ConfirmIssueDeletion from "../confirm-issue-deletion"; type Props = { properties: Properties; @@ -30,10 +29,19 @@ type Props = { groupedByIssues: { [key: string]: IIssue[]; }; - members: ProjectMember[] | undefined; + members: IProjectMember[] | undefined; + handleDeleteIssue: React.Dispatch>; + partialUpdateIssue: (formData: Partial, issueId: string) => void; }; -const BoardView: React.FC = ({ properties, selectedGroup, groupedByIssues, members }) => { +const BoardView: React.FC = ({ + properties, + selectedGroup, + groupedByIssues, + members, + handleDeleteIssue, + partialUpdateIssue, +}) => { const [isOpen, setIsOpen] = useState(false); const [isIssueOpen, setIsIssueOpen] = useState(false); @@ -217,6 +225,8 @@ const BoardView: React.FC = ({ properties, selectedGroup, groupedByIssues ? states?.find((s) => s.name === singleGroup)?.color : undefined } + handleDeleteIssue={handleDeleteIssue} + partialUpdateIssue={partialUpdateIssue} /> ))} diff --git a/apps/app/components/project/issues/BoardView/single-board.tsx b/apps/app/components/project/issues/BoardView/single-board.tsx new file mode 100644 index 000000000..7e843a8ca --- /dev/null +++ b/apps/app/components/project/issues/BoardView/single-board.tsx @@ -0,0 +1,609 @@ +// react +import React, { useState } from "react"; +// next +import Link from "next/link"; +import Image from "next/image"; +// swr +import useSWR from "swr"; +// react-beautiful-dnd +import { Draggable } from "react-beautiful-dnd"; +import StrictModeDroppable from "components/dnd/StrictModeDroppable"; +// services +import workspaceService from "lib/services/workspace.service"; +// hooks +import useUser from "lib/hooks/useUser"; +// headless ui +import { Listbox, Transition } from "@headlessui/react"; +// icons +import { + ArrowsPointingInIcon, + ArrowsPointingOutIcon, + CalendarDaysIcon, + EllipsisHorizontalIcon, + PlusIcon, + TrashIcon, +} from "@heroicons/react/24/outline"; +import User from "public/user.png"; +// common +import { PRIORITIES } from "constants/"; +import { + addSpaceIfCamelCase, + classNames, + findHowManyDaysLeft, + renderShortNumericDateFormat, +} from "constants/common"; +import { WORKSPACE_MEMBERS } from "constants/fetch-keys"; +import { getPriorityIcon } from "constants/global"; +// types +import { IIssue, Properties, NestedKeyOf, IWorkspaceMember } from "types"; + +type Props = { + selectedGroup: NestedKeyOf | null; + groupTitle: string; + groupedByIssues: { + [key: string]: IIssue[]; + }; + index: number; + setIsIssueOpen: React.Dispatch>; + properties: Properties; + setPreloadedData: React.Dispatch< + React.SetStateAction< + | (Partial & { + actionType: "createIssue" | "edit" | "delete"; + }) + | undefined + > + >; + bgColor?: string; + stateId: string | null; + createdBy: string | null; + handleDeleteIssue: React.Dispatch>; + partialUpdateIssue: (formData: Partial, childIssueId: string) => void; +}; + +const SingleBoard: React.FC = ({ + selectedGroup, + groupTitle, + groupedByIssues, + index, + setIsIssueOpen, + properties, + setPreloadedData, + bgColor = "#0f2b16", + stateId, + createdBy, + handleDeleteIssue, + partialUpdateIssue, +}) => { + // Collapse/Expand + const [show, setShow] = useState(true); + + const { activeProject, activeWorkspace, states } = useUser(); + + if (selectedGroup === "priority") + groupTitle === "high" + ? (bgColor = "#dc2626") + : groupTitle === "medium" + ? (bgColor = "#f97316") + : groupTitle === "low" + ? (bgColor = "#22c55e") + : (bgColor = "#ff0000"); + + const { data: people } = useSWR( + activeWorkspace ? WORKSPACE_MEMBERS : null, + activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null + ); + + return ( + + {(provided, snapshot) => ( +
+
+
+
+ +
+ +

+ {groupTitle === null || groupTitle === "null" + ? "None" + : createdBy + ? createdBy + : addSpaceIfCamelCase(groupTitle)} +

+ + {groupedByIssues[groupTitle].length} + +
+
+ +
+ + +
+
+ + {(provided, snapshot) => ( +
+ {groupedByIssues[groupTitle].map((childIssue, index: number) => { + const assignees = [ + ...(childIssue?.assignees_list ?? []), + ...(childIssue?.assignees ?? []), + ]?.map((assignee) => { + const tempPerson = people?.find((p) => p.member.id === assignee)?.member; + + return { + avatar: tempPerson?.avatar, + first_name: tempPerson?.first_name, + email: tempPerson?.email, + }; + }); + + return ( + + {(provided, snapshot) => ( +
+
+
+ +
+ + + {properties.key && ( +
+ {activeProject?.identifier}-{childIssue.sequence_id} +
+ )} +
+ {childIssue.name} +
+
+ +
+ {properties.priority && ( + { + partialUpdateIssue({ priority: data }, childIssue.id); + }} + className="group relative flex-shrink-0" + > + {({ open }) => ( + <> +
+ + {childIssue.priority ?? "None"} + + + + + {PRIORITIES?.map((priority) => ( + + classNames( + active ? "bg-indigo-50" : "bg-white", + "cursor-pointer capitalize select-none px-3 py-2" + ) + } + value={priority} + > + {priority} + + ))} + + +
+
+
+ Priority +
+
+ {childIssue.priority ?? "None"} +
+
+ + )} +
+ )} + {properties.state && ( + { + partialUpdateIssue({ state: data }, childIssue.id); + }} + className="group relative flex-shrink-0" + > + {({ open }) => ( + <> +
+ + + {addSpaceIfCamelCase(childIssue.state_detail.name)} + + + + + {states?.map((state) => ( + + classNames( + active ? "bg-indigo-50" : "bg-white", + "cursor-pointer select-none px-3 py-2" + ) + } + value={state.id} + > + {addSpaceIfCamelCase(state.name)} + + ))} + + +
+
+
State
+
{childIssue.state_detail.name}
+
+ + )} +
+ )} + {properties.start_date && ( +
+ + {childIssue.start_date + ? renderShortNumericDateFormat(childIssue.start_date) + : "N/A"} +
+
Started at
+
+ {renderShortNumericDateFormat(childIssue.start_date ?? "")} +
+
+
+ )} + {properties.target_date && ( +
+ + {childIssue.target_date + ? renderShortNumericDateFormat(childIssue.target_date) + : "N/A"} +
+
+ Target date +
+
+ {renderShortNumericDateFormat(childIssue.target_date ?? "")} +
+
+ {childIssue.target_date && + (childIssue.target_date < new Date().toISOString() + ? `Target date has passed by ${findHowManyDaysLeft( + childIssue.target_date + )} days` + : findHowManyDaysLeft(childIssue.target_date) <= 3 + ? `Target date is in ${findHowManyDaysLeft( + childIssue.target_date + )} days` + : "Target date")} +
+
+
+ )} + {properties.assignee && ( + { + const newData = childIssue.assignees ?? []; + if (newData.includes(data)) { + newData.splice(newData.indexOf(data), 1); + } else { + newData.push(data); + } + partialUpdateIssue( + { assignees_list: newData }, + childIssue.id + ); + }} + className="group relative flex-shrink-0" + > + {({ open }) => ( + <> +
+ +
+ {assignees.length > 0 ? ( + assignees.map((assignee, index: number) => ( +
+ {assignee.avatar && assignee.avatar !== "" ? ( +
+ {assignee?.first_name} +
+ ) : ( +
+ {assignee.first_name?.charAt(0)} +
+ )} +
+ )) + ) : ( +
+ No user +
+ )} +
+
+ + + + {people?.map((person) => ( + + classNames( + active ? "bg-indigo-50" : "bg-white", + "cursor-pointer select-none p-2" + ) + } + value={person.member.id} + > +
+ {person.member.avatar && + person.member.avatar !== "" ? ( +
+ avatar +
+ ) : ( +
+ {person.member.first_name && + person.member.first_name !== "" + ? person.member.first_name.charAt(0) + : person.member.email.charAt(0)} +
+ )} +

+ {person.member.first_name && + person.member.first_name !== "" + ? person.member.first_name + : person.member.email} +

+
+
+ ))} +
+
+
+
+
Assigned to
+
+ {childIssue.assignee_details?.length > 0 + ? childIssue.assignee_details + .map((assignee) => assignee.first_name) + .join(", ") + : "No one"} +
+
+ + )} +
+ )} +
+
+
+ )} +
+ ); + })} + {provided.placeholder} + +
+ )} +
+
+
+ )} +
+ ); +}; + +export default SingleBoard; diff --git a/apps/app/components/project/issues/BoardView/state/ConfirmStateDeletion.tsx b/apps/app/components/project/issues/BoardView/state/ConfirmStateDeletion.tsx index b8754437a..448c50acc 100644 --- a/apps/app/components/project/issues/BoardView/state/ConfirmStateDeletion.tsx +++ b/apps/app/components/project/issues/BoardView/state/ConfirmStateDeletion.tsx @@ -4,7 +4,7 @@ import { mutate } from "swr"; // headless ui import { Dialog, Transition } from "@headlessui/react"; // services -import stateServices from "lib/services/state.services"; +import stateServices from "lib/services/state.service"; // fetch api import { STATE_LIST } from "constants/fetch-keys"; // hooks @@ -43,7 +43,7 @@ const ConfirmStateDeletion: React.FC = ({ isOpen, setIsOpen, data }) => { mutate( STATE_LIST(data.project), (prevData) => prevData?.filter((state) => state.id !== data?.id), - false, + false ); handleClose(); }) @@ -98,18 +98,15 @@ const ConfirmStateDeletion: React.FC = ({ isOpen, setIsOpen, data }) => { />
- + Delete State

Are you sure you want to delete state - {`"`} {data?.name} - {`"`} ? All of the data related to the state will be - permanently removed. This action cannot be undone. + {`"`} ? All of the data related to the state will be permanently removed. + This action cannot be undone.

diff --git a/apps/app/components/project/issues/BoardView/state/CreateUpdateStateModal.tsx b/apps/app/components/project/issues/BoardView/state/CreateUpdateStateModal.tsx index 9f3b3951b..3ba149820 100644 --- a/apps/app/components/project/issues/BoardView/state/CreateUpdateStateModal.tsx +++ b/apps/app/components/project/issues/BoardView/state/CreateUpdateStateModal.tsx @@ -8,7 +8,7 @@ import { TwitterPicker } from "react-color"; // headless import { Dialog, Popover, Transition } from "@headlessui/react"; // services -import stateService from "lib/services/state.services"; +import stateService from "lib/services/state.service"; // fetch keys import { STATE_LIST } from "constants/fetch-keys"; // hooks diff --git a/apps/app/components/project/issues/CreateUpdateIssueModal/SelectAssignee.tsx b/apps/app/components/project/issues/CreateUpdateIssueModal/SelectAssignee.tsx index 376ab65e6..dc5d2b741 100644 --- a/apps/app/components/project/issues/CreateUpdateIssueModal/SelectAssignee.tsx +++ b/apps/app/components/project/issues/CreateUpdateIssueModal/SelectAssignee.tsx @@ -11,7 +11,7 @@ import useUser from "lib/hooks/useUser"; import { PROJECT_MEMBERS } from "constants/fetch-keys"; // types import type { Control } from "react-hook-form"; -import type { IIssue, WorkspaceMember } from "types"; +import type { IIssue } from "types"; import { UserIcon } from "@heroicons/react/24/outline"; import { SearchListbox } from "ui"; @@ -23,7 +23,7 @@ type Props = { const SelectAssignee: React.FC = ({ control }) => { const { activeWorkspace, activeProject } = useUser(); - const { data: people } = useSWR( + const { data: people } = useSWR( activeWorkspace && activeProject ? PROJECT_MEMBERS(activeProject.id) : null, activeWorkspace && activeProject ? () => projectServices.projectMembers(activeWorkspace.slug, activeProject.id) diff --git a/apps/app/components/project/issues/CreateUpdateIssueModal/SelectLabels.tsx b/apps/app/components/project/issues/CreateUpdateIssueModal/SelectLabels.tsx index a7ef75203..a8ca2ee74 100644 --- a/apps/app/components/project/issues/CreateUpdateIssueModal/SelectLabels.tsx +++ b/apps/app/components/project/issues/CreateUpdateIssueModal/SelectLabels.tsx @@ -6,7 +6,7 @@ import { useForm, Controller } from "react-hook-form"; // headless ui import { Listbox, Transition } from "@headlessui/react"; // services -import issuesServices from "lib/services/issues.services"; +import issuesServices from "lib/services/issues.service"; // hooks import useUser from "lib/hooks/useUser"; // fetching keys @@ -106,12 +106,16 @@ const SelectLabels: React.FC = ({ control }) => { className={({ active }) => `${ active ? "text-white bg-theme" : "text-gray-900" - } cursor-pointer select-none w-full p-2 rounded-md` + } flex items-center gap-2 cursor-pointer select-none w-full p-2 rounded-md` } value={label.id} > {({ selected, active }) => ( <> + i === label.id) diff --git a/apps/app/components/project/issues/CreateUpdateIssueModal/index.tsx b/apps/app/components/project/issues/CreateUpdateIssueModal/index.tsx index b7559d988..c7bcd1ad5 100644 --- a/apps/app/components/project/issues/CreateUpdateIssueModal/index.tsx +++ b/apps/app/components/project/issues/CreateUpdateIssueModal/index.tsx @@ -16,7 +16,7 @@ import { // headless import { Dialog, Menu, Transition } from "@headlessui/react"; // services -import issuesServices from "lib/services/issues.services"; +import issuesServices from "lib/services/issues.service"; // hooks import useUser from "lib/hooks/useUser"; import useToast from "lib/hooks/useToast"; @@ -34,11 +34,14 @@ import SelectAssignee from "./SelectAssignee"; import SelectParent from "./SelectParentIssue"; import CreateUpdateStateModal from "components/project/issues/BoardView/state/CreateUpdateStateModal"; import CreateUpdateCycleModal from "components/project/cycles/CreateUpdateCyclesModal"; - // types import type { IIssue, IssueResponse, CycleIssueResponse } from "types"; import { EllipsisHorizontalIcon } from "@heroicons/react/24/outline"; +const RichTextEditor = dynamic(() => import("components/lexical/editor"), { + ssr: false, +}); + type Props = { isOpen: boolean; setIsOpen: React.Dispatch>; @@ -78,10 +81,6 @@ const CreateUpdateIssuesModal: React.FC = ({ // setIssueDescriptionValue(value); // }; - const RichTextEditor = dynamic(() => import("components/lexical/editor"), { - ssr: false, - }); - const router = useRouter(); const handleClose = () => { @@ -117,7 +116,7 @@ const CreateUpdateIssuesModal: React.FC = ({ const addIssueToSprint = async (issueId: string, sprintId: string, issueDetail: IIssue) => { if (!activeWorkspace || !activeProject) return; await issuesServices - .addIssueToSprint(activeWorkspace.slug, activeProject.id, sprintId, { + .addIssueToCycle(activeWorkspace.slug, activeProject.id, sprintId, { issue: issueId, }) .then((res) => { @@ -176,6 +175,7 @@ const CreateUpdateIssuesModal: React.FC = ({ const payload: Partial = { ...formData, target_date: formData.target_date ? renderDateFormat(formData.target_date ?? "") : null, + // description: formData.description ? JSON.parse(formData.description) : null, }; if (!data) { await issuesServices diff --git a/apps/app/components/project/issues/ListView/index.tsx b/apps/app/components/project/issues/ListView/index.tsx index c12495849..a95c83606 100644 --- a/apps/app/components/project/issues/ListView/index.tsx +++ b/apps/app/components/project/issues/ListView/index.tsx @@ -4,7 +4,7 @@ import React, { useState } from "react"; import Link from "next/link"; import Image from "next/image"; // swr -import useSWR, { mutate } from "swr"; +import useSWR from "swr"; // headless ui import { Disclosure, Listbox, Menu, Transition } from "@headlessui/react"; // ui @@ -20,15 +20,14 @@ import User from "public/user.png"; // components import CreateUpdateIssuesModal from "components/project/issues/CreateUpdateIssueModal"; // types -import { IIssue, IssueResponse, NestedKeyOf, Properties, WorkspaceMember } from "types"; +import { IIssue, IWorkspaceMember, NestedKeyOf, Properties } from "types"; +// services +import workspaceService from "lib/services/workspace.service"; // hooks import useUser from "lib/hooks/useUser"; // fetch keys import { PRIORITIES } from "constants/"; -import { PROJECT_ISSUES_LIST, WORKSPACE_MEMBERS } from "constants/fetch-keys"; -// services -import issuesServices from "lib/services/issues.services"; -import workspaceService from "lib/services/workspace.service"; +import { WORKSPACE_MEMBERS } from "constants/fetch-keys"; // constants import { addSpaceIfCamelCase, @@ -44,6 +43,7 @@ type Props = { selectedGroup: NestedKeyOf | null; setSelectedIssue: any; handleDeleteIssue: React.Dispatch>; + partialUpdateIssue: (formData: Partial, issueId: string) => void; }; const ListView: React.FC = ({ @@ -52,6 +52,7 @@ const ListView: React.FC = ({ selectedGroup, setSelectedIssue, handleDeleteIssue, + partialUpdateIssue, }) => { const [isCreateIssuesModalOpen, setIsCreateIssuesModalOpen] = useState(false); const [preloadedData, setPreloadedData] = useState< @@ -60,27 +61,7 @@ const ListView: React.FC = ({ const { activeWorkspace, activeProject, states } = useUser(); - const partialUpdateIssue = (formData: Partial, issueId: string) => { - if (!activeWorkspace || !activeProject) return; - issuesServices - .patchIssue(activeWorkspace.slug, activeProject.id, issueId, formData) - .then((response) => { - mutate( - PROJECT_ISSUES_LIST(activeWorkspace.slug, activeProject.id), - (prevData) => ({ - ...(prevData as IssueResponse), - results: - prevData?.results.map((issue) => (issue.id === response.id ? response : issue)) ?? [], - }), - false - ); - }) - .catch((error) => { - console.log(error); - }); - }; - - const { data: people } = useSWR( + const { data: people } = useSWR( activeWorkspace ? WORKSPACE_MEMBERS : null, activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null ); @@ -95,7 +76,7 @@ const ListView: React.FC = ({ }} projectId={activeProject?.id as string} /> -
+
{Object.keys(groupedByIssues).map((singleGroup) => ( {({ open }) => ( @@ -155,23 +136,27 @@ const ListView: React.FC = ({ return (
@@ -183,7 +168,7 @@ const ListView: React.FC = ({ onChange={(data: string) => { partialUpdateIssue({ priority: data }, issue.id); }} - className="flex-shrink-0" + className="group relative flex-shrink-0" > {({ open }) => ( <> @@ -229,6 +214,26 @@ const ListView: React.FC = ({
+
+
+ Priority +
+
+ {issue.priority ?? "None"} +
+
)} @@ -240,7 +245,7 @@ const ListView: React.FC = ({ onChange={(data: string) => { partialUpdateIssue({ state: data }, issue.id); }} - className="flex-shrink-0" + className="group relative flex-shrink-0" > {({ open }) => ( <> @@ -280,21 +285,31 @@ const ListView: React.FC = ({
+
+
State
+
{issue.state_detail.name}
+
)} )} {properties.start_date && ( -
+
{issue.start_date ? renderShortNumericDateFormat(issue.start_date) : "N/A"} +
+
Started at
+
+ {renderShortNumericDateFormat(issue.start_date ?? "")} +
+
)} {properties.target_date && (
= ({ {issue.target_date ? renderShortNumericDateFormat(issue.target_date) : "N/A"} - {issue.target_date && ( - - {issue.target_date < new Date().toISOString() - ? `Target date has passed by ${findHowManyDaysLeft( - issue.target_date - )} days` - : findHowManyDaysLeft(issue.target_date) <= 3 - ? `Target date is in ${findHowManyDaysLeft( - issue.target_date - )} days` - : "Target date"} - - )} +
+
+ Target date +
+
+ {renderShortNumericDateFormat(issue.target_date ?? "")} +
+
+ {issue.target_date && + (issue.target_date < new Date().toISOString() + ? `Target date has passed by ${findHowManyDaysLeft( + issue.target_date + )} days` + : findHowManyDaysLeft(issue.target_date) <= 3 + ? `Target date is in ${findHowManyDaysLeft( + issue.target_date + )} days` + : "Target date")} +
+
)} {properties.assignee && ( @@ -335,7 +357,7 @@ const ListView: React.FC = ({ } partialUpdateIssue({ assignees_list: newData }, issue.id); }} - className="relative flex-shrink-0" + className="group relative flex-shrink-0" > {({ open }) => ( <> @@ -397,7 +419,7 @@ const ListView: React.FC = ({ className={({ active }) => classNames( active ? "bg-indigo-50" : "bg-white", - "cursor-pointer select-none px-3 py-2" + "cursor-pointer select-none p-2" ) } value={person.member.id} @@ -444,6 +466,16 @@ const ListView: React.FC = ({
+
+
Assigned to
+
+ {issue.assignee_details?.length > 0 + ? issue.assignee_details + .map((assignee) => assignee.first_name) + .join(", ") + : "No one"} +
+
)} diff --git a/apps/app/components/project/issues/ConfirmIssueDeletion.tsx b/apps/app/components/project/issues/confirm-issue-deletion.tsx similarity index 80% rename from apps/app/components/project/issues/ConfirmIssueDeletion.tsx rename to apps/app/components/project/issues/confirm-issue-deletion.tsx index 3b161ae6c..7be85390e 100644 --- a/apps/app/components/project/issues/ConfirmIssueDeletion.tsx +++ b/apps/app/components/project/issues/confirm-issue-deletion.tsx @@ -1,4 +1,4 @@ -import React, { useEffect, useRef, useState } from "react"; +import React, { useRef, useState } from "react"; // swr import { mutate } from "swr"; // headless ui @@ -6,7 +6,7 @@ import { Dialog, Transition } from "@headlessui/react"; // fetching keys import { PROJECT_ISSUES_LIST } from "constants/fetch-keys"; // services -import issueServices from "lib/services/issues.services"; +import issueServices from "lib/services/issues.service"; // hooks import useUser from "lib/hooks/useUser"; import useToast from "lib/hooks/useToast"; @@ -26,7 +26,7 @@ type Props = { const ConfirmIssueDeletion: React.FC = ({ isOpen, handleClose, data }) => { const [isDeleteLoading, setIsDeleteLoading] = useState(false); - const { activeWorkspace } = useUser(); + const { activeWorkspace, activeProject } = useUser(); const { setToastAlert } = useToast(); @@ -70,7 +70,7 @@ const ConfirmIssueDeletion: React.FC = ({ isOpen, handleClose, data }) => return ( - + = ({ isOpen, handleClose, data }) =>
-
-
-
- - Delete Issue +
+
+
+ + Are you sure you want to delete {`"`} + {activeProject?.identifier}-{data?.sequence_id} - {data?.name}?{`"`}

- Are you sure you want to delete issue - {`"`} - {data?.name} - {`"`} ? All of the data related to the issue will be permanently removed. - This action cannot be undone. + All of the data related to the issue will be permanently removed. This + action cannot be undone.

diff --git a/apps/app/components/project/issues/issue-detail/IssueDetailSidebar.tsx b/apps/app/components/project/issues/issue-detail/IssueDetailSidebar.tsx index 12f0773f2..c6e1eb573 100644 --- a/apps/app/components/project/issues/issue-detail/IssueDetailSidebar.tsx +++ b/apps/app/components/project/issues/issue-detail/IssueDetailSidebar.tsx @@ -6,12 +6,14 @@ import { Listbox, Transition } from "@headlessui/react"; // react hook form import { useForm, Controller, UseFormWatch } from "react-hook-form"; // services -import stateServices from "lib/services/state.services"; -import issuesServices from "lib/services/issues.services"; +import stateServices from "lib/services/state.service"; +import issuesServices from "lib/services/issues.service"; import workspaceService from "lib/services/workspace.service"; // hooks import useUser from "lib/hooks/useUser"; import useToast from "lib/hooks/useToast"; +// components +import IssuesListModal from "components/project/issues/IssuesListModal"; // fetching keys import { PROJECT_ISSUES_LIST, @@ -37,25 +39,22 @@ import { LinkIcon, ArrowPathIcon, CalendarDaysIcon, + TrashIcon, + PlusIcon, + XMarkIcon, } from "@heroicons/react/24/outline"; // types import type { Control } from "react-hook-form"; -import type { - IIssue, - IIssueLabels, - IssueResponse, - IState, - NestedKeyOf, - WorkspaceMember, -} from "types"; +import type { IIssue, IIssueLabels, IssueResponse, IState, NestedKeyOf } from "types"; import { TwitterPicker } from "react-color"; -import IssuesListModal from "components/project/issues/IssuesListModal"; +import { positionEditorElement } from "components/lexical/helpers/editor"; type Props = { control: Control; submitChanges: (formData: Partial) => void; issueDetail: IIssue | undefined; watch: UseFormWatch; + setDeleteIssueModal: React.Dispatch>; }; const defaultValues: Partial = { @@ -65,13 +64,15 @@ const defaultValues: Partial = { const IssueDetailSidebar: React.FC = ({ control, - watch: watchIssue, submitChanges, issueDetail, + watch: watchIssue, + setDeleteIssueModal, }) => { const [isBlockerModalOpen, setIsBlockerModalOpen] = useState(false); const [isBlockedModalOpen, setIsBlockedModalOpen] = useState(false); const [isParentModalOpen, setIsParentModalOpen] = useState(false); + const [createLabelForm, setCreateLabelForm] = useState(false); const { activeWorkspace, activeProject, cycles, issues } = useUser(); @@ -84,7 +85,7 @@ const IssueDetailSidebar: React.FC = ({ : null ); - const { data: people } = useSWR( + const { data: people } = useSWR( activeWorkspace ? WORKSPACE_MEMBERS(activeWorkspace.slug) : null, activeWorkspace ? () => workspaceService.workspaceMembers(activeWorkspace.slug) : null ); @@ -124,7 +125,7 @@ const IssueDetailSidebar: React.FC = ({ name: NestedKeyOf; canSelectMultipleOptions: boolean; icon: (props: any) => JSX.Element; - options?: Array<{ label: string; value: any }>; + options?: Array<{ label: string; value: any; color?: string }>; modal: boolean; issuesList?: Array; isOpen?: boolean; @@ -140,6 +141,7 @@ const IssueDetailSidebar: React.FC = ({ options: states?.map((state) => ({ label: state.name, value: state.id, + color: state.color, })), modal: false, }, @@ -228,364 +230,411 @@ const IssueDetailSidebar: React.FC = ({ const handleCycleChange = (cycleId: string) => { if (activeWorkspace && activeProject && issueDetail) - issuesServices.addIssueToSprint(activeWorkspace.slug, activeProject.id, cycleId, { + issuesServices.addIssueToCycle(activeWorkspace.slug, activeProject.id, cycleId, { issue: issueDetail.id, }); }; return ( -
-
-

- {activeProject?.identifier}-{issueDetail?.sequence_id} -

-
- - + <> +
+
+

+ {activeProject?.identifier}-{issueDetail?.sequence_id} +

+
+ + + +
-
-
- {sidebarSections.map((section, index) => ( -
- {section.map((item) => ( -
-
- -

{item.label}

-
-
- {item.name === "target_date" ? ( - ( - { - submitChanges({ target_date: e.target.value }); - onChange(e.target.value); - }} - className="hover:bg-gray-100 border rounded-md shadow-sm px-2 py-1 cursor-pointer focus:outline-none focus:ring-1 focus:ring-indigo-500 focus:border-indigo-500 text-xs duration-300 w-full" - /> - )} - /> - ) : item.modal ? ( - ( - <> - item.setIsOpen && item.setIsOpen(false)} - onChange={(val) => { - console.log(val); - // submitChanges({ [item.name]: val }); - onChange(val); +
+ {sidebarSections.map((section, index) => ( +
+ {section.map((item) => ( +
+
+ +

{item.label}

+
+
+ {item.name === "target_date" ? ( + ( + { + submitChanges({ target_date: e.target.value }); + onChange(e.target.value); }} - issues={item?.issuesList ?? []} - title={`Select ${item.label}`} - multiple={item.canSelectMultipleOptions} - value={value} + className="hover:bg-gray-100 border rounded-md shadow-sm px-2 py-1 cursor-pointer focus:outline-none focus:ring-1 focus:ring-indigo-500 focus:border-indigo-500 text-xs duration-300 w-full" /> - - - )} - /> - ) : ( - ( - { - if (item.name === "cycle") handleCycleChange(value); - else submitChanges({ [item.name]: value }); - }} - className="flex-shrink-0" - > - {({ open }) => ( -
- - - {value - ? Array.isArray(value) - ? value - .map( - (i: any) => - item.options?.find((option) => option.value === i) - ?.label - ) - .join(", ") || item.label - : item.options?.find((option) => option.value === value) - ?.label - : "None"} - - - - - - -
- {item.options ? ( - item.options.length > 0 ? ( - item.options.map((option) => ( - - `${ - active || selected - ? "text-white bg-theme" - : "text-gray-900" - } ${ - item.label === "Priority" && "capitalize" - } cursor-pointer select-none relative p-2 rounded-md truncate` - } - value={option.value} - > - {option.label} - - )) - ) : ( -
No {item.label}s found
- ) - ) : ( - - )} -
-
-
-
- )} -
- )} - /> - )} -
-
- ))} -
- ))} -
-
-
Add new label
-
-
- - {({ open }) => ( - <> - - {watch("colour") && watch("colour") !== "" && ( - - )} - - - - - - ( - onChange(value.hex)} /> + : `Select ${item.label}`} + + )} /> - - + ) : ( + ( + { + if (item.name === "cycle") handleCycleChange(value); + else submitChanges({ [item.name]: value }); + }} + className="flex-shrink-0" + > + {({ open }) => ( +
+ + + {value + ? Array.isArray(value) + ? value + .map( + (i: any) => + item.options?.find((option) => option.value === i) + ?.label + ) + .join(", ") || item.label + : item.options?.find((option) => option.value === value) + ?.label + : "None"} + + + + + + +
+ {item.options ? ( + item.options.length > 0 ? ( + item.options.map((option) => ( + + `${ + active || selected + ? "text-white bg-theme" + : "text-gray-900" + } ${ + item.label === "Priority" && "capitalize" + } flex items-center gap-2 cursor-pointer select-none relative p-2 rounded-md truncate` + } + value={option.value} + > + {option.color && ( + + )} + {option.label} + + )) + ) : ( +
No {item.label}s found
+ ) + ) : ( + + )} +
+
+
+
+ )} +
+ )} + /> + )} +
+
+ ))} +
+ ))} +
+
+
+
+ +

Label

+
+
+
+ {issueDetail?.label_details.map((label) => ( + + // submitChanges({ + // labels_list: issueDetail?.labels_list.filter((l) => l !== label.id), + // }) + // } + > + + {label.name} + + ))} + ( + submitChanges({ labels_list: value })} + className="flex-shrink-0" + > + {({ open }) => ( + <> + Label +
+ + + Select Label + + + + + +
+ {issueLabels ? ( + issueLabels.length > 0 ? ( + issueLabels.map((label: IIssueLabels) => ( + + `${ + active || selected + ? "text-white bg-theme" + : "text-gray-900" + } flex items-center gap-2 cursor-pointer select-none relative p-2 rounded-md truncate` + } + value={label.id} + > + + {label.name} + + )) + ) : ( +
No labels found
+ ) + ) : ( + + )} +
+
+
+
+ + )} +
+ )} + /> +
+
+
+
+
- - - -
-
- -

Label

-
-
- ( - submitChanges({ labels_list: value })} - className="flex-shrink-0" - > + {createLabelForm && ( +
+
+ {({ open }) => ( <> - Label -
- + + {watch("colour") && watch("colour") !== "" && ( - {value && value.length > 0 - ? value - .map( - (i: string) => - issueLabels?.find((option) => option.id === i)?.name - ) - .join(", ") - : "None"} - - - + className="w-5 h-5 rounded" + style={{ + backgroundColor: watch("colour") ?? "green", + }} + > + )} + + - - -
- {issueLabels ? ( - issueLabels.length > 0 ? ( - issueLabels.map((label: IIssueLabels) => ( - - `${ - active || selected - ? "text-white bg-theme" - : "text-gray-900" - } flex items-center gap-2 cursor-pointer select-none relative p-2 rounded-md truncate` - } - value={label.id} - > - - {label.name} - - )) - ) : ( -
No labels found
- ) - ) : ( - - )} -
-
-
-
+ + + ( + onChange(value.hex)} + /> + )} + /> + + )} - - )} - /> -
+ +
+ + + + )}
-
+ ); }; diff --git a/apps/app/components/project/issues/issue-detail/activity/index.tsx b/apps/app/components/project/issues/issue-detail/activity/index.tsx index 2df8c5092..84085d7b1 100644 --- a/apps/app/components/project/issues/issue-detail/activity/index.tsx +++ b/apps/app/components/project/issues/issue-detail/activity/index.tsx @@ -24,12 +24,12 @@ type Props = { const activityIcons: { [key: string]: JSX.Element; } = { - state: , - priority: , - name: , - description: , - target_date: , - parent: , + state: , + priority: , + name: , + description: , + target_date: , + parent: , }; const IssueActivitySection: React.FC = ({ issueActivities, states, issues }) => { diff --git a/apps/app/components/project/issues/issue-detail/comment/IssueCommentSection.tsx b/apps/app/components/project/issues/issue-detail/comment/IssueCommentSection.tsx index 364381ce1..f2317795a 100644 --- a/apps/app/components/project/issues/issue-detail/comment/IssueCommentSection.tsx +++ b/apps/app/components/project/issues/issue-detail/comment/IssueCommentSection.tsx @@ -4,7 +4,7 @@ import { mutate } from "swr"; // react hook form import { useForm } from "react-hook-form"; // services -import issuesServices from "lib/services/issues.services"; +import issuesServices from "lib/services/issues.service"; // fetch keys import { PROJECT_ISSUES_COMMENTS } from "constants/fetch-keys"; // components @@ -71,42 +71,6 @@ const IssueCommentSection: React.FC = ({ comments, issueId, projectId, wo return (
-
-
-
-