chore: refactor and beautify issue properties (#2539)

* chore: update all issue property components

* style: issue properties
This commit is contained in:
Aaryan Khandelwal 2023-10-25 19:47:58 +05:30 committed by GitHub
parent ca2da41dd2
commit a49f00bd39
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
40 changed files with 1066 additions and 1747 deletions

View File

@ -0,0 +1,160 @@
import React, { useState } from "react";
import { usePopper } from "react-popper";
import { Combobox } from "@headlessui/react";
import { Check, ChevronDown, Search, Triangle } from "lucide-react";
// types
import { Tooltip } from "components/ui";
import { Placement } from "@popperjs/core";
// constants
import { IEstimatePoint } from "types";
type Props = {
value: number | null;
onChange: (value: number | null) => void;
estimatePoints: IEstimatePoint[] | undefined;
className?: string;
buttonClassName?: string;
optionsClassName?: string;
placement?: Placement;
hideDropdownArrow?: boolean;
disabled?: boolean;
};
export const EstimateSelect: React.FC<Props> = (props) => {
const {
value,
onChange,
estimatePoints,
className = "",
buttonClassName = "",
optionsClassName = "",
placement,
hideDropdownArrow = false,
disabled = false,
} = props;
const [query, setQuery] = useState("");
const [referenceElement, setReferenceElement] = useState<HTMLButtonElement | null>(null);
const [popperElement, setPopperElement] = useState<HTMLDivElement | null>(null);
const { styles, attributes } = usePopper(referenceElement, popperElement, {
placement: placement ?? "bottom-start",
modifiers: [
{
name: "preventOverflow",
options: {
padding: 12,
},
},
],
});
const options: { value: number | null; query: string; content: any }[] | undefined = estimatePoints?.map(
(estimate) => ({
value: estimate.key,
query: estimate.value,
content: (
<div className="flex items-center gap-2">
<Triangle className="h-3 w-3" strokeWidth={2} />
{estimate.value}
</div>
),
})
);
options?.unshift({
value: null,
query: "none",
content: (
<div className="flex items-center gap-2">
<Triangle className="h-3 w-3" strokeWidth={2} />
None
</div>
),
});
const filteredOptions =
query === "" ? options : options?.filter((option) => option.query.toLowerCase().includes(query.toLowerCase()));
const selectedEstimate = estimatePoints?.find((e) => e.key === value);
const label = (
<Tooltip tooltipHeading="Estimate" tooltipContent={selectedEstimate?.value ?? "None"} position="top">
<div className="flex items-center cursor-pointer w-full gap-2 text-custom-text-200">
<Triangle className="h-3 w-3" strokeWidth={2} />
<span className="truncate">{selectedEstimate?.value ?? "None"}</span>
</div>
</Tooltip>
);
return (
<Combobox
as="div"
className={`flex-shrink-0 text-left ${className}`}
value={value}
onChange={(val) => onChange(val as number | null)}
disabled={disabled}
>
<Combobox.Button as={React.Fragment}>
<button
ref={setReferenceElement}
type="button"
className={`flex items-center justify-between gap-1 w-full text-xs px-2.5 py-1 rounded border-[0.5px] border-custom-border-300 duration-300 focus:outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`}
>
{label}
{!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />}
</button>
</Combobox.Button>
<Combobox.Options className="fixed z-10">
<div
className={`border border-custom-border-300 px-2 py-2.5 rounded bg-custom-background-100 text-xs shadow-custom-shadow-rg focus:outline-none w-48 whitespace-nowrap my-1 ${optionsClassName}`}
ref={setPopperElement}
style={styles.popper}
{...attributes.popper}
>
<div className="flex w-full items-center justify-start rounded border border-custom-border-200 bg-custom-background-90 px-2">
<Search className="h-3.5 w-3.5 text-custom-text-300" />
<Combobox.Input
className="w-full bg-transparent py-1 px-2 text-xs text-custom-text-200 placeholder:text-custom-text-400 focus:outline-none"
value={query}
onChange={(e) => setQuery(e.target.value)}
placeholder="Search"
displayValue={(assigned: any) => assigned?.name}
/>
</div>
<div className={`mt-2 space-y-1 max-h-48 overflow-y-scroll`}>
{filteredOptions ? (
filteredOptions.length > 0 ? (
filteredOptions.map((option) => (
<Combobox.Option
key={option.value}
value={option.value}
className={({ active, selected }) =>
`flex items-center justify-between gap-2 cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active ? "bg-custom-background-80" : ""
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
}
>
{({ selected }) => (
<>
{option.content}
{selected && <Check className="h-3.5 w-3.5" />}
</>
)}
</Combobox.Option>
))
) : (
<span className="flex items-center gap-2 p-1">
<p className="text-left text-custom-text-200 ">No matching results</p>
</span>
)
) : (
<p className="text-center text-custom-text-200">Loading...</p>
)}
</div>
</div>
</Combobox.Options>
</Combobox>
);
};

View File

@ -1,3 +1,4 @@
export * from "./create-update-estimate-modal"; export * from "./create-update-estimate-modal";
export * from "./single-estimate";
export * from "./delete-estimate-modal"; export * from "./delete-estimate-modal";
export * from "./estimate-select";
export * from "./single-estimate";

View File

@ -2,7 +2,7 @@ import { Draggable } from "@hello-pangea/dnd";
// components // components
import { KanBanProperties } from "./properties"; import { KanBanProperties } from "./properties";
// types // types
import { IIssue } from "types"; import { IEstimatePoint, IIssue, IIssueLabels, IState, IUserLite } from "types";
interface IssueBlockProps { interface IssueBlockProps {
sub_group_id: string; sub_group_id: string;
@ -18,10 +18,27 @@ interface IssueBlockProps {
) => void; ) => void;
quickActions: (sub_group_by: string | null, group_by: string | null, issue: IIssue) => React.ReactNode; quickActions: (sub_group_by: string | null, group_by: string | null, issue: IIssue) => React.ReactNode;
displayProperties: any; displayProperties: any;
states: IState[] | null;
labels: IIssueLabels[] | null;
members: IUserLite[] | null;
estimates: IEstimatePoint[] | null;
} }
export const KanbanIssueBlock: React.FC<IssueBlockProps> = (props) => { export const KanbanIssueBlock: React.FC<IssueBlockProps> = (props) => {
const { sub_group_id, columnId, index, issue, isDragDisabled, handleIssues, quickActions, displayProperties } = props; const {
sub_group_id,
columnId,
index,
issue,
isDragDisabled,
handleIssues,
quickActions,
displayProperties,
states,
labels,
members,
estimates,
} = props;
const updateIssue = (sub_group_by: string | null, group_by: string | null, issueToUpdate: IIssue) => { const updateIssue = (sub_group_by: string | null, group_by: string | null, issueToUpdate: IIssue) => {
if (issueToUpdate) handleIssues(sub_group_by, group_by, issueToUpdate, "update"); if (issueToUpdate) handleIssues(sub_group_by, group_by, issueToUpdate, "update");
@ -54,7 +71,7 @@ export const KanbanIssueBlock: React.FC<IssueBlockProps> = (props) => {
{issue.project_detail.identifier}-{issue.sequence_id} {issue.project_detail.identifier}-{issue.sequence_id}
</div> </div>
)} )}
<div className="line-clamp-2 h-[40px] text-sm font-medium text-custom-text-100">{issue.name}</div> <div className="line-clamp-2 text-sm font-medium text-custom-text-100">{issue.name}</div>
<div> <div>
<KanBanProperties <KanBanProperties
sub_group_id={sub_group_id} sub_group_id={sub_group_id}
@ -62,6 +79,10 @@ export const KanbanIssueBlock: React.FC<IssueBlockProps> = (props) => {
issue={issue} issue={issue}
handleIssues={updateIssue} handleIssues={updateIssue}
display_properties={displayProperties} display_properties={displayProperties}
states={states}
labels={labels}
members={members}
estimates={estimates}
/> />
</div> </div>
</div> </div>

View File

@ -1,6 +1,6 @@
// components // components
import { KanbanIssueBlock } from "components/issues"; import { KanbanIssueBlock } from "components/issues";
import { IIssue } from "types"; import { IEstimatePoint, IIssue, IIssueLabels, IState, IUserLite } from "types";
interface IssueBlocksListProps { interface IssueBlocksListProps {
sub_group_id: string; sub_group_id: string;
@ -15,10 +15,26 @@ interface IssueBlocksListProps {
) => void; ) => void;
quickActions: (sub_group_by: string | null, group_by: string | null, issue: IIssue) => React.ReactNode; quickActions: (sub_group_by: string | null, group_by: string | null, issue: IIssue) => React.ReactNode;
display_properties: any; display_properties: any;
states: IState[] | null;
labels: IIssueLabels[] | null;
members: IUserLite[] | null;
estimates: IEstimatePoint[] | null;
} }
export const KanbanIssueBlocksList: React.FC<IssueBlocksListProps> = (props) => { export const KanbanIssueBlocksList: React.FC<IssueBlocksListProps> = (props) => {
const { sub_group_id, columnId, issues, isDragDisabled, handleIssues, quickActions, display_properties } = props; const {
sub_group_id,
columnId,
issues,
isDragDisabled,
handleIssues,
quickActions,
display_properties,
states,
labels,
members,
estimates,
} = props;
return ( return (
<> <>
@ -35,6 +51,10 @@ export const KanbanIssueBlocksList: React.FC<IssueBlocksListProps> = (props) =>
columnId={columnId} columnId={columnId}
sub_group_id={sub_group_id} sub_group_id={sub_group_id}
isDragDisabled={isDragDisabled} isDragDisabled={isDragDisabled}
states={states}
labels={labels}
members={members}
estimates={estimates}
/> />
))} ))}
</> </>

View File

@ -7,7 +7,7 @@ import { useMobxStore } from "lib/mobx/store-provider";
import { KanBanGroupByHeaderRoot } from "./headers/group-by-root"; import { KanBanGroupByHeaderRoot } from "./headers/group-by-root";
import { KanbanIssueBlocksList } from "components/issues"; import { KanbanIssueBlocksList } from "components/issues";
// types // types
import { IIssue } from "types"; import { IEstimatePoint, IIssue, IIssueLabels, IProject, IState, IUserLite } from "types";
// constants // constants
import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES, getValueFromObject } from "constants/issue"; import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES, getValueFromObject } from "constants/issue";
@ -29,6 +29,11 @@ export interface IGroupByKanBan {
display_properties: any; display_properties: any;
kanBanToggle: any; kanBanToggle: any;
handleKanBanToggle: any; handleKanBanToggle: any;
states: IState[] | null;
labels: IIssueLabels[] | null;
members: IUserLite[] | null;
priorities: any;
estimates: IEstimatePoint[] | null;
} }
const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => { const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => {
@ -45,6 +50,11 @@ const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => {
display_properties, display_properties,
kanBanToggle, kanBanToggle,
handleKanBanToggle, handleKanBanToggle,
states,
labels,
members,
priorities,
estimates,
} = props; } = props;
const verticalAlignPosition = (_list: any) => const verticalAlignPosition = (_list: any) =>
@ -93,6 +103,10 @@ const GroupByKanBan: React.FC<IGroupByKanBan> = observer((props) => {
handleIssues={handleIssues} handleIssues={handleIssues}
quickActions={quickActions} quickActions={quickActions}
display_properties={display_properties} display_properties={display_properties}
states={states}
labels={labels}
members={members}
estimates={estimates}
/> />
) : ( ) : (
isDragDisabled && ( isDragDisabled && (
@ -128,14 +142,13 @@ export interface IKanBan {
display_properties: any; display_properties: any;
kanBanToggle: any; kanBanToggle: any;
handleKanBanToggle: any; handleKanBanToggle: any;
states: IState[] | null;
states: any;
stateGroups: any; stateGroups: any;
priorities: any; priorities: any;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
projects: any; projects: IProject[] | null;
estimates: any; estimates: IEstimatePoint[] | null;
} }
export const KanBan: React.FC<IKanBan> = observer((props) => { export const KanBan: React.FC<IKanBan> = observer((props) => {
@ -176,6 +189,11 @@ export const KanBan: React.FC<IKanBan> = observer((props) => {
display_properties={display_properties} display_properties={display_properties}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
priorities={priorities}
estimates={estimates}
/> />
)} )}
@ -193,6 +211,11 @@ export const KanBan: React.FC<IKanBan> = observer((props) => {
display_properties={display_properties} display_properties={display_properties}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
priorities={priorities}
estimates={estimates}
/> />
)} )}
@ -210,6 +233,11 @@ export const KanBan: React.FC<IKanBan> = observer((props) => {
display_properties={display_properties} display_properties={display_properties}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
priorities={priorities}
estimates={estimates}
/> />
)} )}
@ -227,6 +255,11 @@ export const KanBan: React.FC<IKanBan> = observer((props) => {
display_properties={display_properties} display_properties={display_properties}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
priorities={priorities}
estimates={estimates}
/> />
)} )}
@ -244,6 +277,11 @@ export const KanBan: React.FC<IKanBan> = observer((props) => {
display_properties={display_properties} display_properties={display_properties}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
priorities={priorities}
estimates={estimates}
/> />
)} )}
@ -261,6 +299,11 @@ export const KanBan: React.FC<IKanBan> = observer((props) => {
display_properties={display_properties} display_properties={display_properties}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
priorities={priorities}
estimates={estimates}
/> />
)} )}
</div> </div>

View File

@ -1,5 +1,3 @@
export * from "./block"; export * from "./block";
export * from "./roots";
export * from "./blocks-list"; export * from "./blocks-list";
export * from "./cycle-root";
export * from "./module-root";
export * from "./root";

View File

@ -10,55 +10,66 @@ import { IssuePropertyAssignee } from "../properties/assignee";
import { IssuePropertyEstimates } from "../properties/estimates"; import { IssuePropertyEstimates } from "../properties/estimates";
import { IssuePropertyDate } from "../properties/date"; import { IssuePropertyDate } from "../properties/date";
import { Tooltip } from "@plane/ui"; import { Tooltip } from "@plane/ui";
import { IEstimatePoint, IIssue, IIssueLabels, IState, IUserLite, TIssuePriorities } from "types";
export interface IKanBanProperties { export interface IKanBanProperties {
sub_group_id: string; sub_group_id: string;
columnId: string; columnId: string;
issue: any; issue: IIssue;
handleIssues?: (sub_group_by: string | null, group_by: string | null, issue: any) => void; handleIssues: (sub_group_by: string | null, group_by: string | null, issue: IIssue) => void;
display_properties: any; display_properties: any;
states: IState[] | null;
labels: IIssueLabels[] | null;
members: IUserLite[] | null;
estimates: IEstimatePoint[] | null;
} }
export const KanBanProperties: React.FC<IKanBanProperties> = observer( export const KanBanProperties: React.FC<IKanBanProperties> = observer((props) => {
({ sub_group_id, columnId: group_id, issue, handleIssues, display_properties }) => { const {
const handleState = (id: string) => { sub_group_id,
if (handleIssues) columnId: group_id,
issue,
handleIssues,
display_properties,
states,
labels,
members,
estimates,
} = props;
const handleState = (state: IState) => {
handleIssues( handleIssues(
!sub_group_id && sub_group_id === "null" ? null : sub_group_id, !sub_group_id && sub_group_id === "null" ? null : sub_group_id,
!group_id && group_id === "null" ? null : group_id, !group_id && group_id === "null" ? null : group_id,
{ ...issue, state: id } { ...issue, state: state.id }
); );
}; };
const handlePriority = (id: string) => { const handlePriority = (value: TIssuePriorities) => {
if (handleIssues)
handleIssues( handleIssues(
!sub_group_id && sub_group_id === "null" ? null : sub_group_id, !sub_group_id && sub_group_id === "null" ? null : sub_group_id,
!group_id && group_id === "null" ? null : group_id, !group_id && group_id === "null" ? null : group_id,
{ ...issue, priority: id } { ...issue, priority: value }
); );
}; };
const handleLabel = (ids: string[]) => { const handleLabel = (ids: string[]) => {
if (handleIssues)
handleIssues( handleIssues(
!sub_group_id && sub_group_id === "null" ? null : sub_group_id, !sub_group_id && sub_group_id === "null" ? null : sub_group_id,
!group_id && group_id === "null" ? null : group_id, !group_id && group_id === "null" ? null : group_id,
{ ...issue, labels: ids } { ...issue, labels_list: ids }
); );
}; };
const handleAssignee = (ids: string[]) => { const handleAssignee = (ids: string[]) => {
if (handleIssues)
handleIssues( handleIssues(
!sub_group_id && sub_group_id === "null" ? null : sub_group_id, !sub_group_id && sub_group_id === "null" ? null : sub_group_id,
!group_id && group_id === "null" ? null : group_id, !group_id && group_id === "null" ? null : group_id,
{ ...issue, assignees: ids } { ...issue, assignees_list: ids }
); );
}; };
const handleStartDate = (date: string) => { const handleStartDate = (date: string) => {
if (handleIssues)
handleIssues( handleIssues(
!sub_group_id && sub_group_id === "null" ? null : sub_group_id, !sub_group_id && sub_group_id === "null" ? null : sub_group_id,
!group_id && group_id === "null" ? null : group_id, !group_id && group_id === "null" ? null : group_id,
@ -67,7 +78,6 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
}; };
const handleTargetDate = (date: string) => { const handleTargetDate = (date: string) => {
if (handleIssues)
handleIssues( handleIssues(
!sub_group_id && sub_group_id === "null" ? null : sub_group_id, !sub_group_id && sub_group_id === "null" ? null : sub_group_id,
!group_id && group_id === "null" ? null : group_id, !group_id && group_id === "null" ? null : group_id,
@ -75,25 +85,25 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
); );
}; };
const handleEstimate = (id: string) => { const handleEstimate = (value: number | null) => {
if (handleIssues)
handleIssues( handleIssues(
!sub_group_id && sub_group_id === "null" ? null : sub_group_id, !sub_group_id && sub_group_id === "null" ? null : sub_group_id,
!group_id && group_id === "null" ? null : group_id, !group_id && group_id === "null" ? null : group_id,
{ ...issue, estimate_point: id } { ...issue, estimate_point: value }
); );
}; };
return ( return (
<div className="relative flex gap-2 overflow-x-auto whitespace-nowrap"> <div className="flex items-center gap-2 flex-wrap whitespace-nowrap">
{/* basic properties */} {/* basic properties */}
{/* state */} {/* state */}
{display_properties && display_properties?.state && ( {display_properties && display_properties?.state && (
<IssuePropertyState <IssuePropertyState
value={issue?.state || null} value={issue?.state_detail || null}
dropdownArrow={false} onChange={handleState}
onChange={(id: string) => handleState(id)} states={states}
disabled={false} disabled={false}
hideDropdownArrow={true}
/> />
)} )}
@ -101,9 +111,9 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
{display_properties && display_properties?.priority && ( {display_properties && display_properties?.priority && (
<IssuePropertyPriority <IssuePropertyPriority
value={issue?.priority || null} value={issue?.priority || null}
dropdownArrow={false} onChange={handlePriority}
onChange={(id: string) => handlePriority(id)}
disabled={false} disabled={false}
hideDropdownArrow={true}
/> />
)} )}
@ -111,9 +121,10 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
{display_properties && display_properties?.labels && ( {display_properties && display_properties?.labels && (
<IssuePropertyLabels <IssuePropertyLabels
value={issue?.labels || null} value={issue?.labels || null}
dropdownArrow={false} onChange={handleLabel}
onChange={(ids: string[]) => handleLabel(ids)} labels={labels}
disabled={false} disabled={false}
hideDropdownArrow={true}
/> />
)} )}
@ -121,8 +132,9 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
{display_properties && display_properties?.assignee && ( {display_properties && display_properties?.assignee && (
<IssuePropertyAssignee <IssuePropertyAssignee
value={issue?.assignees || null} value={issue?.assignees || null}
dropdownArrow={false} hideDropdownArrow={true}
onChange={(ids: string[]) => handleAssignee(ids)} onChange={handleAssignee}
members={members}
disabled={false} disabled={false}
/> />
)} )}
@ -133,6 +145,7 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
value={issue?.start_date || null} value={issue?.start_date || null}
onChange={(date: string) => handleStartDate(date)} onChange={(date: string) => handleStartDate(date)}
disabled={false} disabled={false}
placeHolder="Start date"
/> />
)} )}
@ -142,30 +155,28 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
value={issue?.target_date || null} value={issue?.target_date || null}
onChange={(date: string) => handleTargetDate(date)} onChange={(date: string) => handleTargetDate(date)}
disabled={false} disabled={false}
placeHolder="Target date"
/> />
)} )}
{/* estimates */} {/* estimates */}
{display_properties && display_properties?.estimate && ( {display_properties && display_properties?.estimate && (
<IssuePropertyEstimates <IssuePropertyEstimates
value={issue?.estimate_point?.toString() || null} value={issue?.estimate_point || null}
dropdownArrow={false} onChange={handleEstimate}
onChange={(id: string) => handleEstimate(id)} estimatePoints={estimates}
disabled={false} disabled={false}
workspaceSlug={issue?.workspace_detail?.slug || null} hideDropdownArrow={true}
projectId={issue?.project_detail?.id || null}
/> />
)} )}
{/* extra render properties */} {/* extra render properties */}
{/* sub-issues */} {/* sub-issues */}
{display_properties && display_properties?.sub_issue_count && ( {display_properties && display_properties?.sub_issue_count && (
<Tooltip tooltipHeading="Sub-issue" tooltipContent={`${issue.sub_issues_count}`}> <Tooltip tooltipHeading="Sub-issues" tooltipContent={`${issue.sub_issues_count}`}>
<div className="flex-shrink-0 border border-custom-border-300 min-w-[22px] h-[22px] overflow-hidden rounded-sm flex justify-center items-center cursor-pointer"> <div className="flex-shrink-0 border-[0.5px] border-custom-border-300 overflow-hidden rounded flex justify-center items-center gap-2 px-2.5 py-1 h-5">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center"> <Layers className="h-3 w-3 flex-shrink-0" strokeWidth={2} />
<Layers width={10} strokeWidth={2} /> <div className="text-xs">{issue.sub_issues_count}</div>
</div>
<div className="pl-0.5 pr-1 text-xs">{issue.sub_issues_count}</div>
</div> </div>
</Tooltip> </Tooltip>
)} )}
@ -173,11 +184,9 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
{/* attachments */} {/* attachments */}
{display_properties && display_properties?.attachment_count && ( {display_properties && display_properties?.attachment_count && (
<Tooltip tooltipHeading="Attachments" tooltipContent={`${issue.attachment_count}`}> <Tooltip tooltipHeading="Attachments" tooltipContent={`${issue.attachment_count}`}>
<div className="flex-shrink-0 border border-custom-border-300 min-w-[22px] h-[22px] overflow-hidden rounded-sm flex justify-center items-center cursor-pointer"> <div className="flex-shrink-0 border-[0.5px] border-custom-border-300 overflow-hidden rounded flex justify-center items-center gap-2 px-2.5 py-1 h-5">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center"> <Paperclip className="h-3 w-3 flex-shrink-0" strokeWidth={2} />
<Paperclip width={10} strokeWidth={2} /> <div className="text-xs">{issue.attachment_count}</div>
</div>
<div className="pl-0.5 pr-1 text-xs">{issue.attachment_count}</div>
</div> </div>
</Tooltip> </Tooltip>
)} )}
@ -185,15 +194,12 @@ export const KanBanProperties: React.FC<IKanBanProperties> = observer(
{/* link */} {/* link */}
{display_properties && display_properties?.link && ( {display_properties && display_properties?.link && (
<Tooltip tooltipHeading="Links" tooltipContent={`${issue.link_count}`}> <Tooltip tooltipHeading="Links" tooltipContent={`${issue.link_count}`}>
<div className="flex-shrink-0 border border-custom-border-300 min-w-[22px] h-[22px] overflow-hidden rounded-sm flex justify-center items-center cursor-pointer"> <div className="flex-shrink-0 border-[0.5px] border-custom-border-300 overflow-hidden rounded flex justify-center items-center gap-2 px-2.5 py-1 h-5">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center"> <Link className="h-3 w-3 flex-shrink-0" strokeWidth={2} />
<Link width={10} strokeWidth={2} /> <div className="text-xs">{issue.link_count}</div>
</div>
<div className="pl-0.5 pr-1 text-xs">{issue.link_count}</div>
</div> </div>
</Tooltip> </Tooltip>
)} )}
</div> </div>
); );
} });
);

View File

@ -5,9 +5,11 @@ import { DragDropContext } from "@hello-pangea/dnd";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { KanBanSwimLanes } from "./swimlanes"; import { KanBanSwimLanes } from "../swimlanes";
import { KanBan } from "./default"; import { KanBan } from "../default";
import { CycleIssueQuickActions } from "components/issues"; import { CycleIssueQuickActions } from "components/issues";
// helpers
import { orderArrayBy } from "helpers/array.helper";
// types // types
import { IIssue } from "types"; import { IIssue } from "types";
// constants // constants
@ -25,7 +27,7 @@ export const CycleKanBanLayout: React.FC = observer(() => {
} = useMobxStore(); } = useMobxStore();
const router = useRouter(); const router = useRouter();
const { workspaceSlug, cycleId } = router.query; const { workspaceSlug, projectId, cycleId } = router.query;
const issues = cycleIssueStore?.getIssues; const issues = cycleIssueStore?.getIssues;
@ -60,12 +62,12 @@ export const CycleKanBanLayout: React.FC = observer(() => {
if (!workspaceSlug || !cycleId) return; if (!workspaceSlug || !cycleId) return;
if (action === "update") { if (action === "update") {
cycleIssueStore.updateIssueStructure(group_by, null, issue); cycleIssueStore.updateIssueStructure(group_by, sub_group_by, issue);
issueDetailStore.updateIssue(workspaceSlug.toString(), issue.project, issue.id, issue); issueDetailStore.updateIssue(workspaceSlug.toString(), issue.project, issue.id, issue);
} }
if (action === "delete") cycleIssueStore.deleteIssue(group_by, null, issue); if (action === "delete") cycleIssueStore.deleteIssue(group_by, sub_group_by, issue);
if (action === "remove" && issue.bridge_id) { if (action === "remove" && issue.bridge_id) {
cycleIssueStore.deleteIssue(group_by, null, issue); cycleIssueStore.deleteIssue(group_by, sub_group_by, issue);
cycleIssueStore.removeIssueFromCycle( cycleIssueStore.removeIssueFromCycle(
workspaceSlug.toString(), workspaceSlug.toString(),
issue.project, issue.project,
@ -81,13 +83,18 @@ export const CycleKanBanLayout: React.FC = observer(() => {
cycleIssueKanBanViewStore.handleKanBanToggle(toggle, value); cycleIssueKanBanViewStore.handleKanBanToggle(toggle, value);
}; };
const projectDetails = projectId ? projectStore.project_details[projectId.toString()] : null;
const states = projectStore?.projectStates || null; const states = projectStore?.projectStates || null;
const priorities = ISSUE_PRIORITIES || null; const priorities = ISSUE_PRIORITIES || null;
const labels = projectStore?.projectLabels || null; const labels = projectStore?.projectLabels || null;
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.projectStates || null; const projects = workspaceSlug ? projectStore?.projects[workspaceSlug.toString()] || null : null;
const estimates = null; const estimates =
projectDetails?.estimate !== null
? projectStore.projectEstimates?.find((e) => e.id === projectDetails?.estimate) || null
: null;
return ( return (
<div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}> <div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}>
@ -113,9 +120,9 @@ export const CycleKanBanLayout: React.FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
) : ( ) : (
<KanBanSwimLanes <KanBanSwimLanes
@ -138,9 +145,9 @@ export const CycleKanBanLayout: React.FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
)} )}
</DragDropContext> </DragDropContext>

View File

@ -0,0 +1,5 @@
export * from "./cycle-root";
export * from "./module-root";
export * from "./profile-issues-root";
export * from "./project-root";
export * from "./project-view-root";

View File

@ -5,9 +5,11 @@ import { DragDropContext } from "@hello-pangea/dnd";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { KanBanSwimLanes } from "./swimlanes"; import { KanBanSwimLanes } from "../swimlanes";
import { KanBan } from "./default"; import { KanBan } from "../default";
import { ModuleIssueQuickActions } from "components/issues"; import { ModuleIssueQuickActions } from "components/issues";
// helpers
import { orderArrayBy } from "helpers/array.helper";
// types // types
import { IIssue } from "types"; import { IIssue } from "types";
// constants // constants
@ -25,7 +27,7 @@ export const ModuleKanBanLayout: React.FC = observer(() => {
} = useMobxStore(); } = useMobxStore();
const router = useRouter(); const router = useRouter();
const { workspaceSlug, moduleId } = router.query; const { workspaceSlug, projectId, moduleId } = router.query;
const issues = moduleIssueStore?.getIssues; const issues = moduleIssueStore?.getIssues;
@ -81,13 +83,18 @@ export const ModuleKanBanLayout: React.FC = observer(() => {
moduleIssueKanBanViewStore.handleKanBanToggle(toggle, value); moduleIssueKanBanViewStore.handleKanBanToggle(toggle, value);
}; };
const projectDetails = projectId ? projectStore.project_details[projectId.toString()] : null;
const states = projectStore?.projectStates || null; const states = projectStore?.projectStates || null;
const priorities = ISSUE_PRIORITIES || null; const priorities = ISSUE_PRIORITIES || null;
const labels = projectStore?.projectLabels || null; const labels = projectStore?.projectLabels || null;
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.projectStates || null; const projects = workspaceSlug ? projectStore?.projects[workspaceSlug.toString()] || null : null;
const estimates = null; const estimates =
projectDetails?.estimate !== null
? projectStore.projectEstimates?.find((e) => e.id === projectDetails?.estimate) || null
: null;
return ( return (
<div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}> <div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}>
@ -113,9 +120,9 @@ export const ModuleKanBanLayout: React.FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
) : ( ) : (
<KanBanSwimLanes <KanBanSwimLanes
@ -138,9 +145,9 @@ export const ModuleKanBanLayout: React.FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
)} )}
</DragDropContext> </DragDropContext>

View File

@ -5,8 +5,8 @@ import { DragDropContext } from "@hello-pangea/dnd";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { KanBanSwimLanes } from "./swimlanes"; import { KanBanSwimLanes } from "../swimlanes";
import { KanBan } from "./default"; import { KanBan } from "../default";
import { ProjectIssueQuickActions } from "components/issues"; import { ProjectIssueQuickActions } from "components/issues";
// constants // constants
import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES } from "constants/issue"; import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES } from "constants/issue";
@ -79,7 +79,6 @@ export const ProfileIssuesKanBanLayout: FC = observer(() => {
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.workspaceProjects || null; const projects = projectStore?.workspaceProjects || null;
const estimates = null;
return ( return (
<div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}> <div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}>
@ -104,9 +103,9 @@ export const ProfileIssuesKanBanLayout: FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={null}
/> />
) : ( ) : (
<KanBanSwimLanes <KanBanSwimLanes
@ -128,9 +127,9 @@ export const ProfileIssuesKanBanLayout: FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={null}
/> />
)} )}
</DragDropContext> </DragDropContext>

View File

@ -1,13 +1,15 @@
import { FC, useCallback } from "react"; import { useCallback } from "react";
import { useRouter } from "next/router"; import { useRouter } from "next/router";
import { DragDropContext } from "@hello-pangea/dnd"; import { DragDropContext } from "@hello-pangea/dnd";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { KanBanSwimLanes } from "./swimlanes"; import { KanBanSwimLanes } from "../swimlanes";
import { KanBan } from "./default"; import { KanBan } from "../default";
import { ProjectIssueQuickActions } from "components/issues"; import { ProjectIssueQuickActions } from "components/issues";
// helpers
import { orderArrayBy } from "helpers/array.helper";
// types // types
import { IIssue } from "types"; import { IIssue } from "types";
// constants // constants
@ -15,9 +17,9 @@ import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES } from "constants/issue";
export interface IKanBanLayout {} export interface IKanBanLayout {}
export const KanBanLayout: FC = observer(() => { export const KanBanLayout: React.FC = observer(() => {
const router = useRouter(); const router = useRouter();
const { workspaceSlug } = router.query; const { workspaceSlug, projectId } = router.query;
const { const {
project: projectStore, project: projectStore,
@ -72,13 +74,18 @@ export const KanBanLayout: FC = observer(() => {
issueKanBanViewStore.handleKanBanToggle(toggle, value); issueKanBanViewStore.handleKanBanToggle(toggle, value);
}; };
const projectDetails = projectId ? projectStore.project_details[projectId.toString()] : null;
const states = projectStore?.projectStates || null; const states = projectStore?.projectStates || null;
const priorities = ISSUE_PRIORITIES || null; const priorities = ISSUE_PRIORITIES || null;
const labels = projectStore?.projectLabels || null; const labels = projectStore?.projectLabels || null;
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.projectStates || null; const projects = workspaceSlug ? projectStore?.projects[workspaceSlug.toString()] || null : null;
const estimates = null; const estimates =
projectDetails?.estimate !== null
? projectStore.projectEstimates?.find((e) => e.id === projectDetails?.estimate) || null
: null;
return ( return (
<div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}> <div className={`relative min-w-full w-max min-h-full h-max bg-custom-background-90 px-3`}>
@ -103,9 +110,9 @@ export const KanBanLayout: FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
) : ( ) : (
<KanBanSwimLanes <KanBanSwimLanes
@ -127,9 +134,9 @@ export const KanBanLayout: FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
)} )}
</DragDropContext> </DragDropContext>

View File

@ -4,8 +4,8 @@ import { DragDropContext } from "@hello-pangea/dnd";
// mobx // mobx
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { KanBanSwimLanes } from "./swimlanes"; import { KanBanSwimLanes } from "../swimlanes";
import { KanBan } from "./default"; import { KanBan } from "../default";
// store // store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";
@ -14,7 +14,7 @@ import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES } from "constants/issue";
export interface IViewKanBanLayout {} export interface IViewKanBanLayout {}
export const ViewKanBanLayout: React.FC = observer(() => { export const ProjectViewKanBanLayout: React.FC = observer(() => {
const { const {
project: projectStore, project: projectStore,
issue: issueStore, issue: issueStore,

View File

@ -7,7 +7,7 @@ import { KanBanGroupByHeaderRoot } from "./headers/group-by-root";
import { KanBanSubGroupByHeaderRoot } from "./headers/sub-group-by-root"; import { KanBanSubGroupByHeaderRoot } from "./headers/sub-group-by-root";
import { KanBan } from "./default"; import { KanBan } from "./default";
// types // types
import { IIssue } from "types"; import { IEstimatePoint, IIssue, IIssueLabels, IProject, IState, IUserLite } from "types";
// constants // constants
import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES, getValueFromObject } from "constants/issue"; import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES, getValueFromObject } from "constants/issue";
@ -19,6 +19,11 @@ interface ISubGroupSwimlaneHeader {
listKey: string; listKey: string;
kanBanToggle: any; kanBanToggle: any;
handleKanBanToggle: any; handleKanBanToggle: any;
states: IState[] | null;
labels: IIssueLabels[] | null;
members: IUserLite[] | null;
projects: IProject[] | null;
estimates: IEstimatePoint[] | null;
} }
const SubGroupSwimlaneHeader: React.FC<ISubGroupSwimlaneHeader> = ({ const SubGroupSwimlaneHeader: React.FC<ISubGroupSwimlaneHeader> = ({
issues, issues,
@ -71,13 +76,13 @@ interface ISubGroupSwimlane extends ISubGroupSwimlaneHeader {
display_properties: any; display_properties: any;
kanBanToggle: any; kanBanToggle: any;
handleKanBanToggle: any; handleKanBanToggle: any;
states: any; states: IState[] | null;
stateGroups: any; stateGroups: any;
priorities: any; priorities: any;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
projects: any; projects: IProject[] | null;
estimates: any; estimates: IEstimatePoint[] | null;
} }
const SubGroupSwimlane: React.FC<ISubGroupSwimlane> = observer((props) => { const SubGroupSwimlane: React.FC<ISubGroupSwimlane> = observer((props) => {
const { const {
@ -171,13 +176,13 @@ export interface IKanBanSwimLanes {
display_properties: any; display_properties: any;
kanBanToggle: any; kanBanToggle: any;
handleKanBanToggle: any; handleKanBanToggle: any;
states: any; states: IState[] | null;
stateGroups: any; stateGroups: any;
priorities: any; priorities: any;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
projects: any; projects: IProject[] | null;
estimates: any; estimates: IEstimatePoint[] | null;
} }
export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => { export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => {
@ -213,6 +218,11 @@ export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => {
listKey={`id`} listKey={`id`}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
projects={projects}
estimates={estimates}
/> />
)} )}
@ -225,6 +235,11 @@ export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => {
listKey={`key`} listKey={`key`}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
projects={projects}
estimates={estimates}
/> />
)} )}
@ -237,6 +252,11 @@ export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => {
listKey={`key`} listKey={`key`}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
projects={projects}
estimates={estimates}
/> />
)} )}
@ -249,6 +269,11 @@ export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => {
listKey={`id`} listKey={`id`}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
projects={projects}
estimates={estimates}
/> />
)} )}
@ -261,6 +286,11 @@ export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => {
listKey={`member.id`} listKey={`member.id`}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
projects={projects}
estimates={estimates}
/> />
)} )}
@ -273,6 +303,11 @@ export const KanBanSwimLanes: React.FC<IKanBanSwimLanes> = observer((props) => {
listKey={`member.id`} listKey={`member.id`}
kanBanToggle={kanBanToggle} kanBanToggle={kanBanToggle}
handleKanBanToggle={handleKanBanToggle} handleKanBanToggle={handleKanBanToggle}
states={states}
labels={labels}
members={members}
projects={projects}
estimates={estimates}
/> />
)} )}
</div> </div>

View File

@ -4,7 +4,7 @@ import { IssuePeekOverview } from "components/issues/issue-peek-overview";
// ui // ui
import { Tooltip } from "@plane/ui"; import { Tooltip } from "@plane/ui";
// types // types
import { IIssue } from "types"; import { IEstimatePoint, IIssue, IIssueLabels, IState, IUserLite } from "types";
interface IssueBlockProps { interface IssueBlockProps {
columnId: string; columnId: string;
@ -12,18 +12,17 @@ interface IssueBlockProps {
handleIssues: (group_by: string | null, issue: IIssue, action: "update" | "delete") => void; handleIssues: (group_by: string | null, issue: IIssue, action: "update" | "delete") => void;
quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode; quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode;
display_properties: any; display_properties: any;
states: any; states: IState[] | null;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
priorities: any; estimates: IEstimatePoint[] | null;
} }
export const IssueBlock: React.FC<IssueBlockProps> = (props) => { export const IssueBlock: React.FC<IssueBlockProps> = (props) => {
const { columnId, issue, handleIssues, quickActions, display_properties, states, labels, members, priorities } = const { columnId, issue, handleIssues, quickActions, display_properties, states, labels, members, estimates } = props;
props;
const updateIssue = (group_by: string | null, issueToUpdate: IIssue) => { const updateIssue = (group_by: string | null, issueToUpdate: IIssue) => {
if (issueToUpdate && handleIssues) handleIssues(group_by, issueToUpdate, "update"); handleIssues(group_by, issueToUpdate, "update");
}; };
return ( return (
@ -55,7 +54,7 @@ export const IssueBlock: React.FC<IssueBlockProps> = (props) => {
states={states} states={states}
labels={labels} labels={labels}
members={members} members={members}
priorities={priorities} estimates={estimates}
/> />
{quickActions(!columnId && columnId === "null" ? null : columnId, issue)} {quickActions(!columnId && columnId === "null" ? null : columnId, issue)}
</div> </div>

View File

@ -2,7 +2,7 @@ import { FC } from "react";
// components // components
import { IssueBlock } from "components/issues"; import { IssueBlock } from "components/issues";
// types // types
import { IIssue } from "types"; import { IEstimatePoint, IIssue, IIssueLabels, IState, IUserLite } from "types";
interface Props { interface Props {
columnId: string; columnId: string;
@ -10,14 +10,14 @@ interface Props {
handleIssues: (group_by: string | null, issue: IIssue, action: "update" | "delete") => void; handleIssues: (group_by: string | null, issue: IIssue, action: "update" | "delete") => void;
quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode; quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode;
display_properties: any; display_properties: any;
states: any; states: IState[] | null;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
priorities: any; estimates: IEstimatePoint[] | null;
} }
export const IssueBlocksList: FC<Props> = (props) => { export const IssueBlocksList: FC<Props> = (props) => {
const { columnId, issues, handleIssues, quickActions, display_properties, states, labels, members, priorities } = const { columnId, issues, handleIssues, quickActions, display_properties, states, labels, members, estimates } =
props; props;
return ( return (
@ -35,7 +35,7 @@ export const IssueBlocksList: FC<Props> = (props) => {
states={states} states={states}
labels={labels} labels={labels}
members={members} members={members}
priorities={priorities} estimates={estimates}
/> />
))} ))}
</> </>

View File

@ -2,11 +2,11 @@ import React from "react";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { ListGroupByHeaderRoot } from "./headers/group-by-root"; import { ListGroupByHeaderRoot } from "./headers/group-by-root";
import { IssueBlock } from "./block"; import { IssueBlocksList } from "components/issues";
// types
import { IEstimatePoint, IIssue, IIssueLabels, IProject, IState, IUserLite } from "types";
// constants // constants
import { getValueFromObject } from "constants/issue"; import { getValueFromObject } from "constants/issue";
import { IIssue } from "types";
import { IssueBlocksList } from "./blocks-list";
export interface IGroupByList { export interface IGroupByList {
issues: any; issues: any;
@ -17,13 +17,13 @@ export interface IGroupByList {
quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode; quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode;
display_properties: any; display_properties: any;
is_list?: boolean; is_list?: boolean;
states: any; states: IState[] | null;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
projects: any; projects: IProject[] | null;
stateGroups: any; stateGroups: any;
priorities: any; priorities: any;
estimates: any; estimates: IEstimatePoint[] | null;
} }
const GroupByList: React.FC<IGroupByList> = observer((props) => { const GroupByList: React.FC<IGroupByList> = observer((props) => {
@ -72,7 +72,7 @@ const GroupByList: React.FC<IGroupByList> = observer((props) => {
states={states} states={states}
labels={labels} labels={labels}
members={members} members={members}
priorities={priorities} estimates={estimates}
/> />
)} )}
</div> </div>
@ -90,13 +90,13 @@ export interface IList {
handleIssues: (group_by: string | null, issue: IIssue, action: "update" | "delete") => void; handleIssues: (group_by: string | null, issue: IIssue, action: "update" | "delete") => void;
quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode; quickActions: (group_by: string | null, issue: IIssue) => React.ReactNode;
display_properties: any; display_properties: any;
states: any; states: IState[] | null;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
projects: any; projects: IProject[] | null;
stateGroups: any; stateGroups: any;
priorities: any; priorities: any;
estimates: any; estimates: IEstimatePoint[] | null;
} }
export const List: React.FC<IList> = observer((props) => { export const List: React.FC<IList> = observer((props) => {

View File

@ -1,5 +1,3 @@
export * from "./roots";
export * from "./block"; export * from "./block";
export * from "./blocks-list"; export * from "./blocks-list";
export * from "./cycle-root";
export * from "./module-root";
export * from "./root";

View File

@ -11,49 +11,48 @@ import { IssuePropertyDate } from "../properties/date";
// ui // ui
import { Tooltip } from "@plane/ui"; import { Tooltip } from "@plane/ui";
// types // types
import { IIssue } from "types"; import { IEstimatePoint, IIssue, IIssueLabels, IState, IUserLite, TIssuePriorities } from "types";
export interface IKanBanProperties { export interface IKanBanProperties {
columnId: string; columnId: string;
issue: any; issue: IIssue;
handleIssues?: (group_by: string | null, issue: IIssue) => void; handleIssues: (group_by: string | null, issue: IIssue) => void;
display_properties: any; display_properties: any;
states: any; states: IState[] | null;
labels: any; labels: IIssueLabels[] | null;
members: any; members: IUserLite[] | null;
priorities: any; estimates: IEstimatePoint[] | null;
} }
export const KanBanProperties: FC<IKanBanProperties> = observer((props) => { export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
const { columnId: group_id, issue, handleIssues, display_properties, states, labels, members, priorities } = props; const { columnId: group_id, issue, handleIssues, display_properties, states, labels, members, estimates } = props;
const handleState = (id: string) => { const handleState = (state: IState) => {
if (handleIssues) handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, state: id }); handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, state: state.id });
}; };
const handlePriority = (id: string) => { const handlePriority = (value: TIssuePriorities) => {
if (handleIssues) handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, priority: id }); handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, priority: value });
}; };
const handleLabel = (ids: string[]) => { const handleLabel = (ids: string[]) => {
if (handleIssues) handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, labels: ids }); handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, labels_list: ids });
}; };
const handleAssignee = (ids: string[]) => { const handleAssignee = (ids: string[]) => {
if (handleIssues) handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, assignees: ids }); handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, assignees_list: ids });
}; };
const handleStartDate = (date: string) => { const handleStartDate = (date: string) => {
if (handleIssues) handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, start_date: date }); handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, start_date: date });
}; };
const handleTargetDate = (date: string) => { const handleTargetDate = (date: string) => {
if (handleIssues) handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, target_date: date }); handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, target_date: date });
}; };
const handleEstimate = (id: string) => { const handleEstimate = (value: number | null) => {
if (handleIssues) handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, estimate_point: value });
handleIssues(!group_id && group_id === "null" ? null : group_id, { ...issue, estimate_point: id });
}; };
return ( return (
@ -62,22 +61,21 @@ export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
{/* state */} {/* state */}
{display_properties && display_properties?.state && states && ( {display_properties && display_properties?.state && states && (
<IssuePropertyState <IssuePropertyState
value={issue?.state || null} value={issue?.state_detail || null}
dropdownArrow={false} hideDropdownArrow={true}
onChange={(id: string) => handleState(id)} onChange={handleState}
disabled={false} disabled={false}
list={states} states={states}
/> />
)} )}
{/* priority */} {/* priority */}
{display_properties && display_properties?.priority && priorities && ( {display_properties && display_properties?.priority && (
<IssuePropertyPriority <IssuePropertyPriority
value={issue?.priority || null} value={issue?.priority || null}
dropdownArrow={false} onChange={handlePriority}
onChange={(id: string) => handlePriority(id)}
disabled={false} disabled={false}
list={priorities} hideDropdownArrow={true}
/> />
)} )}
@ -85,10 +83,10 @@ export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
{display_properties && display_properties?.labels && labels && ( {display_properties && display_properties?.labels && labels && (
<IssuePropertyLabels <IssuePropertyLabels
value={issue?.labels || null} value={issue?.labels || null}
dropdownArrow={false} onChange={handleLabel}
onChange={(ids: string[]) => handleLabel(ids)} labels={labels}
disabled={false} disabled={false}
list={labels} hideDropdownArrow={true}
/> />
)} )}
@ -96,10 +94,10 @@ export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
{display_properties && display_properties?.assignee && members && ( {display_properties && display_properties?.assignee && members && (
<IssuePropertyAssignee <IssuePropertyAssignee
value={issue?.assignees || null} value={issue?.assignees || null}
dropdownArrow={false} hideDropdownArrow={true}
onChange={(ids: string[]) => handleAssignee(ids)} onChange={handleAssignee}
disabled={false} disabled={false}
list={members} members={members}
/> />
)} )}
@ -109,7 +107,7 @@ export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
value={issue?.start_date || null} value={issue?.start_date || null}
onChange={(date: string) => handleStartDate(date)} onChange={(date: string) => handleStartDate(date)}
disabled={false} disabled={false}
placeHolder={`Start date`} placeHolder="Start date"
/> />
)} )}
@ -119,31 +117,28 @@ export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
value={issue?.target_date || null} value={issue?.target_date || null}
onChange={(date: string) => handleTargetDate(date)} onChange={(date: string) => handleTargetDate(date)}
disabled={false} disabled={false}
placeHolder={`Target date`} placeHolder="Target date"
/> />
)} )}
{/* estimates */} {/* estimates */}
{display_properties && display_properties?.estimate && ( {display_properties && display_properties?.estimate && (
<IssuePropertyEstimates <IssuePropertyEstimates
value={issue?.estimate_point?.toString() || null} value={issue?.estimate_point || null}
dropdownArrow={false} estimatePoints={estimates}
onChange={(id: string) => handleEstimate(id)} hideDropdownArrow={true}
onChange={handleEstimate}
disabled={false} disabled={false}
workspaceSlug={issue?.workspace_detail?.slug || null}
projectId={issue?.project_detail?.id || null}
/> />
)} )}
{/* extra render properties */} {/* extra render properties */}
{/* sub-issues */} {/* sub-issues */}
{display_properties && display_properties?.sub_issue_count && ( {display_properties && display_properties?.sub_issue_count && (
<Tooltip tooltipHeading="Sub-issue" tooltipContent={`${issue.sub_issues_count}`}> <Tooltip tooltipHeading="Sub-issues" tooltipContent={`${issue.sub_issues_count}`}>
<div className="flex-shrink-0 border border-custom-border-300 min-w-[22px] h-[22px] overflow-hidden rounded-sm flex justify-center items-center cursor-pointer"> <div className="flex-shrink-0 border-[0.5px] border-custom-border-300 overflow-hidden rounded flex justify-center items-center gap-2 px-2.5 py-1 h-5">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center"> <Layers className="h-3 w-3 flex-shrink-0" strokeWidth={2} />
<Layers width={10} strokeWidth={2} /> <div className="text-xs">{issue.sub_issues_count}</div>
</div>
<div className="pl-0.5 pr-1 text-xs">{issue.sub_issues_count}</div>
</div> </div>
</Tooltip> </Tooltip>
)} )}
@ -151,11 +146,9 @@ export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
{/* attachments */} {/* attachments */}
{display_properties && display_properties?.attachment_count && ( {display_properties && display_properties?.attachment_count && (
<Tooltip tooltipHeading="Attachments" tooltipContent={`${issue.attachment_count}`}> <Tooltip tooltipHeading="Attachments" tooltipContent={`${issue.attachment_count}`}>
<div className="flex-shrink-0 border border-custom-border-300 min-w-[22px] h-[22px] overflow-hidden rounded-sm flex justify-center items-center cursor-pointer"> <div className="flex-shrink-0 border-[0.5px] border-custom-border-300 overflow-hidden rounded flex justify-center items-center gap-2 px-2.5 py-1 h-5">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center"> <Paperclip className="h-3 w-3 flex-shrink-0" strokeWidth={2} />
<Paperclip width={10} strokeWidth={2} /> <div className="text-xs">{issue.attachment_count}</div>
</div>
<div className="pl-0.5 pr-1 text-xs">{issue.attachment_count}</div>
</div> </div>
</Tooltip> </Tooltip>
)} )}
@ -163,11 +156,9 @@ export const KanBanProperties: FC<IKanBanProperties> = observer((props) => {
{/* link */} {/* link */}
{display_properties && display_properties?.link && ( {display_properties && display_properties?.link && (
<Tooltip tooltipHeading="Links" tooltipContent={`${issue.link_count}`}> <Tooltip tooltipHeading="Links" tooltipContent={`${issue.link_count}`}>
<div className="flex-shrink-0 border border-custom-border-300 min-w-[22px] h-[22px] overflow-hidden rounded-sm flex justify-center items-center cursor-pointer"> <div className="flex-shrink-0 border-[0.5px] border-custom-border-300 overflow-hidden rounded flex justify-center items-center gap-2 px-2.5 py-1 h-5">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center"> <Link className="h-3 w-3 flex-shrink-0" strokeWidth={2} />
<Link width={10} strokeWidth={2} /> <div className="text-xs">{issue.link_count}</div>
</div>
<div className="pl-0.5 pr-1 text-xs">{issue.link_count}</div>
</div> </div>
</Tooltip> </Tooltip>
)} )}

View File

@ -4,8 +4,10 @@ import { observer } from "mobx-react-lite";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { List } from "./default"; import { List } from "../default";
import { CycleIssueQuickActions } from "components/issues"; import { CycleIssueQuickActions } from "components/issues";
// helpers
import { orderArrayBy } from "helpers/array.helper";
// types // types
import { IIssue } from "types"; import { IIssue } from "types";
// constants // constants
@ -15,7 +17,7 @@ export interface ICycleListLayout {}
export const CycleListLayout: React.FC = observer(() => { export const CycleListLayout: React.FC = observer(() => {
const router = useRouter(); const router = useRouter();
const { workspaceSlug, cycleId } = router.query; const { workspaceSlug, projectId, cycleId } = router.query;
const { const {
project: projectStore, project: projectStore,
@ -52,13 +54,18 @@ export const CycleListLayout: React.FC = observer(() => {
[cycleIssueStore, issueDetailStore, cycleId, workspaceSlug] [cycleIssueStore, issueDetailStore, cycleId, workspaceSlug]
); );
const projectDetails = projectId ? projectStore.project_details[projectId.toString()] : null;
const states = projectStore?.projectStates || null; const states = projectStore?.projectStates || null;
const priorities = ISSUE_PRIORITIES || null; const priorities = ISSUE_PRIORITIES || null;
const labels = projectStore?.projectLabels || null; const labels = projectStore?.projectLabels || null;
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.projectStates || null; const projects = workspaceSlug ? projectStore?.projects[workspaceSlug.toString()] || null : null;
const estimates = null; const estimates =
projectDetails?.estimate !== null
? projectStore.projectEstimates?.find((e) => e.id === projectDetails?.estimate) || null
: null;
return ( return (
<div className={`relative w-full h-full bg-custom-background-90`}> <div className={`relative w-full h-full bg-custom-background-90`}>
@ -79,9 +86,9 @@ export const CycleListLayout: React.FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
</div> </div>
); );

View File

@ -0,0 +1,5 @@
export * from "./cycle-root";
export * from "./module-root";
export * from "./profile-issues-root";
export * from "./project-root";
export * from "./project-view-root";

View File

@ -4,8 +4,10 @@ import { observer } from "mobx-react-lite";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { List } from "./default"; import { List } from "../default";
import { ModuleIssueQuickActions } from "components/issues"; import { ModuleIssueQuickActions } from "components/issues";
// helpers
import { orderArrayBy } from "helpers/array.helper";
// types // types
import { IIssue } from "types"; import { IIssue } from "types";
// constants // constants
@ -15,7 +17,7 @@ export interface IModuleListLayout {}
export const ModuleListLayout: React.FC = observer(() => { export const ModuleListLayout: React.FC = observer(() => {
const router = useRouter(); const router = useRouter();
const { workspaceSlug, moduleId } = router.query; const { workspaceSlug, projectId, moduleId } = router.query;
const { const {
project: projectStore, project: projectStore,
@ -52,13 +54,18 @@ export const ModuleListLayout: React.FC = observer(() => {
[moduleIssueStore, issueDetailStore, moduleId, workspaceSlug] [moduleIssueStore, issueDetailStore, moduleId, workspaceSlug]
); );
const projectDetails = projectId ? projectStore.project_details[projectId.toString()] : null;
const states = projectStore?.projectStates || null; const states = projectStore?.projectStates || null;
const priorities = ISSUE_PRIORITIES || null; const priorities = ISSUE_PRIORITIES || null;
const labels = projectStore?.projectLabels || null; const labels = projectStore?.projectLabels || null;
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.projectStates || null; const projects = workspaceSlug ? projectStore?.projects[workspaceSlug.toString()] || null : null;
const estimates = null; const estimates =
projectDetails?.estimate !== null
? projectStore.projectEstimates?.find((e) => e.id === projectDetails?.estimate) || null
: null;
return ( return (
<div className={`relative w-full h-full bg-custom-background-90`}> <div className={`relative w-full h-full bg-custom-background-90`}>
@ -79,9 +86,9 @@ export const ModuleListLayout: React.FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
</div> </div>
); );

View File

@ -4,7 +4,7 @@ import { observer } from "mobx-react-lite";
// mobx store // mobx store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { List } from "./default"; import { List } from "../default";
import { ProjectIssueQuickActions } from "components/issues"; import { ProjectIssueQuickActions } from "components/issues";
// types // types
import { IIssue } from "types"; import { IIssue } from "types";
@ -50,7 +50,6 @@ export const ProfileIssuesListLayout: FC = observer(() => {
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.workspaceProjects || null; const projects = projectStore?.workspaceProjects || null;
const estimates = null;
return ( return (
<div className={`relative w-full h-full bg-custom-background-90`}> <div className={`relative w-full h-full bg-custom-background-90`}>
@ -70,9 +69,9 @@ export const ProfileIssuesListLayout: FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={null}
/> />
</div> </div>
); );

View File

@ -4,8 +4,10 @@ import { observer } from "mobx-react-lite";
// hooks // hooks
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
// components // components
import { List } from "./default"; import { List } from "../default";
import { ProjectIssueQuickActions } from "components/issues"; import { ProjectIssueQuickActions } from "components/issues";
// helpers
import { orderArrayBy } from "helpers/array.helper";
// types // types
import { IIssue } from "types"; import { IIssue } from "types";
// constants // constants
@ -13,7 +15,7 @@ import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES } from "constants/issue";
export const ListLayout: FC = observer(() => { export const ListLayout: FC = observer(() => {
const router = useRouter(); const router = useRouter();
const { workspaceSlug } = router.query; const { workspaceSlug, projectId } = router.query;
const { const {
project: projectStore, project: projectStore,
@ -41,13 +43,18 @@ export const ListLayout: FC = observer(() => {
[issueStore, issueDetailStore, workspaceSlug] [issueStore, issueDetailStore, workspaceSlug]
); );
const projectDetails = projectId ? projectStore.project_details[projectId.toString()] : null;
const states = projectStore?.projectStates || null; const states = projectStore?.projectStates || null;
const priorities = ISSUE_PRIORITIES || null; const priorities = ISSUE_PRIORITIES || null;
const labels = projectStore?.projectLabels || null; const labels = projectStore?.projectLabels || null;
const members = projectStore?.projectMembers || null; const members = projectStore?.projectMembers || null;
const stateGroups = ISSUE_STATE_GROUPS || null; const stateGroups = ISSUE_STATE_GROUPS || null;
const projects = projectStore?.projectStates || null; const projects = workspaceSlug ? projectStore?.projects[workspaceSlug.toString()] || null : null;
const estimates = null; const estimates =
projectDetails?.estimate !== null
? projectStore.projectEstimates?.find((e) => e.id === projectDetails?.estimate) || null
: null;
return ( return (
<div className="relative w-full h-full bg-custom-background-90"> <div className="relative w-full h-full bg-custom-background-90">
@ -67,9 +74,9 @@ export const ListLayout: FC = observer(() => {
stateGroups={stateGroups} stateGroups={stateGroups}
priorities={priorities} priorities={priorities}
labels={labels} labels={labels}
members={members} members={members?.map((m) => m.member) ?? null}
projects={projects} projects={projects}
estimates={estimates} estimates={estimates?.points ? orderArrayBy(estimates.points, "key") : null}
/> />
</div> </div>
); );

View File

@ -1,7 +1,7 @@
import React from "react"; import React from "react";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { List } from "./default"; import { List } from "../default";
// store // store
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";
@ -10,7 +10,7 @@ import { ISSUE_STATE_GROUPS, ISSUE_PRIORITIES } from "constants/issue";
export interface IViewListLayout {} export interface IViewListLayout {}
export const ViewListLayout: React.FC = observer(() => { export const ProjectViewListLayout: React.FC = observer(() => {
const { project: projectStore, issue: issueStore, issueFilter: issueFilterStore }: RootStore = useMobxStore(); const { project: projectStore, issue: issueStore, issueFilter: issueFilterStore }: RootStore = useMobxStore();
const issues = issueStore?.getIssues; const issues = issueStore?.getIssues;

View File

@ -1,252 +1,28 @@
import { FC, useRef, useState } from "react";
import { Combobox } from "@headlessui/react";
import { ChevronDown, Search, X, Check } from "lucide-react";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { Tooltip } from "@plane/ui"; import { MembersSelect } from "components/project";
// hooks // types
import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown"; import { IUserLite } from "types";
interface IFiltersOption {
id: string;
title: string;
avatar: string;
}
export interface IIssuePropertyAssignee { export interface IIssuePropertyAssignee {
value?: any; value: string[];
onChange?: (id: any, data: any) => void; onChange: (data: string[]) => void;
members: IUserLite[] | null;
disabled?: boolean; disabled?: boolean;
list?: any; hideDropdownArrow?: boolean;
className?: string;
buttonClassName?: string;
optionsClassName?: string;
dropdownArrow?: boolean;
} }
export const IssuePropertyAssignee: FC<IIssuePropertyAssignee> = observer((props) => { export const IssuePropertyAssignee: React.FC<IIssuePropertyAssignee> = observer((props) => {
const { value, onChange, disabled, list, className, buttonClassName, optionsClassName, dropdownArrow = true } = props; const { value, onChange, members, disabled = false, hideDropdownArrow = false } = props;
const dropdownBtn = useRef<any>(null);
const dropdownOptions = useRef<any>(null);
const [isOpen, setIsOpen] = useState<boolean>(false);
const [search, setSearch] = useState<string>("");
const options: IFiltersOption[] | [] =
(list &&
list?.length > 0 &&
list.map((_member: any) => ({
id: _member?.member?.id,
title: _member?.member?.display_name,
avatar: _member?.member?.avatar && _member?.member?.avatar !== "" ? _member?.member?.avatar : null,
}))) ||
[];
useDynamicDropdownPosition(isOpen, () => setIsOpen(false), dropdownBtn, dropdownOptions);
const selectedOption: IFiltersOption[] =
(value && value?.length > 0 && options.filter((_member: IFiltersOption) => value.includes(_member.id))) || [];
const filteredOptions: IFiltersOption[] =
search === ""
? options && options.length > 0
? options
: []
: options && options.length > 0
? options.filter((_member: IFiltersOption) =>
_member.title.toLowerCase().replace(/\s+/g, "").includes(search.toLowerCase().replace(/\s+/g, ""))
)
: [];
const assigneeRenderLength = 5;
return ( return (
<Combobox <MembersSelect
multiple={true} value={value}
as="div" onChange={onChange}
className={`${className}`} members={members ?? undefined}
value={selectedOption.map((_member: IFiltersOption) => _member.id) as string[]}
onChange={(data: string[]) => {
if (onChange && selectedOption) onChange(data, selectedOption);
}}
disabled={disabled} disabled={disabled}
> hideDropdownArrow={hideDropdownArrow}
{({ open }: { open: boolean }) => { multiple
if (open) {
if (!isOpen) setIsOpen(true);
} else if (isOpen) setIsOpen(false);
return (
<>
<Combobox.Button
ref={dropdownBtn}
type="button"
className={`flex items-center justify-between gap-1 px-1 py-0.5 rounded-sm shadow-sm border border-custom-border-300 duration-300 outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`}
>
{selectedOption && selectedOption?.length > 0 ? (
<>
{selectedOption?.length > 1 ? (
<Tooltip
tooltipHeading={`Assignees`}
tooltipContent={(selectedOption.map((_label: IFiltersOption) => _label.title) || []).join(", ")}
>
<div className="flex-shrink-0 flex justify-center items-center gap-1 pr-[8px]">
{selectedOption.slice(0, assigneeRenderLength).map((_assignee) => (
<div
key={_assignee?.id}
className="flex-shrink-0 w-[16px] h-[16px] rounded-sm bg-gray-700 flex justify-center items-center text-white capitalize relative -mr-[8px] text-xs overflow-hidden border border-custom-border-300"
>
{_assignee && _assignee.avatar ? (
<img
src={_assignee.avatar}
className="absolute top-0 left-0 h-full w-full object-cover"
alt={_assignee.title}
/> />
) : (
_assignee.title[0]
)}
</div>
))}
{selectedOption.length > assigneeRenderLength && (
<div className="flex-shrink-0 h-[16px] px-0.5 rounded-sm bg-gray-700 flex justify-center items-center text-white capitalize relative -mr-[8px] text-xs overflow-hidden border border-custom-border-300">
+{selectedOption?.length - assigneeRenderLength}
</div>
)}
</div>
</Tooltip>
) : (
<Tooltip
tooltipHeading={`Assignees`}
tooltipContent={(selectedOption.map((_label: IFiltersOption) => _label.title) || []).join(", ")}
>
<div className="flex-shrink-0 flex justify-center items-center gap-1 text-xs">
<div className="flex-shrink-0 w-4 h-4 rounded-sm flex justify-center items-center text-white capitalize relative overflow-hidden text-xs">
{selectedOption[0] && selectedOption[0].avatar ? (
<img
src={selectedOption[0].avatar}
className="absolute top-0 left-0 h-full w-full object-cover"
alt={selectedOption[0].title}
/>
) : (
<div className="w-full h-full bg-gray-700 flex justify-center items-center">
{selectedOption[0].title[0]}
</div>
)}
</div>
<div className="line-clamp-1">{selectedOption[0].title}</div>
</div>
</Tooltip>
)}
</>
) : (
<Tooltip tooltipHeading={`Assignees`} tooltipContent={``}>
<div className="text-xs">Select Assignees</div>
</Tooltip>
)}
{dropdownArrow && !disabled && (
<div className="flex-shrink-0 w-[14px] h-[14px] flex justify-center items-center">
<ChevronDown width={14} strokeWidth={2} />
</div>
)}
</Combobox.Button>
<div className={`${open ? "fixed z-20 top-0 left-0 h-full w-full cursor-auto" : ""}`}>
<Combobox.Options
ref={dropdownOptions}
className={`absolute z-10 border border-custom-border-300 p-2 rounded bg-custom-background-100 text-xs shadow-lg focus:outline-none whitespace-nowrap mt-1 space-y-1 ${optionsClassName}`}
>
{options && options.length > 0 ? (
<>
<div className="flex w-full items-center justify-start rounded border border-custom-border-200 bg-custom-background-90 px-1">
<div className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm">
<Search width={12} strokeWidth={2} />
</div>
<div>
<Combobox.Input
className="w-full bg-transparent p-1 text-xs text-custom-text-200 placeholder:text-custom-text-400 focus:outline-none"
value={search}
onChange={(e) => setSearch(e.target.value)}
placeholder="Search"
displayValue={(assigned: any) => assigned?.name}
/>
</div>
{search && search.length > 0 && (
<div
onClick={() => setSearch("")}
className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm cursor-pointer hover:bg-custom-background-80"
>
<X width={12} strokeWidth={2} />
</div>
)}
</div>
<div className={`space-y-0.5 max-h-48 overflow-y-scroll`}>
{filteredOptions ? (
filteredOptions.length > 0 ? (
filteredOptions.map((option) => (
<Combobox.Option
key={option.id}
value={option.id}
className={({ active }) =>
`cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active || (value && value.length > 0 && value.includes(option?.id))
? "bg-custom-background-80"
: ""
} ${
value && value.length > 0 && value.includes(option?.id)
? "text-custom-text-100"
: "text-custom-text-200"
}`
}
>
<div className="flex items-center gap-1 w-full px-1">
<div className="flex-shrink-0 w-[18px] h-[18px] rounded-sm flex justify-center items-center text-white capitalize relative overflow-hidden">
{option && option.avatar ? (
<img
src={option.avatar}
className="absolute top-0 left-0 h-full w-full object-cover"
alt={option.title}
/>
) : (
<div className="w-full h-full bg-gray-700 flex justify-center items-center">
{option.title[0]}
</div>
)}
</div>
<div className="line-clamp-1">{option.title}</div>
{value && value.length > 0 && value.includes(option?.id) && (
<div className="flex-shrink-0 ml-auto w-[13px] h-[13px] flex justify-center items-center">
<Check width={13} strokeWidth={2} />
</div>
)}
</div>
</Combobox.Option>
))
) : (
<span className="flex items-center gap-2 p-1">
<p className="text-left text-custom-text-200 ">No matching results</p>
</span>
)
) : (
<p className="text-center text-custom-text-200">Loading...</p>
)}
</div>
</>
) : (
<p className="text-center text-custom-text-200">No options available.</p>
)}
</Combobox.Options>
</div>
</>
);
}}
</Combobox>
); );
}); });

View File

@ -15,14 +15,15 @@ import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown";
import { renderDateFormat } from "helpers/date-time.helper"; import { renderDateFormat } from "helpers/date-time.helper";
export interface IIssuePropertyDate { export interface IIssuePropertyDate {
value?: any; value: any;
onChange?: (date: any) => void; onChange: (date: any) => void;
disabled?: boolean; disabled?: boolean;
placeHolder?: string; placeHolder?: string;
} }
export const IssuePropertyDate: React.FC<IIssuePropertyDate> = observer( export const IssuePropertyDate: React.FC<IIssuePropertyDate> = observer((props) => {
({ value, onChange, disabled, placeHolder }) => { const { value, onChange, disabled, placeHolder } = props;
const dropdownBtn = React.useRef<any>(null); const dropdownBtn = React.useRef<any>(null);
const dropdownOptions = React.useRef<any>(null); const dropdownOptions = React.useRef<any>(null);
@ -41,29 +42,25 @@ export const IssuePropertyDate: React.FC<IIssuePropertyDate> = observer(
<> <>
<Popover.Button <Popover.Button
ref={dropdownBtn} ref={dropdownBtn}
className={`flex items-center justify-between gap-1 px-1 py-0.5 rounded-sm shadow-sm border border-custom-border-300 duration-300 outline-none ${ className={`px-2.5 py-1 h-5 flex items-center rounded border-[0.5px] border-custom-border-300 duration-300 outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80" disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
}`} }`}
> >
<Tooltip tooltipHeading={placeHolder ? placeHolder : `Select date`} tooltipContent={value}> <Tooltip tooltipHeading={placeHolder} tooltipContent={value ?? "None"}>
<div className="flex-shrink-0 overflow-hidden rounded-sm flex justify-center items-center"> <div className="overflow-hidden flex justify-center items-center gap-2">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center"> <Calendar className="h-3 w-3" strokeWidth={2} />
<Calendar width={10} strokeWidth={2} /> {value && (
</div>
{value ? (
<> <>
<div className="px-1 text-xs">{value}</div> <div className="text-xs">{value}</div>
<div <div
className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center cursor-pointer" className="flex-shrink-0 flex justify-center items-center"
onClick={() => { onClick={() => {
if (onChange) onChange(null); if (onChange) onChange(null);
}} }}
> >
<X width={10} strokeWidth={2} /> <X className="h-2.5 w-2.5" strokeWidth={2} />
</div> </div>
</> </>
) : (
<div className="text-xs">{placeHolder ? placeHolder : `Select date`}</div>
)} )}
</div> </div>
</Tooltip> </Tooltip>
@ -95,5 +92,4 @@ export const IssuePropertyDate: React.FC<IIssuePropertyDate> = observer(
}} }}
</Popover> </Popover>
); );
} });
);

View File

@ -1,217 +1,28 @@
import React from "react";
// headless ui
import { Combobox } from "@headlessui/react";
// lucide icons
import { ChevronDown, Search, X, Check, Triangle } from "lucide-react";
// mobx
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { Tooltip } from "@plane/ui"; import { EstimateSelect } from "components/estimates";
// hooks // types
import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown"; import { IEstimatePoint } from "types";
// mobx
import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root";
interface IFiltersOption {
id: string;
title: string;
key: string;
}
export interface IIssuePropertyEstimates { export interface IIssuePropertyEstimates {
value?: any; value: number | null;
onChange?: (id: any) => void; onChange: (value: number | null) => void;
estimatePoints: IEstimatePoint[] | null;
disabled?: boolean; disabled?: boolean;
hideDropdownArrow?: boolean;
workspaceSlug?: string;
projectId?: string;
className?: string;
buttonClassName?: string;
optionsClassName?: string;
dropdownArrow?: boolean;
} }
export const IssuePropertyEstimates: React.FC<IIssuePropertyEstimates> = observer( export const IssuePropertyEstimates: React.FC<IIssuePropertyEstimates> = observer((props) => {
({ const { value, onChange, estimatePoints, disabled, hideDropdownArrow = false } = props;
value,
onChange,
disabled,
workspaceSlug,
projectId,
className,
buttonClassName,
optionsClassName,
dropdownArrow = true,
}) => {
const { project: projectStore }: RootStore = useMobxStore();
const dropdownBtn = React.useRef<any>(null);
const dropdownOptions = React.useRef<any>(null);
const [isOpen, setIsOpen] = React.useState<boolean>(false);
const [search, setSearch] = React.useState<string>("");
const projectDetail =
(workspaceSlug && projectId && projectStore?.getProjectById(workspaceSlug, projectId)) || null;
const projectEstimateId = (projectDetail && projectDetail?.estimate) || null;
const estimates = (projectEstimateId && projectStore?.getProjectEstimateById(projectEstimateId)) || null;
const options: IFiltersOption[] | [] =
(estimates &&
estimates.points &&
estimates.points.length > 0 &&
estimates.points.map((_estimate) => ({
id: _estimate?.id,
title: _estimate?.value,
key: _estimate?.key.toString(),
}))) ||
[];
useDynamicDropdownPosition(isOpen, () => setIsOpen(false), dropdownBtn, dropdownOptions);
const selectedOption: IFiltersOption | null | undefined =
(value && options.find((_estimate: IFiltersOption) => _estimate.key === value)) || null;
const filteredOptions: IFiltersOption[] =
search === ""
? options && options.length > 0
? options
: []
: options && options.length > 0
? options.filter((_estimate: IFiltersOption) =>
_estimate.title.toLowerCase().replace(/\s+/g, "").includes(search.toLowerCase().replace(/\s+/g, ""))
)
: [];
return ( return (
<Combobox <EstimateSelect
as="div" value={value}
className={`${className}`} onChange={onChange}
value={selectedOption && selectedOption.key} estimatePoints={estimatePoints ?? undefined}
onChange={(data: string) => { buttonClassName="h-5"
if (onChange) onChange(data);
}}
disabled={disabled} disabled={disabled}
> hideDropdownArrow={hideDropdownArrow}
{({ open }: { open: boolean }) => {
if (open) {
if (!isOpen) setIsOpen(true);
} else if (isOpen) setIsOpen(false);
return (
<>
<Combobox.Button
ref={dropdownBtn}
type="button"
className={`flex items-center justify-between gap-1 px-1 py-0.5 rounded-sm shadow-sm border border-custom-border-300 duration-300 outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`}
>
{selectedOption ? (
<Tooltip tooltipHeading={`Estimates`} tooltipContent={selectedOption?.title}>
<div className="flex-shrink-0 flex justify-center items-center gap-1">
<div className="flex-shrink-0 w-[12px] h-[12px] flex justify-center items-center">
<Triangle width={14} strokeWidth={2} />
</div>
<div className="pl-0.5 pr-1 text-xs">{selectedOption?.title}</div>
</div>
</Tooltip>
) : (
<Tooltip tooltipHeading={`Estimates`} tooltipContent={``}>
<div className="text-xs">Select Estimates</div>
</Tooltip>
)}
{dropdownArrow && !disabled && (
<div className="flex-shrink-0 w-[14px] h-[14px] flex justify-center items-center">
<ChevronDown width={14} strokeWidth={2} />
</div>
)}
</Combobox.Button>
<div className={`${open ? "fixed z-20 top-0 left-0 h-full w-full cursor-auto" : ""}`}>
<Combobox.Options
ref={dropdownOptions}
className={`absolute z-10 border border-custom-border-300 p-2 rounded bg-custom-background-100 text-xs shadow-lg focus:outline-none whitespace-nowrap mt-1 space-y-1 ${optionsClassName}`}
>
{options && options.length > 0 ? (
<>
<div className="flex w-full items-center justify-start rounded border border-custom-border-200 bg-custom-background-90 px-1">
<div className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm">
<Search width={12} strokeWidth={2} />
</div>
<div>
<Combobox.Input
className="w-full bg-transparent p-1 text-xs text-custom-text-200 placeholder:text-custom-text-400 focus:outline-none"
value={search}
onChange={(e) => setSearch(e.target.value)}
placeholder="Search"
displayValue={(assigned: any) => assigned?.name}
/> />
</div>
{search && search.length > 0 && (
<div
onClick={() => setSearch("")}
className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm cursor-pointer hover:bg-custom-background-80"
>
<X width={12} strokeWidth={2} />
</div>
)}
</div>
<div className={`space-y-0.5 max-h-48 overflow-y-scroll`}>
{filteredOptions ? (
filteredOptions.length > 0 ? (
filteredOptions.map((option) => (
<Combobox.Option
key={option.key}
value={option.key}
className={({ active, selected }) =>
`cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active || selected ? "bg-custom-background-80" : ""
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
}
>
{({ selected }) => (
<div className="flex items-center gap-1 w-full px-1">
<div className="flex-shrink-0 w-[13px] h-[13px] flex justify-center items-center">
<Triangle width={14} strokeWidth={2} />
</div>
<div className="line-clamp-1">{option.title}</div>
{selected && (
<div className="flex-shrink-0 ml-auto w-[13px] h-[13px] flex justify-center items-center">
<Check width={13} strokeWidth={2} />
</div>
)}
</div>
)}
</Combobox.Option>
))
) : (
<span className="flex items-center gap-2 p-1">
<p className="text-left text-custom-text-200 ">No matching results</p>
</span>
)
) : (
<p className="text-center text-custom-text-200">Loading...</p>
)}
</div>
</>
) : (
<p className="text-center text-custom-text-200">No options available.</p>
)}
</Combobox.Options>
</div>
</>
);
}}
</Combobox>
);
}
); );
});

