Fixed filter dropdown on task page (#3469)
* Fixed filter dropdown on task page * Fixed ts in test * Change avatarUrl to nullable in ObjectRecordIdentifier
This commit is contained in:
@ -4,6 +4,7 @@ import { getBasePathToShowPage } from '@/object-metadata/utils/getBasePathToShow
|
|||||||
import { getLabelIdentifierFieldMetadataItem } from '@/object-metadata/utils/getLabelIdentifierFieldMetadataItem';
|
import { getLabelIdentifierFieldMetadataItem } from '@/object-metadata/utils/getLabelIdentifierFieldMetadataItem';
|
||||||
import { ObjectRecord } from '@/object-record/types/ObjectRecord';
|
import { ObjectRecord } from '@/object-record/types/ObjectRecord';
|
||||||
import { ObjectRecordIdentifier } from '@/object-record/types/ObjectRecordIdentifier';
|
import { ObjectRecordIdentifier } from '@/object-record/types/ObjectRecordIdentifier';
|
||||||
|
import { WorkspaceMember } from '@/workspace-member/types/WorkspaceMember';
|
||||||
import { FieldMetadataType } from '~/generated-metadata/graphql';
|
import { FieldMetadataType } from '~/generated-metadata/graphql';
|
||||||
import { getLogoUrlFromDomainName } from '~/utils';
|
import { getLogoUrlFromDomainName } from '~/utils';
|
||||||
|
|
||||||
@ -14,14 +15,27 @@ export const getObjectRecordIdentifier = ({
|
|||||||
objectMetadataItem: ObjectMetadataItem;
|
objectMetadataItem: ObjectMetadataItem;
|
||||||
record: ObjectRecord;
|
record: ObjectRecord;
|
||||||
}): ObjectRecordIdentifier => {
|
}): ObjectRecordIdentifier => {
|
||||||
if (objectMetadataItem.nameSingular === CoreObjectNameSingular.Opportunity) {
|
switch (objectMetadataItem.nameSingular) {
|
||||||
return {
|
case CoreObjectNameSingular.Opportunity: {
|
||||||
id: record.id,
|
return {
|
||||||
name: record?.company?.name ?? record.name,
|
id: record.id,
|
||||||
avatarUrl: record.avatarUrl,
|
name: record?.company?.name,
|
||||||
avatarType: 'rounded',
|
avatarUrl: record.avatarUrl,
|
||||||
linkToShowPage: `/opportunities/${record.id}`,
|
avatarType: 'rounded',
|
||||||
};
|
linkToShowPage: `/opportunities/${record.id}`,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
case CoreObjectNameSingular.WorkspaceMember: {
|
||||||
|
const workspaceMember = record as WorkspaceMember;
|
||||||
|
|
||||||
|
return {
|
||||||
|
id: workspaceMember.id,
|
||||||
|
name:
|
||||||
|
workspaceMember.name.firstName + ' ' + workspaceMember.name.lastName,
|
||||||
|
avatarUrl: workspaceMember.avatarUrl ?? undefined,
|
||||||
|
avatarType: 'rounded',
|
||||||
|
};
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const labelIdentifierFieldMetadataItem =
|
const labelIdentifierFieldMetadataItem =
|
||||||
|
|||||||
@ -0,0 +1,27 @@
|
|||||||
|
import { MenuItem } from 'tsup.ui.index';
|
||||||
|
|
||||||
|
import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown';
|
||||||
|
import { IconFilterOff } from '@/ui/display/icon';
|
||||||
|
import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer';
|
||||||
|
import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown';
|
||||||
|
|
||||||
|
export const ObjectFilterDropdownRecordRemoveFilterMenuItem = () => {
|
||||||
|
const { emptyFilterButKeepDefinition } = useFilterDropdown();
|
||||||
|
|
||||||
|
const { closeDropdown } = useDropdown();
|
||||||
|
|
||||||
|
const handleRemoveFilter = () => {
|
||||||
|
emptyFilterButKeepDefinition();
|
||||||
|
closeDropdown();
|
||||||
|
};
|
||||||
|
|
||||||
|
return (
|
||||||
|
<DropdownMenuItemsContainer>
|
||||||
|
<MenuItem
|
||||||
|
onClick={handleRemoveFilter}
|
||||||
|
LeftIcon={IconFilterOff}
|
||||||
|
text={'Remove filter'}
|
||||||
|
/>
|
||||||
|
</DropdownMenuItemsContainer>
|
||||||
|
);
|
||||||
|
};
|
||||||
@ -3,7 +3,7 @@ import { MultipleRecordSelectDropdown } from '@/object-record/select/components/
|
|||||||
import { useRecordsForSelect } from '@/object-record/select/hooks/useRecordsForSelect';
|
import { useRecordsForSelect } from '@/object-record/select/hooks/useRecordsForSelect';
|
||||||
import { SelectableRecord } from '@/object-record/select/types/SelectableRecord';
|
import { SelectableRecord } from '@/object-record/select/types/SelectableRecord';
|
||||||
|
|
||||||
export const EMPTY_FILTER_VALUE = '';
|
export const EMPTY_FILTER_VALUE = '[]';
|
||||||
export const MAX_RECORDS_TO_DISPLAY = 3;
|
export const MAX_RECORDS_TO_DISPLAY = 3;
|
||||||
|
|
||||||
export const ObjectFilterDropdownRecordSelect = () => {
|
export const ObjectFilterDropdownRecordSelect = () => {
|
||||||
@ -14,6 +14,7 @@ export const ObjectFilterDropdownRecordSelect = () => {
|
|||||||
setObjectFilterDropdownSelectedRecordIds,
|
setObjectFilterDropdownSelectedRecordIds,
|
||||||
objectFilterDropdownSelectedRecordIds,
|
objectFilterDropdownSelectedRecordIds,
|
||||||
selectFilter,
|
selectFilter,
|
||||||
|
emptyFilterButKeepDefinition,
|
||||||
} = useFilterDropdown();
|
} = useFilterDropdown();
|
||||||
|
|
||||||
const objectNameSingular =
|
const objectNameSingular =
|
||||||
@ -41,6 +42,11 @@ export const ObjectFilterDropdownRecordSelect = () => {
|
|||||||
(id) => id !== recordToSelect.id,
|
(id) => id !== recordToSelect.id,
|
||||||
);
|
);
|
||||||
|
|
||||||
|
if (newSelectedRecordIds.length === 0) {
|
||||||
|
emptyFilterButKeepDefinition();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
setObjectFilterDropdownSelectedRecordIds(newSelectedRecordIds);
|
setObjectFilterDropdownSelectedRecordIds(newSelectedRecordIds);
|
||||||
|
|
||||||
const selectedRecordNames = [
|
const selectedRecordNames = [
|
||||||
|
|||||||
@ -1,9 +1,11 @@
|
|||||||
import React from 'react';
|
import React from 'react';
|
||||||
import { useTheme } from '@emotion/react';
|
import { useTheme } from '@emotion/react';
|
||||||
|
|
||||||
|
import { ObjectFilterDropdownRecordRemoveFilterMenuItem } from '@/object-record/object-filter-dropdown/components/ObjectFilterDropdownRecordRemoveFilterMenuItem';
|
||||||
import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown';
|
import { useFilterDropdown } from '@/object-record/object-filter-dropdown/hooks/useFilterDropdown';
|
||||||
import { IconChevronDown } from '@/ui/display/icon/index';
|
import { IconChevronDown } from '@/ui/display/icon/index';
|
||||||
import { Dropdown } from '@/ui/layout/dropdown/components/Dropdown';
|
import { Dropdown } from '@/ui/layout/dropdown/components/Dropdown';
|
||||||
|
import { DropdownMenuSeparator } from '@/ui/layout/dropdown/components/DropdownMenuSeparator';
|
||||||
import { StyledHeaderDropdownButton } from '@/ui/layout/dropdown/components/StyledHeaderDropdownButton';
|
import { StyledHeaderDropdownButton } from '@/ui/layout/dropdown/components/StyledHeaderDropdownButton';
|
||||||
import { HotkeyScope } from '@/ui/utilities/hotkey/types/HotkeyScope';
|
import { HotkeyScope } from '@/ui/utilities/hotkey/types/HotkeyScope';
|
||||||
import { ViewFilterOperand } from '@/views/types/ViewFilterOperand';
|
import { ViewFilterOperand } from '@/views/types/ViewFilterOperand';
|
||||||
@ -65,6 +67,8 @@ export const SingleEntityObjectFilterDropdownButton = ({
|
|||||||
dropdownComponents={
|
dropdownComponents={
|
||||||
<>
|
<>
|
||||||
<ObjectFilterDropdownRecordSearchInput />
|
<ObjectFilterDropdownRecordSearchInput />
|
||||||
|
<DropdownMenuSeparator />
|
||||||
|
<ObjectFilterDropdownRecordRemoveFilterMenuItem />
|
||||||
<ObjectFilterDropdownRecordSelect />
|
<ObjectFilterDropdownRecordSelect />
|
||||||
</>
|
</>
|
||||||
}
|
}
|
||||||
|
|||||||
@ -59,7 +59,7 @@ describe('useFilterDropdown', () => {
|
|||||||
|
|
||||||
act(() => {
|
act(() => {
|
||||||
result.current.setOnFilterSelect(
|
result.current.setOnFilterSelect(
|
||||||
(_currVal?: Filter) => (_filter: Filter) => {},
|
(_currVal?: Filter | null) => (_filter: Filter | null) => {},
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
await waitFor(() => {
|
await waitFor(() => {
|
||||||
|
|||||||
@ -40,13 +40,25 @@ export const useFilterDropdown = (props?: UseFilterDropdownProps) => {
|
|||||||
} = useFilterDropdownStates(scopeId);
|
} = useFilterDropdownStates(scopeId);
|
||||||
|
|
||||||
const selectFilter = useCallback(
|
const selectFilter = useCallback(
|
||||||
(filter: Filter) => {
|
(filter: Filter | null) => {
|
||||||
setSelectedFilter(filter);
|
setSelectedFilter(filter);
|
||||||
onFilterSelect?.(filter);
|
onFilterSelect?.(filter);
|
||||||
},
|
},
|
||||||
[setSelectedFilter, onFilterSelect],
|
[setSelectedFilter, onFilterSelect],
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const emptyFilterButKeepDefinition = useCallback(() => {
|
||||||
|
setObjectFilterDropdownSearchInput('');
|
||||||
|
setObjectFilterDropdownSelectedEntityId(null);
|
||||||
|
setObjectFilterDropdownSelectedRecordIds([]);
|
||||||
|
setSelectedFilter(undefined);
|
||||||
|
}, [
|
||||||
|
setSelectedFilter,
|
||||||
|
setObjectFilterDropdownSelectedRecordIds,
|
||||||
|
setObjectFilterDropdownSelectedEntityId,
|
||||||
|
setObjectFilterDropdownSearchInput,
|
||||||
|
]);
|
||||||
|
|
||||||
const resetFilter = useCallback(() => {
|
const resetFilter = useCallback(() => {
|
||||||
setObjectFilterDropdownSearchInput('');
|
setObjectFilterDropdownSearchInput('');
|
||||||
setObjectFilterDropdownSelectedEntityId(null);
|
setObjectFilterDropdownSelectedEntityId(null);
|
||||||
@ -87,5 +99,6 @@ export const useFilterDropdown = (props?: UseFilterDropdownProps) => {
|
|||||||
resetFilter,
|
resetFilter,
|
||||||
onFilterSelect,
|
onFilterSelect,
|
||||||
setOnFilterSelect,
|
setOnFilterSelect,
|
||||||
|
emptyFilterButKeepDefinition,
|
||||||
};
|
};
|
||||||
};
|
};
|
||||||
|
|||||||
@ -3,7 +3,7 @@ import { createStateScopeMap } from '@/ui/utilities/recoil-scope/utils/createSta
|
|||||||
import { Filter } from '../types/Filter';
|
import { Filter } from '../types/Filter';
|
||||||
|
|
||||||
export const onFilterSelectScopedState = createStateScopeMap<
|
export const onFilterSelectScopedState = createStateScopeMap<
|
||||||
((filter: Filter) => void) | undefined
|
((filter: Filter | null) => void) | undefined
|
||||||
>({
|
>({
|
||||||
key: 'onFilterSelectScopedState',
|
key: 'onFilterSelectScopedState',
|
||||||
defaultValue: undefined,
|
defaultValue: undefined,
|
||||||
|
|||||||
@ -3,7 +3,7 @@ import { createStateScopeMap } from '@/ui/utilities/recoil-scope/utils/createSta
|
|||||||
import { Filter } from '../types/Filter';
|
import { Filter } from '../types/Filter';
|
||||||
|
|
||||||
export const selectedFilterScopedState = createStateScopeMap<
|
export const selectedFilterScopedState = createStateScopeMap<
|
||||||
Filter | undefined
|
Filter | undefined | null
|
||||||
>({
|
>({
|
||||||
key: 'selectedFilterScopedState',
|
key: 'selectedFilterScopedState',
|
||||||
defaultValue: undefined,
|
defaultValue: undefined,
|
||||||
|
|||||||
@ -3,7 +3,7 @@ import { AvatarType } from '@/users/components/Avatar';
|
|||||||
export type ObjectRecordIdentifier = {
|
export type ObjectRecordIdentifier = {
|
||||||
id: string;
|
id: string;
|
||||||
name: string;
|
name: string;
|
||||||
avatarUrl: string;
|
avatarUrl?: string;
|
||||||
avatarType?: AvatarType | null;
|
avatarType?: AvatarType | null;
|
||||||
linkToShowPage?: string;
|
linkToShowPage?: string;
|
||||||
};
|
};
|
||||||
|
|||||||
@ -57,6 +57,7 @@ export {
|
|||||||
IconFileText,
|
IconFileText,
|
||||||
IconFileUpload,
|
IconFileUpload,
|
||||||
IconFileZip,
|
IconFileZip,
|
||||||
|
IconFilterOff,
|
||||||
IconForbid,
|
IconForbid,
|
||||||
IconGripVertical,
|
IconGripVertical,
|
||||||
IconHeadphones,
|
IconHeadphones,
|
||||||
|
|||||||
@ -13,10 +13,13 @@ type TasksEffectProps = {
|
|||||||
|
|
||||||
export const TasksEffect = ({ filterDropdownId }: TasksEffectProps) => {
|
export const TasksEffect = ({ filterDropdownId }: TasksEffectProps) => {
|
||||||
const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState);
|
const currentWorkspaceMember = useRecoilValue(currentWorkspaceMemberState);
|
||||||
const { setSelectedFilter, setAvailableFilterDefinitions } =
|
const {
|
||||||
useFilterDropdown({
|
setSelectedFilter,
|
||||||
filterDropdownId: filterDropdownId,
|
setAvailableFilterDefinitions,
|
||||||
});
|
setObjectFilterDropdownSelectedRecordIds,
|
||||||
|
} = useFilterDropdown({
|
||||||
|
filterDropdownId: filterDropdownId,
|
||||||
|
});
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setAvailableFilterDefinitions(tasksFilterDefinitions);
|
setAvailableFilterDefinitions(tasksFilterDefinitions);
|
||||||
@ -35,7 +38,13 @@ export const TasksEffect = ({ filterDropdownId }: TasksEffectProps) => {
|
|||||||
displayAvatarUrl: currentWorkspaceMember.avatarUrl ?? undefined,
|
displayAvatarUrl: currentWorkspaceMember.avatarUrl ?? undefined,
|
||||||
definition: tasksFilterDefinitions[0],
|
definition: tasksFilterDefinitions[0],
|
||||||
});
|
});
|
||||||
|
|
||||||
|
setObjectFilterDropdownSelectedRecordIds([currentWorkspaceMember.id]);
|
||||||
}
|
}
|
||||||
}, [currentWorkspaceMember, setSelectedFilter]);
|
}, [
|
||||||
|
currentWorkspaceMember,
|
||||||
|
setSelectedFilter,
|
||||||
|
setObjectFilterDropdownSelectedRecordIds,
|
||||||
|
]);
|
||||||
return <></>;
|
return <></>;
|
||||||
};
|
};
|
||||||
|
|||||||
Reference in New Issue
Block a user