forked from github/plane
9831418a11
* chore: project issues topbar * style: theming and minor UI fixes * refactor: file structure * chore: layout wise authorization added * style: filter dropdowns * chore: add fetch keys * feat: search option for filters * fix: sticky headers * chore: sub_group_by section added
298 lines
7.6 KiB
TypeScript
298 lines
7.6 KiB
TypeScript
import { orderArrayBy } from "helpers/array.helper";
|
|
import { renderDateFormat } from "helpers/date-time.helper";
|
|
// types
|
|
import { IIssue, TIssueGroupByOptions, TIssueLayouts, TIssueOrderByOptions, TIssueParams } from "types";
|
|
|
|
type THandleIssuesMutation = (
|
|
formData: Partial<IIssue>,
|
|
oldGroupTitle: string,
|
|
selectedGroupBy: TIssueGroupByOptions,
|
|
issueIndex: number,
|
|
orderBy: TIssueOrderByOptions,
|
|
prevData?:
|
|
| {
|
|
[key: string]: IIssue[];
|
|
}
|
|
| IIssue[]
|
|
) =>
|
|
| {
|
|
[key: string]: IIssue[];
|
|
}
|
|
| IIssue[]
|
|
| undefined;
|
|
|
|
export const handleIssuesMutation: THandleIssuesMutation = (
|
|
formData,
|
|
oldGroupTitle,
|
|
selectedGroupBy,
|
|
issueIndex,
|
|
orderBy,
|
|
prevData
|
|
) => {
|
|
if (!prevData) return prevData;
|
|
|
|
if (Array.isArray(prevData)) {
|
|
const updatedIssue = {
|
|
...prevData[issueIndex],
|
|
...formData,
|
|
assignees: formData?.assignees_list ?? prevData[issueIndex]?.assignees,
|
|
labels: formData?.labels_list ?? prevData[issueIndex]?.labels,
|
|
};
|
|
|
|
prevData.splice(issueIndex, 1, updatedIssue);
|
|
|
|
return [...prevData];
|
|
} else {
|
|
const oldGroup = prevData[oldGroupTitle ?? ""] ?? [];
|
|
|
|
let newGroup: IIssue[] = [];
|
|
|
|
if (selectedGroupBy === "priority") newGroup = prevData[formData.priority ?? ""] ?? [];
|
|
else if (selectedGroupBy === "state") newGroup = prevData[formData.state ?? ""] ?? [];
|
|
|
|
const updatedIssue = {
|
|
...oldGroup[issueIndex],
|
|
...formData,
|
|
assignees: formData?.assignees_list ?? oldGroup[issueIndex]?.assignees,
|
|
labels: formData?.labels_list ?? oldGroup[issueIndex]?.labels,
|
|
};
|
|
|
|
if (selectedGroupBy !== Object.keys(formData)[0])
|
|
return {
|
|
...prevData,
|
|
[oldGroupTitle ?? ""]: orderArrayBy(
|
|
oldGroup.map((i) => (i.id === updatedIssue.id ? updatedIssue : i)),
|
|
orderBy
|
|
),
|
|
};
|
|
|
|
const groupThatIsUpdated = selectedGroupBy === "priority" ? formData.priority : formData.state;
|
|
|
|
return {
|
|
...prevData,
|
|
[oldGroupTitle ?? ""]: orderArrayBy(
|
|
oldGroup.filter((i) => i.id !== updatedIssue.id),
|
|
orderBy
|
|
),
|
|
[groupThatIsUpdated ?? ""]: orderArrayBy([...newGroup, updatedIssue], orderBy),
|
|
};
|
|
}
|
|
};
|
|
|
|
export const handleIssueQueryParamsByLayout = (_layout: TIssueLayouts | undefined): TIssueParams[] | null => {
|
|
if (_layout === "list")
|
|
return [
|
|
"priority",
|
|
"state_group",
|
|
"state",
|
|
"assignees",
|
|
"created_by",
|
|
"labels",
|
|
"start_date",
|
|
"target_date",
|
|
"group_by",
|
|
"sub_group_by",
|
|
"order_by",
|
|
"type",
|
|
"sub_issue",
|
|
"show_empty_groups",
|
|
];
|
|
if (_layout === "kanban")
|
|
return [
|
|
"priority",
|
|
"state_group",
|
|
"state",
|
|
"assignees",
|
|
"created_by",
|
|
"labels",
|
|
"start_date",
|
|
"target_date",
|
|
"group_by",
|
|
"sub_group_by",
|
|
"order_by",
|
|
"type",
|
|
"sub_issue",
|
|
"show_empty_groups",
|
|
];
|
|
if (_layout === "calendar")
|
|
return [
|
|
"priority",
|
|
"state_group",
|
|
"state",
|
|
"assignees",
|
|
"created_by",
|
|
"labels",
|
|
"start_date",
|
|
"target_date",
|
|
"type",
|
|
"calendar_date_range",
|
|
];
|
|
if (_layout === "spreadsheet")
|
|
return [
|
|
"priority",
|
|
"state_group",
|
|
"state",
|
|
"assignees",
|
|
"created_by",
|
|
"labels",
|
|
"start_date",
|
|
"target_date",
|
|
"type",
|
|
"sub_issue",
|
|
];
|
|
if (_layout === "gantt_chart")
|
|
return [
|
|
"priority",
|
|
"state",
|
|
"assignees",
|
|
"created_by",
|
|
"labels",
|
|
"start_date",
|
|
"target_date",
|
|
"order_by",
|
|
"type",
|
|
"sub_issue",
|
|
"start_target_date",
|
|
];
|
|
|
|
return null;
|
|
};
|
|
|
|
export const handleIssueParamsDateFormat = (key: string, start_date: any | null, target_date: any | null) => {
|
|
if (key === "last_week")
|
|
return `${renderDateFormat(new Date(new Date().getTime() - 7 * 24 * 60 * 60 * 1000))};after,${renderDateFormat(
|
|
new Date()
|
|
)};before`;
|
|
|
|
if (key === "2_weeks_from_now")
|
|
return `${renderDateFormat(new Date())};after,
|
|
${renderDateFormat(new Date(new Date().getTime() + 14 * 24 * 60 * 60 * 1000))};before`;
|
|
|
|
if (key === "1_month_from_now")
|
|
return `${renderDateFormat(new Date())};after,${renderDateFormat(
|
|
new Date(new Date().getFullYear(), new Date().getMonth() + 1, new Date().getDate())
|
|
)};before`;
|
|
|
|
if (key === "2_months_from_now")
|
|
return `${renderDateFormat(new Date())};after,${renderDateFormat(
|
|
new Date(new Date().getFullYear(), new Date().getMonth() + 2, new Date().getDate())
|
|
)};before`;
|
|
|
|
if (key === "custom" && start_date && target_date)
|
|
return `${renderDateFormat(start_date)};after,${renderDateFormat(target_date)};before`;
|
|
};
|
|
|
|
export const issueFilterVisibilityData: {
|
|
[key: string]: {
|
|
layout: TIssueLayouts[];
|
|
filters: {
|
|
[key in TIssueLayouts]: string[];
|
|
};
|
|
display_properties: {
|
|
[key in TIssueLayouts]: boolean;
|
|
};
|
|
display_filters: {
|
|
[key in TIssueLayouts]: string[];
|
|
};
|
|
extra_options: {
|
|
[key in TIssueLayouts]: {
|
|
access: boolean;
|
|
values: string[];
|
|
};
|
|
};
|
|
};
|
|
} = {
|
|
my_issues: {
|
|
layout: ["list", "kanban"],
|
|
filters: {
|
|
list: ["priority", "state_group", "labels", "start_date", "due_date"],
|
|
kanban: ["priority", "state_group", "labels", "start_date", "due_date"],
|
|
calendar: [],
|
|
spreadsheet: [],
|
|
gantt_chart: [],
|
|
},
|
|
display_properties: {
|
|
list: true,
|
|
kanban: true,
|
|
calendar: true,
|
|
spreadsheet: true,
|
|
gantt_chart: false,
|
|
},
|
|
display_filters: {
|
|
list: ["group_by", "order_by", "issue_type"],
|
|
kanban: ["group_by", "order_by", "issue_type"],
|
|
calendar: ["issue_type"],
|
|
spreadsheet: ["issue_type"],
|
|
gantt_chart: ["order_by", "issue_type"],
|
|
},
|
|
extra_options: {
|
|
list: {
|
|
access: true,
|
|
values: ["show_empty_groups", "sub_issue"],
|
|
},
|
|
kanban: {
|
|
access: true,
|
|
values: ["show_empty_groups", "sub_issue"],
|
|
},
|
|
calendar: {
|
|
access: false,
|
|
values: [],
|
|
},
|
|
spreadsheet: {
|
|
access: false,
|
|
values: [],
|
|
},
|
|
gantt_chart: {
|
|
access: false,
|
|
values: [],
|
|
},
|
|
},
|
|
},
|
|
issues: {
|
|
layout: ["list", "kanban", "calendar", "spreadsheet", "gantt_chart"],
|
|
filters: {
|
|
list: ["priority", "state", "assignees", "created_by", "labels", "start_date", "due_date"],
|
|
kanban: ["priority", "state", "assignees", "created_by", "labels", "start_date", "due_date"],
|
|
calendar: ["priority", "state", "assignees", "created_by", "labels"],
|
|
spreadsheet: ["priority", "state", "assignees", "created_by", "labels", "start_date", "due_date"],
|
|
gantt_chart: ["priority", "state", "assignees", "created_by", "labels", "start_date", "due_date"],
|
|
},
|
|
display_properties: {
|
|
list: true,
|
|
kanban: true,
|
|
calendar: true,
|
|
spreadsheet: true,
|
|
gantt_chart: false,
|
|
},
|
|
display_filters: {
|
|
list: ["group_by", "order_by", "issue_type"],
|
|
kanban: ["group_by", "order_by", "issue_type"],
|
|
calendar: ["issue_type"],
|
|
spreadsheet: ["issue_type"],
|
|
gantt_chart: ["order_by", "issue_type"],
|
|
},
|
|
extra_options: {
|
|
list: {
|
|
access: true,
|
|
values: ["show_empty_groups", "sub_issue"],
|
|
},
|
|
kanban: {
|
|
access: true,
|
|
values: ["show_empty_groups", "sub_issue"],
|
|
},
|
|
calendar: {
|
|
access: false,
|
|
values: [],
|
|
},
|
|
spreadsheet: {
|
|
access: false,
|
|
values: [],
|
|
},
|
|
gantt_chart: {
|
|
access: true,
|
|
values: ["sub_issue"],
|
|
},
|
|
},
|
|
},
|
|
};
|