View File

@ -1,230 +1,28 @@
import { FC, useRef, useState } from "react";
import { Combobox } from "@headlessui/react";
import { ChevronDown, Search, X, Check } from "lucide-react";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { Tooltip } from "@plane/ui"; import { LabelSelect } from "components/project";
// hooks // types
import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown"; import { IIssueLabels } from "types";
interface IFiltersOption {
id: string;
title: string;
color: string | null;
}
export interface IIssuePropertyLabels { export interface IIssuePropertyLabels {
value?: any; value: string[];
onChange?: (id: any, data: any) => void; onChange: (data: string[]) => void;
labels: IIssueLabels[] | null;
disabled?: boolean; disabled?: boolean;
list?: any; hideDropdownArrow?: boolean;
className?: string;
buttonClassName?: string;
optionsClassName?: string;
dropdownArrow?: boolean;
} }
export const IssuePropertyLabels: FC<IIssuePropertyLabels> = observer((props) => { export const IssuePropertyLabels: React.FC<IIssuePropertyLabels> = observer((props) => {
const { const { value, onChange, labels, disabled, hideDropdownArrow = false } = props;
value,
onChange,
disabled,
list,
className,
buttonClassName,
optionsClassName,
dropdownArrow = true,
} = props;
const dropdownBtn = useRef<any>(null);
const dropdownOptions = useRef<any>(null);
const [isOpen, setIsOpen] = useState<boolean>(false);
const [search, setSearch] = useState<string>("");
const options: IFiltersOption[] | [] =
(list &&
list?.length > 0 &&
list.map((_label: any) => ({
id: _label?.id,
title: _label?.name,
color: _label?.color || null,
}))) ||
[];
useDynamicDropdownPosition(isOpen, () => setIsOpen(false), dropdownBtn, dropdownOptions);
const selectedOption: IFiltersOption[] =
(value && value?.length > 0 && options.filter((_label: IFiltersOption) => value.includes(_label.id))) || [];
const filteredOptions: IFiltersOption[] =
search === ""
? options && options.length > 0
? options
: []
: options && options.length > 0
? options.filter((_label: IFiltersOption) =>
_label.title.toLowerCase().replace(/\s+/g, "").includes(search.toLowerCase().replace(/\s+/g, ""))
)
: [];
return ( return (
<Combobox <LabelSelect
multiple={true} value={value}
as="div" onChange={onChange}
className={`${className}`} labels={labels ?? undefined}
value={selectedOption.map((_label: IFiltersOption) => _label.id) as string[]} buttonClassName="h-5"
onChange={(data: string[]) => {
if (onChange && selectedOption) onChange(data, selectedOption);
}}
disabled={disabled} disabled={disabled}
> hideDropdownArrow={hideDropdownArrow}
{({ open }: { open: boolean }) => {
if (open) {
if (!isOpen) setIsOpen(true);
} else if (isOpen) setIsOpen(false);
return (
<>
<Combobox.Button
ref={dropdownBtn}
type="button"
className={`flex items-center justify-between gap-1 px-1 py-0.5 rounded-sm shadow-sm border border-custom-border-300 duration-300 outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`}
>
{selectedOption && selectedOption?.length > 0 ? (
<>
{selectedOption?.length === 1 ? (
<Tooltip
tooltipHeading={`Labels`}
tooltipContent={(selectedOption.map((_label: IFiltersOption) => _label.title) || []).join(", ")}
>
<div className="flex-shrink-0 flex justify-center items-center gap-1">
<div
className="flex-shrink-0 w-[10px] h-[10px] rounded-full"
style={{
backgroundColor: selectedOption[0]?.color || "#444",
}}
/> />
<div className="pl-0.5 pr-1 text-xs">{selectedOption[0]?.title}</div>
</div>
</Tooltip>
) : (
<Tooltip
tooltipHeading={`Labels`}
tooltipContent={(selectedOption.map((_label: IFiltersOption) => _label.title) || []).join(", ")}
>
<div className="flex-shrink-0 flex justify-center items-center gap-1">
<div
className="flex-shrink-0 w-[10px] h-[10px] rounded-full"
style={{ backgroundColor: "#444" }}
/>
<div className="pl-0.5 pr-1 text-xs">{selectedOption?.length} Labels</div>
</div>
</Tooltip>
)}
</>
) : (
<Tooltip tooltipHeading={`Labels`} tooltipContent={``}>
<div className="text-xs">Select Labels</div>
</Tooltip>
)}
{dropdownArrow && !disabled && (
<div className="flex-shrink-0 w-[14px] h-[14px] flex justify-center items-center">
<ChevronDown width={14} strokeWidth={2} />
</div>
)}
</Combobox.Button>
<div className={`${open ? "fixed z-20 top-0 left-0 h-full w-full cursor-auto" : ""}`}>
<Combobox.Options
ref={dropdownOptions}
className={`absolute z-10 border border-custom-border-300 p-2 rounded bg-custom-background-100 text-xs shadow-lg focus:outline-none whitespace-nowrap mt-1 space-y-1 ${optionsClassName}`}
>
{options && options.length > 0 ? (
<>
<div className="flex w-full items-center justify-start rounded border border-custom-border-200 bg-custom-background-90 px-1">
<div className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm">
<Search width={12} strokeWidth={2} />
</div>
<div>
<Combobox.Input
className="w-full bg-transparent p-1 text-xs text-custom-text-200 placeholder:text-custom-text-400 focus:outline-none"
value={search}
onChange={(e) => setSearch(e.target.value)}
placeholder="Search"
displayValue={(assigned: any) => assigned?.name}
/>
</div>
{search && search.length > 0 && (
<div
onClick={() => setSearch("")}
className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm cursor-pointer hover:bg-custom-background-80"
>
<X width={12} strokeWidth={2} />
</div>
)}
</div>
<div className={`space-y-0.5 max-h-48 overflow-y-scroll`}>
{filteredOptions ? (
filteredOptions.length > 0 ? (
filteredOptions.map((option) => (
<Combobox.Option
key={option.id}
value={option.id}
className={({ active }) =>
`cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active || (value && value.length > 0 && value.includes(option?.id))
? "bg-custom-background-80"
: ""
} ${
value && value.length > 0 && value.includes(option?.id)
? "text-custom-text-100"
: "text-custom-text-200"
}`
}
>
<div className="flex items-center gap-1 w-full px-1">
<div
className="flex-shrink-0 w-[10px] h-[10px] rounded-full"
style={{
backgroundColor: option.color || "#444",
}}
/>
<div className="line-clamp-1">{option.title}</div>
{value && value.length > 0 && value.includes(option?.id) && (
<div className="flex-shrink-0 ml-auto w-[13px] h-[13px] flex justify-center items-center">
<Check width={13} strokeWidth={2} />
</div>
)}
</div>
</Combobox.Option>
))
) : (
<span className="flex items-center gap-2 p-1">
<p className="text-left text-custom-text-200 ">No matching results</p>
</span>
)
) : (
<p className="text-center text-custom-text-200">Loading...</p>
)}
</div>
</>
) : (
<p className="text-center text-custom-text-200">No options available.</p>
)}
</Combobox.Options>
</div>
</>
);
}}
</Combobox>
); );
}); });

