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:
Lucas Bordeau
2024-01-23 16:59:26 +01:00
committed by GitHub
parent 2b6d66f1bc
commit 23a3614b54
11 changed files with 93 additions and 19 deletions

View File

@ -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 =

View File

@ -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>
);
};

View File

@ -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 = [

View File

@ -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 />
</> </>
} }

View File

@ -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(() => {

View File

@ -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,
}; };
}; };

View File

@ -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,

View File

@ -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,

View File

@ -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;
}; };

View File

@ -57,6 +57,7 @@ export {
IconFileText, IconFileText,
IconFileUpload, IconFileUpload,
IconFileZip, IconFileZip,
IconFilterOff,
IconForbid, IconForbid,
IconGripVertical, IconGripVertical,
IconHeadphones, IconHeadphones,

View File

@ -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 <></>;
}; };