forked from github/plane
fix: spreadsheet date validation and sorting (#3607)
* fix validation for start and end date in spreadsheet layout * revamp logic for sorting in all fields
This commit is contained in:
parent
41a3cb708c
commit
8d730e6680
@ -21,6 +21,7 @@ export const SpreadsheetDueDateColumn: React.FC<Props> = observer((props: Props)
|
|||||||
<div className="h-11 border-b-[0.5px] border-custom-border-200">
|
<div className="h-11 border-b-[0.5px] border-custom-border-200">
|
||||||
<DateDropdown
|
<DateDropdown
|
||||||
value={issue.target_date}
|
value={issue.target_date}
|
||||||
|
minDate={issue.start_date ? new Date(issue.start_date) : undefined}
|
||||||
onChange={(data) => {
|
onChange={(data) => {
|
||||||
const targetDate = data ? renderFormattedPayloadDate(data) : null;
|
const targetDate = data ? renderFormattedPayloadDate(data) : null;
|
||||||
onChange(
|
onChange(
|
||||||
|
@ -21,6 +21,7 @@ export const SpreadsheetStartDateColumn: React.FC<Props> = observer((props: Prop
|
|||||||
<div className="h-11 border-b-[0.5px] border-custom-border-200">
|
<div className="h-11 border-b-[0.5px] border-custom-border-200">
|
||||||
<DateDropdown
|
<DateDropdown
|
||||||
value={issue.start_date}
|
value={issue.start_date}
|
||||||
|
maxDate={issue.target_date ? new Date(issue.target_date) : undefined}
|
||||||
onChange={(data) => {
|
onChange={(data) => {
|
||||||
const startDate = data ? renderFormattedPayloadDate(data) : null;
|
const startDate = data ? renderFormattedPayloadDate(data) : null;
|
||||||
onChange(
|
onChange(
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import sortBy from "lodash/sortBy";
|
import orderBy from "lodash/orderBy";
|
||||||
import get from "lodash/get";
|
import get from "lodash/get";
|
||||||
import indexOf from "lodash/indexOf";
|
import indexOf from "lodash/indexOf";
|
||||||
import reverse from "lodash/reverse";
|
import isEmpty from "lodash/isEmpty";
|
||||||
import values from "lodash/values";
|
import values from "lodash/values";
|
||||||
// types
|
// types
|
||||||
import { TIssue, TIssueMap, TIssueGroupByOptions, TIssueOrderByOptions } from "@plane/types";
|
import { TIssue, TIssueMap, TIssueGroupByOptions, TIssueOrderByOptions } from "@plane/types";
|
||||||
@ -144,98 +144,189 @@ export class IssueHelperStore implements TIssueHelperStore {
|
|||||||
issueDisplayFiltersDefaultData = (groupBy: string | null): string[] => {
|
issueDisplayFiltersDefaultData = (groupBy: string | null): string[] => {
|
||||||
switch (groupBy) {
|
switch (groupBy) {
|
||||||
case "state":
|
case "state":
|
||||||
return this.rootStore?.states || [];
|
return Object.keys(this.rootStore?.stateMap || {});
|
||||||
case "state_detail.group":
|
case "state_detail.group":
|
||||||
return Object.keys(STATE_GROUPS);
|
return Object.keys(STATE_GROUPS);
|
||||||
case "priority":
|
case "priority":
|
||||||
return ISSUE_PRIORITIES.map((i) => i.key);
|
return ISSUE_PRIORITIES.map((i) => i.key);
|
||||||
case "labels":
|
case "labels":
|
||||||
return this.rootStore?.labels || [];
|
return Object.keys(this.rootStore?.labelMap || {});
|
||||||
case "created_by":
|
case "created_by":
|
||||||
return this.rootStore?.members || [];
|
return Object.keys(this.rootStore?.workSpaceMemberRolesMap || {});
|
||||||
case "assignees":
|
case "assignees":
|
||||||
return this.rootStore?.members || [];
|
return Object.keys(this.rootStore?.workSpaceMemberRolesMap || {});
|
||||||
case "project":
|
case "project":
|
||||||
return this.rootStore?.projects || [];
|
return Object.keys(this.rootStore?.projectMap || {});
|
||||||
default:
|
default:
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This Method is used to get data of the issue based on the ids of the data for states, labels adn assignees
|
||||||
|
* @param dataType what type of data is being sent
|
||||||
|
* @param dataIds id/ids of the data that is to be populated
|
||||||
|
* @param order ascending or descending for arrays of data
|
||||||
|
* @returns string | string[] of sortable fields to be used for sorting
|
||||||
|
*/
|
||||||
|
populateIssueDataForSorting(
|
||||||
|
dataType: "state_id" | "label_ids" | "assignee_ids",
|
||||||
|
dataIds: string | string[] | null | undefined,
|
||||||
|
order?: "asc" | "desc"
|
||||||
|
) {
|
||||||
|
if (!dataIds) return;
|
||||||
|
|
||||||
|
const dataValues: string[] = [];
|
||||||
|
const isDataIdsArray = Array.isArray(dataIds);
|
||||||
|
const dataIdsArray = isDataIdsArray ? dataIds : [dataIds];
|
||||||
|
|
||||||
|
switch (dataType) {
|
||||||
|
case "state_id":
|
||||||
|
const stateMap = this.rootStore?.stateMap;
|
||||||
|
if (!stateMap) break;
|
||||||
|
for (const dataId of dataIdsArray) {
|
||||||
|
const state = stateMap[dataId];
|
||||||
|
if (state && state.name) dataValues.push(state.name.toLocaleLowerCase());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case "label_ids":
|
||||||
|
const labelMap = this.rootStore?.labelMap;
|
||||||
|
if (!labelMap) break;
|
||||||
|
for (const dataId of dataIdsArray) {
|
||||||
|
const label = labelMap[dataId];
|
||||||
|
if (label && label.name) dataValues.push(label.name.toLocaleLowerCase());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case "assignee_ids":
|
||||||
|
const memberMap = this.rootStore?.memberMap;
|
||||||
|
if (!memberMap) break;
|
||||||
|
for (const dataId of dataIdsArray) {
|
||||||
|
const member = memberMap[dataId];
|
||||||
|
if (memberMap && member.first_name) dataValues.push(member.first_name.toLocaleLowerCase());
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
return isDataIdsArray ? (order ? orderBy(dataValues, undefined, [order]) : dataValues) : dataValues[0];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This Method is mainly used to filter out empty values in the begining
|
||||||
|
* @param key key of the value that is to be checked if empty
|
||||||
|
* @param object any object in which the key's value is to be checked
|
||||||
|
* @returns 1 if emoty, 0 if not empty
|
||||||
|
*/
|
||||||
|
getSortOrderToFilterEmptyValues(key: string, object: any) {
|
||||||
|
const value = object?.[key];
|
||||||
|
|
||||||
|
if (typeof value !== "number" && isEmpty(value)) return 1;
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
issuesSortWithOrderBy = (issueObject: TIssueMap, key: Partial<TIssueOrderByOptions>): TIssue[] => {
|
issuesSortWithOrderBy = (issueObject: TIssueMap, key: Partial<TIssueOrderByOptions>): TIssue[] => {
|
||||||
let array = values(issueObject);
|
let array = values(issueObject);
|
||||||
array = reverse(sortBy(array, "created_at"));
|
array = orderBy(array, "created_at");
|
||||||
|
|
||||||
switch (key) {
|
switch (key) {
|
||||||
case "sort_order":
|
case "sort_order":
|
||||||
return sortBy(array, "sort_order");
|
return orderBy(array, "sort_order");
|
||||||
|
|
||||||
case "state__name":
|
case "state__name":
|
||||||
return reverse(sortBy(array, "state"));
|
return orderBy(array, (issue) => this.populateIssueDataForSorting("state_id", issue["state_id"]));
|
||||||
case "-state__name":
|
case "-state__name":
|
||||||
return sortBy(array, "state");
|
return orderBy(array, (issue) => this.populateIssueDataForSorting("state_id", issue["state_id"]), ["desc"]);
|
||||||
|
|
||||||
// dates
|
// dates
|
||||||
case "created_at":
|
case "created_at":
|
||||||
return sortBy(array, "created_at");
|
return orderBy(array, "created_at");
|
||||||
case "-created_at":
|
case "-created_at":
|
||||||
return reverse(sortBy(array, "created_at"));
|
return orderBy(array, "created_at", ["desc"]);
|
||||||
|
|
||||||
case "updated_at":
|
case "updated_at":
|
||||||
return sortBy(array, "updated_at");
|
return orderBy(array, "updated_at");
|
||||||
case "-updated_at":
|
case "-updated_at":
|
||||||
return reverse(sortBy(array, "updated_at"));
|
return orderBy(array, "updated_at", ["desc"]);
|
||||||
|
|
||||||
case "start_date":
|
case "start_date":
|
||||||
return sortBy(array, "start_date");
|
return orderBy(array, [this.getSortOrderToFilterEmptyValues.bind(null, "start_date"), "start_date"]); //preferring sorting based on empty values to always keep the empty values below
|
||||||
case "-start_date":
|
case "-start_date":
|
||||||
return reverse(sortBy(array, "start_date"));
|
return orderBy(
|
||||||
|
array,
|
||||||
|
[this.getSortOrderToFilterEmptyValues.bind(null, "start_date"), "start_date"], //preferring sorting based on empty values to always keep the empty values below
|
||||||
|
["asc", "desc"]
|
||||||
|
);
|
||||||
|
|
||||||
case "target_date":
|
case "target_date":
|
||||||
return sortBy(array, "target_date");
|
return orderBy(array, [this.getSortOrderToFilterEmptyValues.bind(null, "target_date"), "target_date"]); //preferring sorting based on empty values to always keep the empty values below
|
||||||
case "-target_date":
|
case "-target_date":
|
||||||
return reverse(sortBy(array, "target_date"));
|
return orderBy(
|
||||||
|
array,
|
||||||
|
[this.getSortOrderToFilterEmptyValues.bind(null, "target_date"), "target_date"], //preferring sorting based on empty values to always keep the empty values below
|
||||||
|
["asc", "desc"]
|
||||||
|
);
|
||||||
|
|
||||||
// custom
|
// custom
|
||||||
case "priority": {
|
case "priority": {
|
||||||
const sortArray = ISSUE_PRIORITIES.map((i) => i.key);
|
const sortArray = ISSUE_PRIORITIES.map((i) => i.key);
|
||||||
return reverse(sortBy(array, (_issue: TIssue) => indexOf(sortArray, _issue.priority)));
|
return orderBy(array, (_issue: TIssue) => indexOf(sortArray, _issue.priority), ["desc"]);
|
||||||
}
|
}
|
||||||
case "-priority": {
|
case "-priority": {
|
||||||
const sortArray = ISSUE_PRIORITIES.map((i) => i.key);
|
const sortArray = ISSUE_PRIORITIES.map((i) => i.key);
|
||||||
return sortBy(array, (_issue: TIssue) => indexOf(sortArray, _issue.priority));
|
return orderBy(array, (_issue: TIssue) => indexOf(sortArray, _issue.priority));
|
||||||
}
|
}
|
||||||
|
|
||||||
// number
|
// number
|
||||||
case "attachment_count":
|
case "attachment_count":
|
||||||
return sortBy(array, "attachment_count");
|
return orderBy(array, "attachment_count");
|
||||||
case "-attachment_count":
|
case "-attachment_count":
|
||||||
return reverse(sortBy(array, "attachment_count"));
|
return orderBy(array, "attachment_count", ["desc"]);
|
||||||
|
|
||||||
case "estimate_point":
|
case "estimate_point":
|
||||||
return sortBy(array, "estimate_point");
|
return orderBy(array, [this.getSortOrderToFilterEmptyValues.bind(null, "estimate_point"), "estimate_point"]); //preferring sorting based on empty values to always keep the empty values below
|
||||||
case "-estimate_point":
|
case "-estimate_point":
|
||||||
return reverse(sortBy(array, "estimate_point"));
|
return orderBy(
|
||||||
|
array,
|
||||||
|
[this.getSortOrderToFilterEmptyValues.bind(null, "estimate_point"), "estimate_point"], //preferring sorting based on empty values to always keep the empty values below
|
||||||
|
["asc", "desc"]
|
||||||
|
);
|
||||||
|
|
||||||
case "link_count":
|
case "link_count":
|
||||||
return sortBy(array, "link_count");
|
return orderBy(array, "link_count");
|
||||||
case "-link_count":
|
case "-link_count":
|
||||||
return reverse(sortBy(array, "link_count"));
|
return orderBy(array, "link_count", ["desc"]);
|
||||||
|
|
||||||
case "sub_issues_count":
|
case "sub_issues_count":
|
||||||
return sortBy(array, "sub_issues_count");
|
return orderBy(array, "sub_issues_count");
|
||||||
case "-sub_issues_count":
|
case "-sub_issues_count":
|
||||||
return reverse(sortBy(array, "sub_issues_count"));
|
return orderBy(array, "sub_issues_count", ["desc"]);
|
||||||
|
|
||||||
// Array
|
// Array
|
||||||
case "labels__name":
|
case "labels__name":
|
||||||
return reverse(sortBy(array, "labels"));
|
return orderBy(array, [
|
||||||
|
this.getSortOrderToFilterEmptyValues.bind(null, "label_ids"), //preferring sorting based on empty values to always keep the empty values below
|
||||||
|
(issue) => this.populateIssueDataForSorting("label_ids", issue["label_ids"], "asc"),
|
||||||
|
]);
|
||||||
case "-labels__name":
|
case "-labels__name":
|
||||||
return sortBy(array, "labels");
|
return orderBy(
|
||||||
|
array,
|
||||||
|
[
|
||||||
|
this.getSortOrderToFilterEmptyValues.bind(null, "label_ids"), //preferring sorting based on empty values to always keep the empty values below
|
||||||
|
(issue) => this.populateIssueDataForSorting("label_ids", issue["label_ids"], "desc"),
|
||||||
|
],
|
||||||
|
["asc", "desc"]
|
||||||
|
);
|
||||||
|
|
||||||
case "assignees__first_name":
|
case "assignees__first_name":
|
||||||
return reverse(sortBy(array, "assignees"));
|
return orderBy(array, [
|
||||||
|
this.getSortOrderToFilterEmptyValues.bind(null, "assignee_ids"), //preferring sorting based on empty values to always keep the empty values below
|
||||||
|
(issue) => this.populateIssueDataForSorting("assignee_ids", issue["assignee_ids"], "asc"),
|
||||||
|
]);
|
||||||
case "-assignees__first_name":
|
case "-assignees__first_name":
|
||||||
return sortBy(array, "assignees");
|
return orderBy(
|
||||||
|
array,
|
||||||
|
[
|
||||||
|
this.getSortOrderToFilterEmptyValues.bind(null, "assignee_ids"), //preferring sorting based on empty values to always keep the empty values below
|
||||||
|
(issue) => this.populateIssueDataForSorting("assignee_ids", issue["assignee_ids"], "desc"),
|
||||||
|
],
|
||||||
|
["asc", "desc"]
|
||||||
|
);
|
||||||
|
|
||||||
default:
|
default:
|
||||||
return array;
|
return array;
|
||||||
|
@ -4,7 +4,7 @@ import isEmpty from "lodash/isEmpty";
|
|||||||
import { RootStore } from "../root.store";
|
import { RootStore } from "../root.store";
|
||||||
import { IStateStore, StateStore } from "../state.store";
|
import { IStateStore, StateStore } from "../state.store";
|
||||||
// issues data store
|
// issues data store
|
||||||
import { IState } from "@plane/types";
|
import { IIssueLabel, IProject, IState, IUserLite } from "@plane/types";
|
||||||
import { IIssueStore, IssueStore } from "./issue.store";
|
import { IIssueStore, IssueStore } from "./issue.store";
|
||||||
import { IIssueDetail, IssueDetail } from "./issue-details/root.store";
|
import { IIssueDetail, IssueDetail } from "./issue-details/root.store";
|
||||||
import { IWorkspaceIssuesFilter, WorkspaceIssuesFilter, IWorkspaceIssues, WorkspaceIssues } from "./workspace";
|
import { IWorkspaceIssuesFilter, WorkspaceIssuesFilter, IWorkspaceIssues, WorkspaceIssues } from "./workspace";
|
||||||
@ -22,6 +22,7 @@ import { IArchivedIssuesFilter, ArchivedIssuesFilter, IArchivedIssues, ArchivedI
|
|||||||
import { IDraftIssuesFilter, DraftIssuesFilter, IDraftIssues, DraftIssues } from "./draft";
|
import { IDraftIssuesFilter, DraftIssuesFilter, IDraftIssues, DraftIssues } from "./draft";
|
||||||
import { IIssueKanBanViewStore, IssueKanBanViewStore } from "./issue_kanban_view.store";
|
import { IIssueKanBanViewStore, IssueKanBanViewStore } from "./issue_kanban_view.store";
|
||||||
import { ICalendarStore, CalendarStore } from "./issue_calendar_view.store";
|
import { ICalendarStore, CalendarStore } from "./issue_calendar_view.store";
|
||||||
|
import { IWorkspaceMembership } from "store/member/workspace-member.store";
|
||||||
|
|
||||||
export interface IIssueRootStore {
|
export interface IIssueRootStore {
|
||||||
currentUserId: string | undefined;
|
currentUserId: string | undefined;
|
||||||
@ -32,11 +33,12 @@ export interface IIssueRootStore {
|
|||||||
viewId: string | undefined;
|
viewId: string | undefined;
|
||||||
globalViewId: string | undefined; // all issues view id
|
globalViewId: string | undefined; // all issues view id
|
||||||
userId: string | undefined; // user profile detail Id
|
userId: string | undefined; // user profile detail Id
|
||||||
states: string[] | undefined;
|
stateMap: Record<string, IState> | undefined;
|
||||||
stateDetails: IState[] | undefined;
|
stateDetails: IState[] | undefined;
|
||||||
labels: string[] | undefined;
|
labelMap: Record<string, IIssueLabel> | undefined;
|
||||||
members: string[] | undefined;
|
workSpaceMemberRolesMap: Record<string, IWorkspaceMembership> | undefined;
|
||||||
projects: string[] | undefined;
|
memberMap: Record<string, IUserLite> | undefined;
|
||||||
|
projectMap: Record<string, IProject> | undefined;
|
||||||
|
|
||||||
rootStore: RootStore;
|
rootStore: RootStore;
|
||||||
|
|
||||||
@ -83,11 +85,12 @@ export class IssueRootStore implements IIssueRootStore {
|
|||||||
viewId: string | undefined = undefined;
|
viewId: string | undefined = undefined;
|
||||||
globalViewId: string | undefined = undefined;
|
globalViewId: string | undefined = undefined;
|
||||||
userId: string | undefined = undefined;
|
userId: string | undefined = undefined;
|
||||||
states: string[] | undefined = undefined;
|
stateMap: Record<string, IState> | undefined = undefined;
|
||||||
stateDetails: IState[] | undefined = undefined;
|
stateDetails: IState[] | undefined = undefined;
|
||||||
labels: string[] | undefined = undefined;
|
labelMap: Record<string, IIssueLabel> | undefined = undefined;
|
||||||
members: string[] | undefined = undefined;
|
workSpaceMemberRolesMap: Record<string, IWorkspaceMembership> | undefined = undefined;
|
||||||
projects: string[] | undefined = undefined;
|
memberMap: Record<string, IUserLite> | undefined = undefined;
|
||||||
|
projectMap: Record<string, IProject> | undefined = undefined;
|
||||||
|
|
||||||
rootStore: RootStore;
|
rootStore: RootStore;
|
||||||
|
|
||||||
@ -133,11 +136,12 @@ export class IssueRootStore implements IIssueRootStore {
|
|||||||
viewId: observable.ref,
|
viewId: observable.ref,
|
||||||
userId: observable.ref,
|
userId: observable.ref,
|
||||||
globalViewId: observable.ref,
|
globalViewId: observable.ref,
|
||||||
states: observable,
|
stateMap: observable,
|
||||||
stateDetails: observable,
|
stateDetails: observable,
|
||||||
labels: observable,
|
labelMap: observable,
|
||||||
members: observable,
|
memberMap: observable,
|
||||||
projects: observable,
|
workSpaceMemberRolesMap: observable,
|
||||||
|
projectMap: observable,
|
||||||
});
|
});
|
||||||
|
|
||||||
this.rootStore = rootStore;
|
this.rootStore = rootStore;
|
||||||
@ -151,13 +155,14 @@ export class IssueRootStore implements IIssueRootStore {
|
|||||||
if (rootStore.app.router.viewId) this.viewId = rootStore.app.router.viewId;
|
if (rootStore.app.router.viewId) this.viewId = rootStore.app.router.viewId;
|
||||||
if (rootStore.app.router.globalViewId) this.globalViewId = rootStore.app.router.globalViewId;
|
if (rootStore.app.router.globalViewId) this.globalViewId = rootStore.app.router.globalViewId;
|
||||||
if (rootStore.app.router.userId) this.userId = rootStore.app.router.userId;
|
if (rootStore.app.router.userId) this.userId = rootStore.app.router.userId;
|
||||||
if (!isEmpty(rootStore?.state?.stateMap)) this.states = Object.keys(rootStore?.state?.stateMap);
|
if (!isEmpty(rootStore?.state?.stateMap)) this.stateMap = rootStore?.state?.stateMap;
|
||||||
if (!isEmpty(rootStore?.state?.projectStates)) this.stateDetails = rootStore?.state?.projectStates;
|
if (!isEmpty(rootStore?.state?.projectStates)) this.stateDetails = rootStore?.state?.projectStates;
|
||||||
if (!isEmpty(rootStore?.label?.labelMap)) this.labels = Object.keys(rootStore?.label?.labelMap);
|
if (!isEmpty(rootStore?.label?.labelMap)) this.labelMap = rootStore?.label?.labelMap;
|
||||||
if (!isEmpty(rootStore?.memberRoot?.workspace?.workspaceMemberMap))
|
if (!isEmpty(rootStore?.memberRoot?.workspace?.workspaceMemberMap))
|
||||||
this.members = Object.keys(rootStore?.memberRoot?.workspace?.workspaceMemberMap);
|
this.workSpaceMemberRolesMap = rootStore?.memberRoot?.workspace?.memberMap || undefined;
|
||||||
|
if (!isEmpty(rootStore?.memberRoot?.memberMap)) this.memberMap = rootStore?.memberRoot?.memberMap || undefined;
|
||||||
if (!isEmpty(rootStore?.projectRoot?.project?.projectMap))
|
if (!isEmpty(rootStore?.projectRoot?.project?.projectMap))
|
||||||
this.projects = Object.keys(rootStore?.projectRoot?.project?.projectMap);
|
this.projectMap = rootStore?.projectRoot?.project?.projectMap;
|
||||||
});
|
});
|
||||||
|
|
||||||
this.issues = new IssueStore();
|
this.issues = new IssueStore();
|
||||||
|
@ -26,6 +26,7 @@ export interface IWorkspaceMemberStore {
|
|||||||
// computed
|
// computed
|
||||||
workspaceMemberIds: string[] | null;
|
workspaceMemberIds: string[] | null;
|
||||||
workspaceMemberInvitationIds: string[] | null;
|
workspaceMemberInvitationIds: string[] | null;
|
||||||
|
memberMap: Record<string, IWorkspaceMembership> | null;
|
||||||
// computed actions
|
// computed actions
|
||||||
getSearchedWorkspaceMemberIds: (searchQuery: string) => string[] | null;
|
getSearchedWorkspaceMemberIds: (searchQuery: string) => string[] | null;
|
||||||
getSearchedWorkspaceInvitationIds: (searchQuery: string) => string[] | null;
|
getSearchedWorkspaceInvitationIds: (searchQuery: string) => string[] | null;
|
||||||
@ -68,6 +69,7 @@ export class WorkspaceMemberStore implements IWorkspaceMemberStore {
|
|||||||
// computed
|
// computed
|
||||||
workspaceMemberIds: computed,
|
workspaceMemberIds: computed,
|
||||||
workspaceMemberInvitationIds: computed,
|
workspaceMemberInvitationIds: computed,
|
||||||
|
memberMap: computed,
|
||||||
// actions
|
// actions
|
||||||
fetchWorkspaceMembers: action,
|
fetchWorkspaceMembers: action,
|
||||||
updateMember: action,
|
updateMember: action,
|
||||||
@ -100,6 +102,12 @@ export class WorkspaceMemberStore implements IWorkspaceMemberStore {
|
|||||||
return memberIds;
|
return memberIds;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
get memberMap() {
|
||||||
|
const workspaceSlug = this.routerStore.workspaceSlug;
|
||||||
|
if (!workspaceSlug) return null;
|
||||||
|
return this.workspaceMemberMap?.[workspaceSlug] ?? {};
|
||||||
|
}
|
||||||
|
|
||||||
get workspaceMemberInvitationIds() {
|
get workspaceMemberInvitationIds() {
|
||||||
const workspaceSlug = this.routerStore.workspaceSlug;
|
const workspaceSlug = this.routerStore.workspaceSlug;
|
||||||
if (!workspaceSlug) return null;
|
if (!workspaceSlug) return null;
|
||||||
|
Loading…
Reference in New Issue
Block a user