View File

@ -1,223 +1,25 @@
import { FC, useRef, useState } from "react"; import { PrioritySelect } from "components/project";
import { Combobox } from "@headlessui/react";
import { ChevronDown, Search, X, Check, AlertCircle, SignalHigh, SignalMedium, SignalLow, Ban } from "lucide-react";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // types
import { Tooltip } from "@plane/ui"; import { TIssuePriorities } from "types";
// hooks
import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown";
interface IFiltersOption {
id: string;
title: string;
}
export interface IIssuePropertyPriority { export interface IIssuePropertyPriority {
value?: any; value: TIssuePriorities;
onChange?: (id: any, data: IFiltersOption) => void; onChange: (value: TIssuePriorities) => void;
disabled?: boolean; disabled?: boolean;
list?: any; hideDropdownArrow?: boolean;
className?: string;
buttonClassName?: string;
optionsClassName?: string;
dropdownArrow?: boolean;
} }
const Icon = ({ priority }: any) => ( export const IssuePropertyPriority: React.FC<IIssuePropertyPriority> = observer((props) => {
<div className="w-full h-full"> const { value, onChange, disabled, hideDropdownArrow = false } = props;
{priority === "urgent" ? (
<div className="border border-red-500 bg-red-500 text-white w-full h-full overflow-hidden flex justify-center items-center rounded-sm">
<AlertCircle size={12} strokeWidth={2} />
</div>
) : priority === "high" ? (
<div className="border border-red-500/20 bg-red-500/10 text-red-500 w-full h-full overflow-hidden flex justify-center items-center rounded-sm">
<SignalHigh size={12} strokeWidth={2} className="pl-[3px]" />
</div>
) : priority === "medium" ? (
<div className="border border-orange-500/20 bg-orange-500/10 text-orange-500 w-full h-full overflow-hidden flex justify-center items-center rounded-sm">
<SignalMedium size={12} strokeWidth={2} className="pl-[3px]" />
</div>
) : priority === "low" ? (
<div className="border border-green-500/20 bg-green-500/10 text-green-500 w-full h-full overflow-hidden flex justify-center items-center rounded-sm">
<SignalLow size={12} strokeWidth={2} className="pl-[3px]" />
</div>
) : (
<div className="border border-custom-border-400/20 bg-custom-text-400/10 text-custom-text-400 w-full h-full overflow-hidden flex justify-center items-center rounded-sm">
<Ban size={12} strokeWidth={2} />
</div>
)}
</div>
);
export const IssuePropertyPriority: FC<IIssuePropertyPriority> = observer((props) => {
const {
value,
onChange,
disabled,
list,
className,
buttonClassName,
optionsClassName,
dropdownArrow = true,
} = props;
const dropdownBtn = useRef<any>(null);
const dropdownOptions = useRef<any>(null);
const [isOpen, setIsOpen] = useState<boolean>(false);
const [search, setSearch] = useState<string>("");
const options: IFiltersOption[] | [] =
(list &&
list?.length > 0 &&
list.map((_priority: any) => ({
id: _priority?.key,
title: _priority?.title,
}))) ||
[];
useDynamicDropdownPosition(isOpen, () => setIsOpen(false), dropdownBtn, dropdownOptions);
const selectedOption: IFiltersOption | null | undefined =
(value && options.find((_priority: IFiltersOption) => _priority.id === value)) || null;
const filteredOptions: IFiltersOption[] =
search === ""
? options && options.length > 0
? options
: []
: options && options.length > 0
? options.filter((_priority: IFiltersOption) =>
_priority.title.toLowerCase().replace(/\s+/g, "").includes(search.toLowerCase().replace(/\s+/g, ""))
)
: [];
return ( return (
<Combobox <PrioritySelect
as="div" value={value}
className={`${className}`} onChange={onChange}
value={selectedOption && selectedOption.id} buttonClassName="!h-5 p-1.5"
onChange={(data: string) => {
if (onChange && selectedOption) onChange(data, selectedOption);
}}
disabled={disabled} disabled={disabled}
> hideDropdownArrow={hideDropdownArrow}
{({ open }: { open: boolean }) => {
if (open) {
if (!isOpen) setIsOpen(true);
} else if (isOpen) setIsOpen(false);
return (
<>
<Combobox.Button
ref={dropdownBtn}
type="button"
className={`flex items-center justify-between gap-1 px-1 py-0.5 rounded-sm shadow-sm border border-custom-border-300 duration-300 outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`}
>
{selectedOption ? (
<Tooltip tooltipHeading={`Priority`} tooltipContent={selectedOption?.title}>
<div className="flex-shrink-0 flex justify-center items-center gap-1">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center">
<Icon priority={selectedOption?.id} />
</div>
<div className="pl-0.5 pr-1 text-xs">{selectedOption?.title}</div>
</div>
</Tooltip>
) : (
<Tooltip tooltipHeading={`Priority`} tooltipContent={``}>
<div className="text-xs">Select Priority</div>
</Tooltip>
)}
{dropdownArrow && !disabled && (
<div className="flex-shrink-0 w-[14px] h-[14px] flex justify-center items-center">
<ChevronDown width={14} strokeWidth={2} />
</div>
)}
</Combobox.Button>
<div className={`${open ? "fixed z-20 top-0 left-0 h-full w-full cursor-auto" : ""}`}>
<Combobox.Options
ref={dropdownOptions}
className={`absolute z-10 border border-custom-border-300 p-2 rounded bg-custom-background-100 text-xs shadow-lg focus:outline-none whitespace-nowrap mt-1 space-y-1 ${optionsClassName}`}
>
{options && options.length > 0 ? (
<>
<div className="flex w-full items-center justify-start rounded border border-custom-border-200 bg-custom-background-90 px-1">
<div className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm">
<Search width={12} strokeWidth={2} />
</div>
<div>
<Combobox.Input
className="w-full bg-transparent p-1 text-xs text-custom-text-200 placeholder:text-custom-text-400 focus:outline-none"
value={search}
onChange={(e) => setSearch(e.target.value)}
placeholder="Search"
displayValue={(assigned: any) => assigned?.name}
/> />
</div>
{search && search.length > 0 && (
<div
onClick={() => setSearch("")}
className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm cursor-pointer hover:bg-custom-background-80"
>
<X width={12} strokeWidth={2} />
</div>
)}
</div>
<div className={`space-y-0.5 max-h-48 overflow-y-scroll`}>
{filteredOptions ? (
filteredOptions.length > 0 ? (
filteredOptions.map((option) => (
<Combobox.Option
key={option.id}
value={option.id}
className={({ active, selected }) =>
`cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active || selected ? "bg-custom-background-80" : ""
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
}
>
{({ selected }) => (
<div className="flex items-center gap-1 w-full px-1">
<div className="flex-shrink-0 w-[16px] h-[16px] flex justify-center items-center">
<Icon priority={option?.id} />
</div>
<div className="line-clamp-1">{option.title}</div>
{selected && (
<div className="flex-shrink-0 ml-auto w-[13px] h-[13px] flex justify-center items-center">
<Check width={13} strokeWidth={2} />
</div>
)}
</div>
)}
</Combobox.Option>
))
) : (
<span className="flex items-center gap-2 p-1">
<p className="text-left text-custom-text-200 ">No matching results</p>
</span>
)
) : (
<p className="text-center text-custom-text-200">Loading...</p>
)}
</div>
</>
) : (
<p className="text-center text-custom-text-200">No options available.</p>
)}
</Combobox.Options>
</div>
</>
);
}}
</Combobox>
); );
}); });

View File

@ -1,214 +1,28 @@
import { FC, useRef, useState } from "react";
import { Combobox } from "@headlessui/react";
import { ChevronDown, Search, X, Check } from "lucide-react";
import { observer } from "mobx-react-lite"; import { observer } from "mobx-react-lite";
// components // components
import { Tooltip, StateGroupIcon } from "@plane/ui"; import { StateSelect } from "components/states";
// hooks
import useDynamicDropdownPosition from "hooks/use-dynamic-dropdown";
// types // types
import { IState } from "types"; import { IState } from "types";
interface IFiltersOption {
id: string;
title: string;
group: string;
color: string | null;
}
export interface IIssuePropertyState { export interface IIssuePropertyState {
value?: any; value: IState;
onChange?: (id: any, data: IFiltersOption) => void; onChange: (state: IState) => void;
states: IState[] | null;
disabled?: boolean; disabled?: boolean;
list?: any; hideDropdownArrow?: boolean;
className?: string;
buttonClassName?: string;
optionsClassName?: string;
dropdownArrow?: boolean;
} }
export const IssuePropertyState: FC<IIssuePropertyState> = observer((props) => { export const IssuePropertyState: React.FC<IIssuePropertyState> = observer((props) => {
const { const { value, onChange, states, disabled, hideDropdownArrow = false } = props;
value,
onChange,
disabled,
list,
className,
buttonClassName,
optionsClassName,
dropdownArrow = true,
} = props;
const dropdownBtn = useRef<any>(null);
const dropdownOptions = useRef<any>(null);
const [isOpen, setIsOpen] = useState<boolean>(false);
const [search, setSearch] = useState<string>("");
const options: IFiltersOption[] | [] =
(list &&
list?.length > 0 &&
list.map((_state: IState) => ({
id: _state?.id,
title: _state?.name,
group: _state?.group,
color: _state?.color || null,
}))) ||
[];
useDynamicDropdownPosition(isOpen, () => setIsOpen(false), dropdownBtn, dropdownOptions);
const selectedOption: IFiltersOption | null | undefined =
(value && options.find((_state: IFiltersOption) => _state.id === value)) || null;
const filteredOptions: IFiltersOption[] =
search === ""
? options && options.length > 0
? options
: []
: options && options.length > 0
? options.filter((_state: IFiltersOption) =>
_state.title.toLowerCase().replace(/\s+/g, "").includes(search.toLowerCase().replace(/\s+/g, ""))
)
: [];
return ( return (
<Combobox <StateSelect
as="div" value={value}
className={`${className}`} onChange={onChange}
value={selectedOption && selectedOption.id} states={states ?? undefined}
onChange={(data: string) => { buttonClassName="h-5"
if (onChange && selectedOption) onChange(data, selectedOption);
}}
disabled={disabled} disabled={disabled}
> hideDropdownArrow={hideDropdownArrow}
{({ open }: { open: boolean }) => {
if (open) {
if (!isOpen) setIsOpen(true);
} else if (isOpen) setIsOpen(false);
return (
<>
<Combobox.Button
ref={dropdownBtn}
type="button"
className={`flex items-center justify-between gap-1 px-1 py-0.5 rounded-sm shadow-sm border border-custom-border-300 duration-300 outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`}
>
{selectedOption ? (
<Tooltip tooltipHeading={`State`} tooltipContent={selectedOption?.title}>
<div className="flex-shrink-0 flex justify-center items-center gap-1">
<div className="flex-shrink-0 w-[12px] h-[12px] flex justify-center items-center">
<StateGroupIcon
stateGroup={selectedOption?.group as any}
color={(selectedOption?.color || null) as any}
width="12"
height="12"
/> />
</div>
<div className="pl-0.5 pr-1 text-xs">{selectedOption?.title}</div>
</div>
</Tooltip>
) : (
<Tooltip tooltipHeading={`State`} tooltipContent={``}>
<div className="text-xs">Select State</div>
</Tooltip>
)}
{dropdownArrow && !disabled && (
<div className="flex-shrink-0 w-[14px] h-[14px] flex justify-center items-center">
<ChevronDown width={14} strokeWidth={2} />
</div>
)}
</Combobox.Button>
<div className={`${open ? "fixed z-20 top-0 left-0 h-full w-full cursor-auto" : ""}`}>
<Combobox.Options
ref={dropdownOptions}
className={`absolute z-10 border border-custom-border-300 p-2 rounded bg-custom-background-100 text-xs shadow-lg focus:outline-none whitespace-nowrap mt-1 space-y-1 ${optionsClassName}`}
>
{options && options.length > 0 ? (
<>
<div className="flex w-full items-center justify-start rounded border border-custom-border-200 bg-custom-background-90 px-1">
<div className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm">
<Search width={12} strokeWidth={2} />
</div>
<div>
<Combobox.Input
className="w-full bg-transparent p-1 text-xs text-custom-text-200 placeholder:text-custom-text-400 focus:outline-none"
value={search}
onChange={(e) => setSearch(e.target.value)}
placeholder="Search"
displayValue={(assigned: any) => assigned?.name}
/>
</div>
{search && search.length > 0 && (
<div
onClick={() => setSearch("")}
className="flex-shrink-0 flex justify-center items-center w-[16px] h-[16px] rounded-sm cursor-pointer hover:bg-custom-background-80"
>
<X width={12} strokeWidth={2} />
</div>
)}
</div>
<div className={`space-y-0.5 max-h-48 overflow-y-scroll`}>
{filteredOptions ? (
filteredOptions.length > 0 ? (
filteredOptions.map((option) => (
<Combobox.Option
key={option.id}
value={option.id}
className={({ active, selected }) =>
`cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active || selected ? "bg-custom-background-80" : ""
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
}
>
{({ selected }) => (
<div className="flex items-center gap-1 w-full px-1">
<div className="flex-shrink-0 w-[13px] h-[13px] flex justify-center items-center">
<StateGroupIcon
stateGroup={option?.group as any}
color={(option?.color || null) as any}
width="13"
height="13"
/>
</div>
<div className="line-clamp-1">{option.title}</div>
{selected && (
<div className="flex-shrink-0 ml-auto w-[13px] h-[13px] flex justify-center items-center">
<Check width={13} strokeWidth={2} />
</div>
)}
</div>
)}
</Combobox.Option>
))
) : (
<span className="flex items-center gap-2 p-1">
<p className="text-left text-custom-text-200 ">No matching results</p>
</span>
)
) : (
<p className="text-center text-custom-text-200">Loading...</p>
)}
</div>
</>
) : (
<p className="text-center text-custom-text-200">No options available.</p>
)}
</Combobox.Options>
</div>
</>
);
}}
</Combobox>
); );
}); });

View File

@ -4,6 +4,8 @@ import React from "react";
import { StateSelect } from "components/states"; import { StateSelect } from "components/states";
// hooks // hooks
import useSubIssue from "hooks/use-sub-issue"; import useSubIssue from "hooks/use-sub-issue";
// helpers
import { getStatesList } from "helpers/state.helper";
// types // types
import { IIssue, IStateResponse } from "types"; import { IIssue, IStateResponse } from "types";
@ -22,12 +24,14 @@ export const SpreadsheetStateColumn: React.FC<Props> = (props) => {
const { subIssues, isLoading } = useSubIssue(issue.project_detail.id, issue.id, isExpanded); const { subIssues, isLoading } = useSubIssue(issue.project_detail.id, issue.id, isExpanded);
const statesList = getStatesList(states);
return ( return (
<> <>
<StateSelect <StateSelect
value={issue.state_detail} value={issue.state_detail}
onChange={(data) => onChange({ state: data.id, state_detail: data })} onChange={(data) => onChange({ state: data.id, state_detail: data })}
stateGroups={states} states={statesList}
buttonClassName="!shadow-none !border-0" buttonClassName="!shadow-none !border-0"
hideDropdownArrow hideDropdownArrow
disabled={disabled} disabled={disabled}

View File

@ -8,38 +8,37 @@ import { IssuePropertyPriority } from "components/issues/issue-layouts/propertie
import { IssuePropertyAssignee } from "components/issues/issue-layouts/properties/assignee"; import { IssuePropertyAssignee } from "components/issues/issue-layouts/properties/assignee";
import { IssuePropertyDate } from "components/issues/issue-layouts/properties/date"; import { IssuePropertyDate } from "components/issues/issue-layouts/properties/date";
// types // types
import { IIssue } from "types"; import { IIssue, IState, IUserLite, TIssuePriorities } from "types";
interface IPeekOverviewProperties { interface IPeekOverviewProperties {
issue: IIssue; issue: IIssue;
issueUpdate: (issue: Partial<IIssue>) => void; issueUpdate: (issue: Partial<IIssue>) => void;
states: IState[] | null;
states: any; members: IUserLite[] | null;
members: any;
priorities: any; priorities: any;
} }
export const PeekOverviewProperties: FC<IPeekOverviewProperties> = (props) => { export const PeekOverviewProperties: FC<IPeekOverviewProperties> = (props) => {
const { issue, issueUpdate, states, members, priorities } = props; const { issue, issueUpdate, states, members, priorities } = props;
const handleState = (_state: string) => { const handleState = (_state: IState) => {
if (issueUpdate) issueUpdate({ ...issue, state: _state }); issueUpdate({ ...issue, state: _state.id });
}; };
const handlePriority = (_priority: any) => { const handlePriority = (_priority: TIssuePriorities) => {
if (issueUpdate) issueUpdate({ ...issue, priority: _priority }); issueUpdate({ ...issue, priority: _priority });
}; };
const handleAssignee = (_assignees: string[]) => { const handleAssignee = (_assignees: string[]) => {
if (issueUpdate) issueUpdate({ ...issue, assignees: _assignees }); issueUpdate({ ...issue, assignees: _assignees });
}; };
const handleStartDate = (_startDate: string) => { const handleStartDate = (_startDate: string) => {
if (issueUpdate) issueUpdate({ ...issue, start_date: _startDate }); issueUpdate({ ...issue, start_date: _startDate });
}; };
const handleTargetDate = (_targetDate: string) => { const handleTargetDate = (_targetDate: string) => {
if (issueUpdate) issueUpdate({ ...issue, target_date: _targetDate }); issueUpdate({ ...issue, target_date: _targetDate });
}; };
return ( return (
@ -54,11 +53,11 @@ export const PeekOverviewProperties: FC<IPeekOverviewProperties> = (props) => {
</div> </div>
<div className="w-full"> <div className="w-full">
<IssuePropertyState <IssuePropertyState
value={issue?.state || null} value={issue?.state_detail || null}
dropdownArrow={false} onChange={handleState}
onChange={(id: string) => handleState(id)} states={states}
disabled={false} disabled={false}
list={states} hideDropdownArrow={true}
/> />
</div> </div>
</div> </div>
@ -74,10 +73,10 @@ export const PeekOverviewProperties: FC<IPeekOverviewProperties> = (props) => {
<div className="w-full"> <div className="w-full">
<IssuePropertyAssignee <IssuePropertyAssignee
value={issue?.assignees || null} value={issue?.assignees || null}
dropdownArrow={false}
onChange={(ids: string[]) => handleAssignee(ids)} onChange={(ids: string[]) => handleAssignee(ids)}
disabled={false} disabled={false}
list={members} hideDropdownArrow={true}
members={members}
/> />
</div> </div>
</div> </div>
@ -93,10 +92,9 @@ export const PeekOverviewProperties: FC<IPeekOverviewProperties> = (props) => {
<div className="w-full"> <div className="w-full">
<IssuePropertyPriority <IssuePropertyPriority
value={issue?.priority || null} value={issue?.priority || null}
dropdownArrow={false} onChange={handlePriority}
onChange={(id: string) => handlePriority(id)}
disabled={false} disabled={false}
list={priorities} hideDropdownArrow={true}
/> />
</div> </div>
</div> </div>

View File

@ -10,6 +10,8 @@ import { TrackEventService } from "services/track_event.service";
import { ViewDueDateSelect, ViewStartDateSelect } from "components/issues"; import { ViewDueDateSelect, ViewStartDateSelect } from "components/issues";
import { MembersSelect, PrioritySelect } from "components/project"; import { MembersSelect, PrioritySelect } from "components/project";
import { StateSelect } from "components/states"; import { StateSelect } from "components/states";
// helpers
import { getStatesList } from "helpers/state.helper";
// types // types
import { IUser, IIssue, IState } from "types"; import { IUser, IIssue, IState } from "types";
// fetch-keys // fetch-keys
@ -115,6 +117,8 @@ export const IssueProperty: React.FC<IIssueProperty> = observer((props) => {
); );
}; };
const statesList = getStatesList(projectStore.states?.[issue.project]);
return ( return (
<div className="relative flex items-center gap-1"> <div className="relative flex items-center gap-1">
{displayProperties.priority && ( {displayProperties.priority && (
@ -132,7 +136,7 @@ export const IssueProperty: React.FC<IIssueProperty> = observer((props) => {
<div className="flex-shrink-0"> <div className="flex-shrink-0">
<StateSelect <StateSelect
value={issue.state_detail} value={issue.state_detail}
stateGroups={projectStore.states ? projectStore.states[issue.project] : undefined} states={statesList}
onChange={(data) => handleStateChange(data)} onChange={(data) => handleStateChange(data)}
hideDropdownArrow hideDropdownArrow
disabled={!editable} disabled={!editable}

View File

@ -3,9 +3,6 @@ import { usePopper } from "react-popper";
import { Placement } from "@popperjs/core"; import { Placement } from "@popperjs/core";
import { Combobox } from "@headlessui/react"; import { Combobox } from "@headlessui/react";
import { Check, ChevronDown, PlusIcon, Search } from "lucide-react"; import { Check, ChevronDown, PlusIcon, Search } from "lucide-react";
// components
import { CreateLabelModal } from "components/labels";
// ui // ui
import { Tooltip } from "components/ui"; import { Tooltip } from "components/ui";
// types // types
@ -14,7 +11,7 @@ import { IIssueLabels } from "types";
type Props = { type Props = {
value: string[]; value: string[];
onChange: (data: string[]) => void; onChange: (data: string[]) => void;
labels: IIssueLabels[]; labels: IIssueLabels[] | undefined;
className?: string; className?: string;
buttonClassName?: string; buttonClassName?: string;
optionsClassName?: string; optionsClassName?: string;
@ -41,10 +38,16 @@ export const LabelSelect: React.FC<Props> = ({
const [referenceElement, setReferenceElement] = useState<HTMLButtonElement | null>(null); const [referenceElement, setReferenceElement] = useState<HTMLButtonElement | null>(null);
const [popperElement, setPopperElement] = useState<HTMLDivElement | null>(null); const [popperElement, setPopperElement] = useState<HTMLDivElement | null>(null);
const [labelModal, setLabelModal] = useState(false);
const { styles, attributes } = usePopper(referenceElement, popperElement, { const { styles, attributes } = usePopper(referenceElement, popperElement, {
placement: placement ?? "bottom-start", placement: placement ?? "bottom-start",
modifiers: [
{
name: "preventOverflow",
options: {
padding: 12,
},
},
],
}); });
const options = labels?.map((label) => ({ const options = labels?.map((label) => ({
@ -66,30 +69,7 @@ export const LabelSelect: React.FC<Props> = ({
const filteredOptions = const filteredOptions =
query === "" ? options : options?.filter((option) => option.query.toLowerCase().includes(query.toLowerCase())); query === "" ? options : options?.filter((option) => option.query.toLowerCase().includes(query.toLowerCase()));
const footerOption = (
<button
type="button"
className="flex w-full select-none items-center rounded py-2 px-1 hover:bg-custom-background-80"
onClick={() => setLabelModal(true)}
>
<span className="flex items-center justify-start gap-1 text-custom-text-200">
<PlusIcon className="h-4 w-4" aria-hidden="true" />
<span>Create New Label</span>
</span>
</button>
);
return ( return (
<>
{/* TODO: update this logic */}
{/* {projectId && (
<CreateLabelModal
isOpen={labelModal}
handleClose={() => setLabelModal(false)}
projectId={projectId}
user={user}
/>
)} */}
<Combobox <Combobox
as="div" as="div"
className={`flex-shrink-0 text-left ${className}`} className={`flex-shrink-0 text-left ${className}`}
@ -110,16 +90,16 @@ export const LabelSelect: React.FC<Props> = ({
: "cursor-pointer hover:bg-custom-background-80" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`} } ${buttonClassName}`}
> >
<div className={`flex items-center gap-2 text-custom-text-200`}> <div className="flex items-center gap-2 text-custom-text-200 h-full">
{value.length > 0 ? ( {value.length > 0 ? (
value.length <= maxRender ? ( value.length <= maxRender ? (
<> <>
{labels {labels
.filter((l) => value.includes(l.id)) ?.filter((l) => value.includes(l.id))
.map((label) => ( .map((label) => (
<div <div
key={label.id} key={label.id}
className="flex cursor-default items-center flex-shrink-0 rounded-md border border-custom-border-300 px-2.5 py-1 text-xs shadow-sm" className="flex cursor-default items-center flex-shrink-0 rounded border-[0.5px] border-custom-border-300 px-2.5 py-1 text-xs h-full"
> >
<div className="flex items-center gap-1.5 text-custom-text-200"> <div className="flex items-center gap-1.5 text-custom-text-200">
<span <span
@ -134,16 +114,16 @@ export const LabelSelect: React.FC<Props> = ({
))} ))}
</> </>
) : ( ) : (
<div className="flex cursor-default items-center flex-shrink-0 rounded-md border border-custom-border-300 px-2.5 py-1 text-xs shadow-sm"> <div className="h-full flex cursor-default items-center flex-shrink-0 rounded border-[0.5px] border-custom-border-300 px-2.5 py-1 text-xs">
<Tooltip <Tooltip
position="top" position="top"
tooltipHeading="Labels" tooltipHeading="Labels"
tooltipContent={labels tooltipContent={labels
.filter((l) => value.includes(l.id)) ?.filter((l) => value.includes(l.id))
.map((l) => l.name) .map((l) => l.name)
.join(", ")} .join(", ")}
> >
<div className="flex items-center gap-1.5 text-custom-text-200"> <div className="h-full flex items-center gap-1.5 text-custom-text-200">
<span className="h-2 w-2 flex-shrink-0 rounded-full bg-custom-primary" /> <span className="h-2 w-2 flex-shrink-0 rounded-full bg-custom-primary" />
{`${value.length} Labels`} {`${value.length} Labels`}
</div> </div>
@ -151,7 +131,9 @@ export const LabelSelect: React.FC<Props> = ({
</div> </div>
) )
) : ( ) : (
"" <div className="h-full flex items-center justify-center text-xs rounded border-[0.5px] border-custom-border-300 px-2.5 py-1 hover:bg-custom-background-80">
Select labels
</div>
)} )}
</div> </div>
{!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />} {!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />}
@ -184,7 +166,7 @@ export const LabelSelect: React.FC<Props> = ({
value={option.value} value={option.value}
className={({ active, selected }) => className={({ active, selected }) =>
`flex items-center justify-between gap-2 cursor-pointer select-none truncate rounded px-1 py-1.5 ${ `flex items-center justify-between gap-2 cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active && !selected ? "bg-custom-background-80" : "" active ? "bg-custom-background-80" : ""
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}` } ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
} }
> >
@ -205,10 +187,8 @@ export const LabelSelect: React.FC<Props> = ({
<p className="text-center text-custom-text-200">Loading...</p> <p className="text-center text-custom-text-200">Loading...</p>
)} )}
</div> </div>
{footerOption}
</div> </div>
</Combobox.Options> </Combobox.Options>
</Combobox> </Combobox>
</>
); );
}; };

View File

@ -1,18 +1,15 @@
import React, { useState } from "react"; import React, { useState } from "react";
import { usePopper } from "react-popper"; import { usePopper } from "react-popper";
import { Placement } from "@popperjs/core"; import { Placement } from "@popperjs/core";
// headless ui
import { Combobox } from "@headlessui/react"; import { Combobox } from "@headlessui/react";
import { Check, ChevronDown, Search, User2 } from "lucide-react";
// components // components
import { AssigneesList, Avatar, Tooltip } from "components/ui"; import { AssigneesList, Avatar, Tooltip } from "components/ui";
// icons
import { Check, ChevronDown, Search, User2 } from "lucide-react";
// types // types
import { IUserLite } from "types"; import { IUserLite } from "types";
type Props = { type Props = {
members: IUserLite[]; members: IUserLite[] | undefined;
className?: string; className?: string;
buttonClassName?: string; buttonClassName?: string;
optionsClassName?: string; optionsClassName?: string;
@ -51,6 +48,14 @@ export const MembersSelect: React.FC<Props> = ({
const { styles, attributes } = usePopper(referenceElement, popperElement, { const { styles, attributes } = usePopper(referenceElement, popperElement, {
placement: placement ?? "bottom-start", placement: placement ?? "bottom-start",
modifiers: [
{
name: "preventOverflow",
options: {
padding: 12,
},
},
],
}); });
const options = members?.map((member) => ({ const options = members?.map((member) => ({
@ -73,22 +78,22 @@ export const MembersSelect: React.FC<Props> = ({
tooltipContent={ tooltipContent={
value && value.length > 0 value && value.length > 0
? members ? members
.filter((m) => value.includes(m.display_name)) ?.filter((m) => value.includes(m.display_name))
.map((m) => m.display_name) .map((m) => m.display_name)
.join(", ") .join(", ")
: "No Assignee" : "No Assignee"
} }
position="top" position="top"
> >
<div className="flex items-center cursor-pointer w-full gap-2 text-custom-text-200"> <div className="flex items-center cursor-pointer h-full w-full gap-2 text-custom-text-200">
{value && value.length > 0 && Array.isArray(value) ? ( {value && value.length > 0 && Array.isArray(value) ? (
<AssigneesList userIds={value} length={3} showLength={true} /> <AssigneesList userIds={value} length={3} showLength={true} />
) : ( ) : (
<span <span
className="flex items-center justify-between gap-1 w-full text-xs px-2.5 py-1 rounded-md shadow-sm border border-custom-border-300 duration-300 focus:outline-none className="flex items-center justify-between gap-1 h-full w-full text-xs px-2.5 py-1 rounded border-[0.5px] border-custom-border-300 duration-300 focus:outline-none
" "
> >
<User2 className="h-3.5 w-3.5" /> <User2 className="h-3 w-3" />
</span> </span>
)} )}
</div> </div>
@ -112,9 +117,9 @@ export const MembersSelect: React.FC<Props> = ({
{!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />} {!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />}
</button> </button>
</Combobox.Button> </Combobox.Button>
<Combobox.Options> <Combobox.Options className="fixed z-10">
<div <div
className={`z-10 border border-custom-border-300 px-2 py-2.5 rounded bg-custom-background-100 text-xs shadow-custom-shadow-rg focus:outline-none w-48 whitespace-nowrap my-1 ${optionsClassName}`} className={`border border-custom-border-300 px-2 py-2.5 rounded bg-custom-background-100 text-xs shadow-custom-shadow-rg focus:outline-none w-48 whitespace-nowrap my-1 ${optionsClassName}`}
ref={setPopperElement} ref={setPopperElement}
style={styles.popper} style={styles.popper}
{...attributes.popper} {...attributes.popper}

View File

@ -1,23 +1,21 @@
import React, { useState } from "react"; import React, { useState } from "react";
// react-popper
import { usePopper } from "react-popper"; import { usePopper } from "react-popper";
// headless ui import { Placement } from "@popperjs/core";
import { Combobox } from "@headlessui/react"; import { Combobox } from "@headlessui/react";
// icons
import { Check, ChevronDown, Search } from "lucide-react"; import { Check, ChevronDown, Search } from "lucide-react";
import { PriorityIcon } from "@plane/ui"; import { PriorityIcon } from "@plane/ui";
// components // components
import { Tooltip } from "components/ui"; import { Tooltip } from "components/ui";
// helpers
import { capitalizeFirstLetter } from "helpers/string.helper";
// types // types
import { TIssuePriorities } from "types"; import { TIssuePriorities } from "types";
import { Placement } from "@popperjs/core";
// constants // constants
import { PRIORITIES } from "constants/project"; import { ISSUE_PRIORITIES } from "constants/issue";
type Props = { type Props = {
value: TIssuePriorities; value: TIssuePriorities;
onChange: (data: any) => void; onChange: (data: TIssuePriorities) => void;
className?: string; className?: string;
buttonClassName?: string; buttonClassName?: string;
optionsClassName?: string; optionsClassName?: string;
@ -43,15 +41,23 @@ export const PrioritySelect: React.FC<Props> = ({
const { styles, attributes } = usePopper(referenceElement, popperElement, { const { styles, attributes } = usePopper(referenceElement, popperElement, {
placement: placement ?? "bottom-start", placement: placement ?? "bottom-start",
modifiers: [
{
name: "preventOverflow",
options: {
padding: 12,
},
},
],
}); });
const options = PRIORITIES?.map((priority) => ({ const options = ISSUE_PRIORITIES?.map((priority) => ({
value: priority, value: priority.key,
query: priority, query: priority.key,
content: ( content: (
<div className="flex items-center gap-2"> <div className="flex items-center gap-2">
<PriorityIcon priority={priority} className="h-3.5 w-3.5" /> <PriorityIcon priority={priority.key} className="h-3.5 w-3.5" />
{priority ?? "None"} {priority.title}
</div> </div>
), ),
})); }));
@ -59,19 +65,13 @@ export const PrioritySelect: React.FC<Props> = ({
const filteredOptions = const filteredOptions =
query === "" ? options : options?.filter((option) => option.query.toLowerCase().includes(query.toLowerCase())); query === "" ? options : options?.filter((option) => option.query.toLowerCase().includes(query.toLowerCase()));
const selectedOption = value ?? "None"; const selectedOption = value ? capitalizeFirstLetter(value) : "None";
const label = ( const label = (
<Tooltip tooltipHeading="Priority" tooltipContent={selectedOption} position="top"> <Tooltip tooltipHeading="Priority" tooltipContent={selectedOption} position="top">
<div
className={`grid place-items-center rounded "h-6 w-6 border shadow-sm ${
value === "urgent" ? "border-red-500/20 bg-red-500" : "border-custom-border-300 bg-custom-background-100"
} items-center`}
>
<span className="flex gap-1 items-center text-custom-text-200 ">
<PriorityIcon <PriorityIcon
priority={value} priority={value}
className={`w-3.5 ${ className={`h-3.5 w-3.5 ${
value === "urgent" value === "urgent"
? "text-white" ? "text-white"
: value === "high" : value === "high"
@ -83,8 +83,6 @@ export const PrioritySelect: React.FC<Props> = ({
: "text-custom-text-200" : "text-custom-text-200"
}`} }`}
/> />
</span>
</div>
</Tooltip> </Tooltip>
); );
@ -100,17 +98,19 @@ export const PrioritySelect: React.FC<Props> = ({
<button <button
ref={setReferenceElement} ref={setReferenceElement}
type="button" type="button"
className={`flex items-center justify-between gap-1 w-full text-xs ${ className={`flex items-center justify-between gap-1 h-full w-full text-xs rounded border-[0.5px] ${
value === "urgent" ? "border-red-500/20 bg-red-500" : "border-custom-border-300"
} ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80" disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`} } ${buttonClassName}`}
> >
{label} {label}
{!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />} {!hideDropdownArrow && !disabled && <ChevronDown className="h-2.5 w-2.5" aria-hidden="true" />}
</button> </button>
</Combobox.Button> </Combobox.Button>
<Combobox.Options> <Combobox.Options className="fixed z-10">
<div <div
className={`z-10 border border-custom-border-300 px-2 py-2.5 rounded bg-custom-background-100 text-xs shadow-custom-shadow-rg focus:outline-none w-48 whitespace-nowrap my-1 ${optionsClassName}`} className={`border border-custom-border-300 px-2 py-2.5 rounded bg-custom-background-100 text-xs shadow-custom-shadow-rg focus:outline-none w-48 whitespace-nowrap my-1 ${optionsClassName}`}
ref={setPopperElement} ref={setPopperElement}
style={styles.popper} style={styles.popper}
{...attributes.popper} {...attributes.popper}
@ -134,7 +134,7 @@ export const PrioritySelect: React.FC<Props> = ({
value={option.value} value={option.value}
className={({ active, selected }) => className={({ active, selected }) =>
`flex items-center justify-between gap-2 cursor-pointer select-none truncate rounded px-1 py-1.5 ${ `flex items-center justify-between gap-2 cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active && !selected ? "bg-custom-background-80" : "" active ? "bg-custom-background-80" : ""
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}` } ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
} }
> >

View File

@ -1,21 +1,18 @@
import React, { useState } from "react"; import React, { useState } from "react";
import { usePopper } from "react-popper"; import { usePopper } from "react-popper";
import { Placement } from "@popperjs/core";
import { Combobox } from "@headlessui/react"; import { Combobox } from "@headlessui/react";
import { Check, ChevronDown, Search } from "lucide-react"; import { Check, ChevronDown, Search } from "lucide-react";
// ui // ui
import { StateGroupIcon } from "@plane/ui"; import { StateGroupIcon } from "@plane/ui";
// helpers
import { getStatesList } from "helpers/state.helper";
// types
import { Tooltip } from "components/ui"; import { Tooltip } from "components/ui";
import { Placement } from "@popperjs/core"; // types
// constants import { IState } from "types";
import { IState, IStateResponse } from "types";
type Props = { type Props = {
value: IState; value: IState;
onChange: (state: IState) => void; onChange: (state: IState) => void;
stateGroups: IStateResponse | undefined; states: IState[] | undefined;
className?: string; className?: string;
buttonClassName?: string; buttonClassName?: string;
optionsClassName?: string; optionsClassName?: string;
@ -27,7 +24,7 @@ type Props = {
export const StateSelect: React.FC<Props> = ({ export const StateSelect: React.FC<Props> = ({
value, value,
onChange, onChange,
stateGroups, states,
className = "", className = "",
buttonClassName = "", buttonClassName = "",
optionsClassName = "", optionsClassName = "",
@ -42,10 +39,16 @@ export const StateSelect: React.FC<Props> = ({
const { styles, attributes } = usePopper(referenceElement, popperElement, { const { styles, attributes } = usePopper(referenceElement, popperElement, {
placement: placement ?? "bottom-start", placement: placement ?? "bottom-start",
modifiers: [
{
name: "preventOverflow",
options: {
padding: 12,
},
},
],
}); });
const states = getStatesList(stateGroups);
const options = states?.map((state) => ({ const options = states?.map((state) => ({
value: state.id, value: state.id,
query: state.name, query: state.name,
@ -63,7 +66,7 @@ export const StateSelect: React.FC<Props> = ({
const label = ( const label = (
<Tooltip tooltipHeading="State" tooltipContent={value?.name ?? ""} position="top"> <Tooltip tooltipHeading="State" tooltipContent={value?.name ?? ""} position="top">
<div className="flex items-center cursor-pointer w-full gap-2 text-custom-text-200"> <div className="flex items-center cursor-pointer w-full gap-2 text-custom-text-200">
<span className="h-3.5 w-3.5">{value && <StateGroupIcon stateGroup={value.group} color={value.color} />}</span> {value && <StateGroupIcon stateGroup={value.group} color={value.color} />}
<span className="truncate">{value?.name ?? "State"}</span> <span className="truncate">{value?.name ?? "State"}</span>
</div> </div>
</Tooltip> </Tooltip>
@ -85,7 +88,7 @@ export const StateSelect: React.FC<Props> = ({
<button <button
ref={setReferenceElement} ref={setReferenceElement}
type="button" type="button"
className={`flex items-center justify-between gap-1 w-full text-xs px-2.5 py-1 rounded-md shadow-sm border border-custom-border-300 duration-300 focus:outline-none ${ className={`flex items-center justify-between gap-1 w-full text-xs px-2.5 py-1 rounded border-[0.5px] border-custom-border-300 duration-300 focus:outline-none ${
disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80" disabled ? "cursor-not-allowed text-custom-text-200" : "cursor-pointer hover:bg-custom-background-80"
} ${buttonClassName}`} } ${buttonClassName}`}
> >
@ -93,9 +96,9 @@ export const StateSelect: React.FC<Props> = ({
{!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />} {!hideDropdownArrow && !disabled && <ChevronDown className="h-3 w-3" aria-hidden="true" />}
</button> </button>
</Combobox.Button> </Combobox.Button>
<Combobox.Options> <Combobox.Options className="fixed z-10">
<div <div
className={`z-10 border border-custom-border-300 px-2 py-2.5 rounded bg-custom-background-100 text-xs shadow-custom-shadow-rg focus:outline-none w-48 whitespace-nowrap my-1 ${optionsClassName}`} className={`border border-custom-border-300 px-2 py-2.5 rounded bg-custom-background-100 text-xs shadow-custom-shadow-rg focus:outline-none w-48 whitespace-nowrap my-1 ${optionsClassName}`}
ref={setPopperElement} ref={setPopperElement}
style={styles.popper} style={styles.popper}
{...attributes.popper} {...attributes.popper}
@ -119,7 +122,7 @@ export const StateSelect: React.FC<Props> = ({
value={option.value} value={option.value}
className={({ active, selected }) => className={({ active, selected }) =>
`flex items-center justify-between gap-2 cursor-pointer select-none truncate rounded px-1 py-1.5 ${ `flex items-center justify-between gap-2 cursor-pointer select-none truncate rounded px-1 py-1.5 ${
active && !selected ? "bg-custom-background-80" : "" active ? "bg-custom-background-80" : ""
} ${selected ? "text-custom-text-100" : "text-custom-text-200"}` } ${selected ? "text-custom-text-100" : "text-custom-text-200"}`
} }
> >

View File

@ -65,7 +65,13 @@ export const ProjectAuthWrapper: FC<IProjectAuthWrapper> = observer((props) => {
? () => projectStore.fetchProjectStates(workspaceSlug.toString(), projectId.toString()) ? () => projectStore.fetchProjectStates(workspaceSlug.toString(), projectId.toString())
: null : null
); );
// TODO: fetch project estimates // fetching project estimates
useSWR(
workspaceSlug && projectId ? `PROJECT_ESTIMATES_${workspaceSlug}_${projectId}` : null,
workspaceSlug && projectId
? () => projectStore.fetchProjectEstimates(workspaceSlug.toString(), projectId.toString())
: null
);
// fetching project cycles // fetching project cycles
useSWR( useSWR(
workspaceSlug && projectId ? `PROJECT_ALL_CYCLES_${workspaceSlug}_${projectId}` : null, workspaceSlug && projectId ? `PROJECT_ALL_CYCLES_${workspaceSlug}_${projectId}` : null,

View File

@ -8,8 +8,8 @@ import { AppLayout } from "layouts/app-layout";
import { ProfileAuthWrapper } from "layouts/profile-layout"; import { ProfileAuthWrapper } from "layouts/profile-layout";
// components // components
import { UserProfileHeader } from "components/headers"; import { UserProfileHeader } from "components/headers";
import { ProfileIssuesListLayout } from "components/issues/issue-layouts/list/profile-issues-root"; import { ProfileIssuesListLayout } from "components/issues/issue-layouts/list/roots/profile-issues-root";
import { ProfileIssuesKanBanLayout } from "components/issues/issue-layouts/kanban/profile-issues-root"; import { ProfileIssuesKanBanLayout } from "components/issues/issue-layouts/kanban/roots/profile-issues-root";
// hooks // hooks
import { useMobxStore } from "lib/mobx/store-provider"; import { useMobxStore } from "lib/mobx/store-provider";
import { RootStore } from "store/root"; import { RootStore } from "store/root";