refactor: improve SingleEntitySelect empty option (#1543)
Closes #1331 Co-authored-by: Charles Bochet <charles@twenty.com>
This commit is contained in:
@ -97,13 +97,11 @@ export function ActivityAssigneePicker({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<SingleEntitySelect
|
<SingleEntitySelect
|
||||||
onEntitySelected={handleEntitySelected}
|
entitiesToSelect={users.entitiesToSelect}
|
||||||
|
loading={users.loading}
|
||||||
onCancel={onCancel}
|
onCancel={onCancel}
|
||||||
entities={{
|
onEntitySelected={handleEntitySelected}
|
||||||
loading: users.loading,
|
selectedEntity={users.selectedEntities[0]}
|
||||||
entitiesToSelect: users.entitiesToSelect,
|
|
||||||
selectedEntity: users.selectedEntities[0],
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -34,13 +34,11 @@ export function CompanyPicker({ companyId, onSubmit, onCancel }: OwnProps) {
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<SingleEntitySelect
|
<SingleEntitySelect
|
||||||
onEntitySelected={handleEntitySelected}
|
entitiesToSelect={companies.entitiesToSelect}
|
||||||
|
loading={companies.loading}
|
||||||
onCancel={onCancel}
|
onCancel={onCancel}
|
||||||
entities={{
|
onEntitySelected={handleEntitySelected}
|
||||||
loading: companies.loading,
|
selectedEntity={companies.selectedEntities[0]}
|
||||||
entitiesToSelect: companies.entitiesToSelect,
|
|
||||||
selectedEntity: companies.selectedEntities[0],
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,4 +1,5 @@
|
|||||||
import { useFilteredSearchCompanyQuery } from '@/companies/hooks/useFilteredSearchCompanyQuery';
|
import { useFilteredSearchCompanyQuery } from '@/companies/hooks/useFilteredSearchCompanyQuery';
|
||||||
|
import { IconBuildingSkyscraper } from '@/ui/icon';
|
||||||
import { SingleEntitySelect } from '@/ui/input/relation-picker/components/SingleEntitySelect';
|
import { SingleEntitySelect } from '@/ui/input/relation-picker/components/SingleEntitySelect';
|
||||||
import { relationPickerSearchFilterScopedState } from '@/ui/input/relation-picker/states/relationPickerSearchFilterScopedState';
|
import { relationPickerSearchFilterScopedState } from '@/ui/input/relation-picker/states/relationPickerSearchFilterScopedState';
|
||||||
import { EntityForSelect } from '@/ui/input/relation-picker/types/EntityForSelect';
|
import { EntityForSelect } from '@/ui/input/relation-picker/types/EntityForSelect';
|
||||||
@ -77,34 +78,25 @@ export function CompanyPickerCell({
|
|||||||
});
|
});
|
||||||
setIsCreateMode(false);
|
setIsCreateMode(false);
|
||||||
}
|
}
|
||||||
const noCompany: CompanyPickerSelectedCompany = {
|
|
||||||
entityType: Entity.Company,
|
|
||||||
id: '',
|
|
||||||
name: 'No Company',
|
|
||||||
avatarType: 'rounded',
|
|
||||||
domainName: '',
|
|
||||||
avatarUrl: '',
|
|
||||||
};
|
|
||||||
return isCreateMode ? (
|
return isCreateMode ? (
|
||||||
<DoubleTextCellEdit
|
<DoubleTextCellEdit
|
||||||
firstValue={relationPickerSearchFilter}
|
firstValue={relationPickerSearchFilter}
|
||||||
secondValue={''}
|
secondValue=""
|
||||||
firstValuePlaceholder={'Name'}
|
firstValuePlaceholder="Name"
|
||||||
secondValuePlaceholder={'Url'}
|
secondValuePlaceholder="Url"
|
||||||
onSubmit={handleCreate}
|
onSubmit={handleCreate}
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<SingleEntitySelect
|
<SingleEntitySelect
|
||||||
width={width}
|
EmptyIcon={IconBuildingSkyscraper}
|
||||||
onCreate={createModeEnabled ? handleStartCreation : undefined}
|
emptyLabel="No Company"
|
||||||
|
entitiesToSelect={companies.entitiesToSelect}
|
||||||
|
loading={companies.loading}
|
||||||
onCancel={onCancel}
|
onCancel={onCancel}
|
||||||
|
onCreate={createModeEnabled ? handleStartCreation : undefined}
|
||||||
onEntitySelected={handleCompanySelected}
|
onEntitySelected={handleCompanySelected}
|
||||||
entities={{
|
selectedEntity={companies.selectedEntities[0]}
|
||||||
entitiesToSelect: companies.entitiesToSelect,
|
width={width}
|
||||||
selectedEntity: companies.selectedEntities[0],
|
|
||||||
loading: companies.loading,
|
|
||||||
}}
|
|
||||||
noUser={noCompany}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -114,13 +114,11 @@ export function CompanyProgressPicker({
|
|||||||
<StyledDropdownMenuSeparator />
|
<StyledDropdownMenuSeparator />
|
||||||
<RecoilScope>
|
<RecoilScope>
|
||||||
<SingleEntitySelectBase
|
<SingleEntitySelectBase
|
||||||
onEntitySelected={handleEntitySelected}
|
entitiesToSelect={companies.entitiesToSelect}
|
||||||
|
loading={companies.loading}
|
||||||
onCancel={onCancel}
|
onCancel={onCancel}
|
||||||
entities={{
|
onEntitySelected={handleEntitySelected}
|
||||||
loading: companies.loading,
|
selectedEntity={companies.selectedEntities[0]}
|
||||||
entitiesToSelect: companies.entitiesToSelect,
|
|
||||||
selectedEntity: companies.selectedEntities[0],
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
</RecoilScope>
|
</RecoilScope>
|
||||||
</>
|
</>
|
||||||
|
|||||||
@ -64,14 +64,12 @@ export function NewCompanyProgressButton() {
|
|||||||
<RecoilScope>
|
<RecoilScope>
|
||||||
{isCreatingCard ? (
|
{isCreatingCard ? (
|
||||||
<SingleEntitySelect
|
<SingleEntitySelect
|
||||||
onEntitySelected={(value) => handleEntitySelect(value)}
|
disableBackgroundBlur
|
||||||
|
entitiesToSelect={companies.entitiesToSelect}
|
||||||
|
loading={companies.loading}
|
||||||
onCancel={handleCancel}
|
onCancel={handleCancel}
|
||||||
entities={{
|
onEntitySelected={handleEntitySelect}
|
||||||
entitiesToSelect: companies.entitiesToSelect,
|
selectedEntity={companies.selectedEntities[0]}
|
||||||
selectedEntity: companies.selectedEntities[0],
|
|
||||||
loading: companies.loading,
|
|
||||||
}}
|
|
||||||
disableBackgroundBlur={true}
|
|
||||||
/>
|
/>
|
||||||
) : (
|
) : (
|
||||||
<NewButton onClick={handleNewClick} />
|
<NewButton onClick={handleNewClick} />
|
||||||
|
|||||||
@ -68,14 +68,12 @@ export function PeoplePicker({
|
|||||||
|
|
||||||
return (
|
return (
|
||||||
<SingleEntitySelect
|
<SingleEntitySelect
|
||||||
onEntitySelected={handleEntitySelected}
|
entitiesToSelect={people.entitiesToSelect}
|
||||||
|
loading={people.loading}
|
||||||
onCancel={onCancel}
|
onCancel={onCancel}
|
||||||
onCreate={onCreate}
|
onCreate={onCreate}
|
||||||
entities={{
|
onEntitySelected={handleEntitySelected}
|
||||||
loading: people.loading,
|
selectedEntity={people.selectedEntities[0]}
|
||||||
entitiesToSelect: people.entitiesToSelect,
|
|
||||||
selectedEntity: people.selectedEntities[0],
|
|
||||||
}}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@ -157,14 +157,12 @@ export function BoardColumnMenu({
|
|||||||
)}
|
)}
|
||||||
{currentMenu === 'add' && (
|
{currentMenu === 'add' && (
|
||||||
<SingleEntitySelect
|
<SingleEntitySelect
|
||||||
onEntitySelected={(value) => handleCompanySelected(value)}
|
disableBackgroundBlur
|
||||||
|
entitiesToSelect={companies.entitiesToSelect}
|
||||||
|
loading={companies.loading}
|
||||||
onCancel={closeMenu}
|
onCancel={closeMenu}
|
||||||
entities={{
|
onEntitySelected={handleCompanySelected}
|
||||||
entitiesToSelect: companies.entitiesToSelect,
|
selectedEntity={companies.selectedEntities[0]}
|
||||||
selectedEntity: companies.selectedEntities[0],
|
|
||||||
loading: companies.loading,
|
|
||||||
}}
|
|
||||||
disableBackgroundBlur={true}
|
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
</StyledDropdownMenu>
|
</StyledDropdownMenu>
|
||||||
|
|||||||
@ -2,7 +2,7 @@ import { useContext } from 'react';
|
|||||||
|
|
||||||
import { EditableFieldMutationContext } from '../contexts/EditableFieldMutationContext';
|
import { EditableFieldMutationContext } from '../contexts/EditableFieldMutationContext';
|
||||||
import { FieldDefinition } from '../types/FieldDefinition';
|
import { FieldDefinition } from '../types/FieldDefinition';
|
||||||
import {
|
import type {
|
||||||
FieldBooleanMetadata,
|
FieldBooleanMetadata,
|
||||||
FieldBooleanValue,
|
FieldBooleanValue,
|
||||||
FieldChipMetadata,
|
FieldChipMetadata,
|
||||||
@ -82,163 +82,85 @@ export function useUpdateGenericEntityField() {
|
|||||||
>(
|
>(
|
||||||
currentEntityId: string,
|
currentEntityId: string,
|
||||||
field: FieldDefinition<FieldMetadata>,
|
field: FieldDefinition<FieldMetadata>,
|
||||||
newFieldValue: ValueType,
|
newFieldValue: ValueType | null,
|
||||||
) {
|
) {
|
||||||
const newFieldValueUnknown = newFieldValue as unknown;
|
|
||||||
// TODO: improve type guards organization, maybe with a common typeguard for all fields
|
// TODO: improve type guards organization, maybe with a common typeguard for all fields
|
||||||
// taking an object of options as parameter ?
|
// taking an object of options as parameter ?
|
||||||
//
|
//
|
||||||
// The goal would be to check that the field value not only is valid,
|
// The goal would be to check that the field value not only is valid,
|
||||||
// but also that it is validated against the corresponding field type
|
// but also that it is validated against the corresponding field type
|
||||||
|
|
||||||
// Relation
|
if (
|
||||||
if (isFieldRelation(field) && isFieldRelationValue(newFieldValueUnknown)) {
|
// Relation
|
||||||
const newSelectedEntity = newFieldValueUnknown;
|
isFieldRelation(field) &&
|
||||||
|
isFieldRelationValue(newFieldValue)
|
||||||
const fieldName = field.metadata.fieldName;
|
|
||||||
|
|
||||||
if (!newSelectedEntity) {
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: {
|
|
||||||
[fieldName]: {
|
|
||||||
disconnect: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: {
|
|
||||||
[fieldName]: {
|
|
||||||
connect: { id: newSelectedEntity.id },
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// Chip
|
|
||||||
} else if (isFieldChip(field) && isFieldChipValue(newFieldValueUnknown)) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [field.metadata.contentFieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Text
|
|
||||||
} else if (isFieldText(field) && isFieldTextValue(newFieldValueUnknown)) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [field.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Double text
|
|
||||||
} else if (
|
|
||||||
isFieldDoubleText(field) &&
|
|
||||||
isFieldDoubleTextValue(newFieldValueUnknown)
|
|
||||||
) {
|
) {
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
updateEntity({
|
||||||
variables: {
|
variables: {
|
||||||
where: { id: currentEntityId },
|
where: { id: currentEntityId },
|
||||||
data: {
|
data: {
|
||||||
[field.metadata.firstValueFieldName]: newContent.firstValue,
|
[field.metadata.fieldName]: newFieldValue
|
||||||
[field.metadata.secondValueFieldName]: newContent.secondValue,
|
? { connect: { id: newFieldValue.id } }
|
||||||
|
: { disconnect: true },
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
// Double Text Chip
|
return;
|
||||||
} else if (
|
|
||||||
isFieldDoubleTextChip(field) &&
|
|
||||||
isFieldDoubleTextChipValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: {
|
|
||||||
[field.metadata.firstValueFieldName]: newContent.firstValue,
|
|
||||||
[field.metadata.secondValueFieldName]: newContent.secondValue,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Phone
|
|
||||||
} else if (isFieldPhone(field) && isFieldPhoneValue(newFieldValueUnknown)) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [field.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// URL
|
|
||||||
} else if (isFieldURL(field) && isFieldURLValue(newFieldValueUnknown)) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [field.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Number
|
|
||||||
} else if (
|
|
||||||
isFieldNumber(field) &&
|
|
||||||
isFieldNumberValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [field.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Date
|
|
||||||
} else if (isFieldDate(field) && isFieldDateValue(newFieldValueUnknown)) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [field.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} else if (
|
|
||||||
isFieldProbability(field) &&
|
|
||||||
isFieldProbabilityValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [field.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
// Boolean
|
|
||||||
else if (
|
|
||||||
isFieldBoolean(field) &&
|
|
||||||
isFieldBooleanValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
|
if (
|
||||||
|
// Chip
|
||||||
|
isFieldChip(field) &&
|
||||||
|
isFieldChipValue(newFieldValue)
|
||||||
|
) {
|
||||||
updateEntity({
|
updateEntity({
|
||||||
variables: {
|
variables: {
|
||||||
where: { id: currentEntityId },
|
where: { id: currentEntityId },
|
||||||
data: { [field.metadata.fieldName]: newContent },
|
data: { [field.metadata.contentFieldName]: newFieldValue },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
// Text
|
||||||
|
(isFieldText(field) && isFieldTextValue(newFieldValue)) ||
|
||||||
|
// Phone
|
||||||
|
(isFieldPhone(field) && isFieldPhoneValue(newFieldValue)) ||
|
||||||
|
// URL
|
||||||
|
(isFieldURL(field) && isFieldURLValue(newFieldValue)) ||
|
||||||
|
// Number
|
||||||
|
(isFieldNumber(field) && isFieldNumberValue(newFieldValue)) ||
|
||||||
|
// Date
|
||||||
|
(isFieldDate(field) && isFieldDateValue(newFieldValue)) ||
|
||||||
|
// Probability
|
||||||
|
(isFieldProbability(field) && isFieldProbabilityValue(newFieldValue)) ||
|
||||||
|
// Boolean
|
||||||
|
(isFieldBoolean(field) && isFieldBooleanValue(newFieldValue))
|
||||||
|
) {
|
||||||
|
updateEntity({
|
||||||
|
variables: {
|
||||||
|
where: { id: currentEntityId },
|
||||||
|
data: { [field.metadata.fieldName]: newFieldValue },
|
||||||
|
},
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
// Double text
|
||||||
|
(isFieldDoubleText(field) && isFieldDoubleTextValue(newFieldValue)) ||
|
||||||
|
// Double Text Chip
|
||||||
|
(isFieldDoubleTextChip(field) &&
|
||||||
|
isFieldDoubleTextChipValue(newFieldValue))
|
||||||
|
) {
|
||||||
|
updateEntity({
|
||||||
|
variables: {
|
||||||
|
where: { id: currentEntityId },
|
||||||
|
data: {
|
||||||
|
[field.metadata.firstValueFieldName]: newFieldValue.firstValue,
|
||||||
|
[field.metadata.secondValueFieldName]: newFieldValue.secondValue,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@ -4,9 +4,5 @@ import { FieldRelationValue } from '../FieldMetadata';
|
|||||||
export function isFieldRelationValue(
|
export function isFieldRelationValue(
|
||||||
fieldValue: unknown,
|
fieldValue: unknown,
|
||||||
): fieldValue is FieldRelationValue {
|
): fieldValue is FieldRelationValue {
|
||||||
return (
|
return fieldValue !== undefined && typeof fieldValue === 'object';
|
||||||
fieldValue !== null &&
|
|
||||||
fieldValue !== undefined &&
|
|
||||||
typeof fieldValue === 'object'
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -4,9 +4,5 @@ import { ViewFieldRelationValue } from '../ViewField';
|
|||||||
export function isViewFieldRelationValue(
|
export function isViewFieldRelationValue(
|
||||||
fieldValue: unknown,
|
fieldValue: unknown,
|
||||||
): fieldValue is ViewFieldRelationValue {
|
): fieldValue is ViewFieldRelationValue {
|
||||||
return (
|
return fieldValue !== undefined && typeof fieldValue === 'object';
|
||||||
fieldValue !== null &&
|
|
||||||
fieldValue !== undefined &&
|
|
||||||
typeof fieldValue === 'object'
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|||||||
@ -13,29 +13,36 @@ import { useEntitySelectSearch } from '../hooks/useEntitySelectSearch';
|
|||||||
import { EntityForSelect } from '../types/EntityForSelect';
|
import { EntityForSelect } from '../types/EntityForSelect';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
EntitiesForSingleEntitySelect,
|
|
||||||
SingleEntitySelectBase,
|
SingleEntitySelectBase,
|
||||||
|
type SingleEntitySelectBaseProps,
|
||||||
} from './SingleEntitySelectBase';
|
} from './SingleEntitySelectBase';
|
||||||
|
|
||||||
|
export type SingleEntitySelectProps<
|
||||||
|
CustomEntityForSelect extends EntityForSelect,
|
||||||
|
> = {
|
||||||
|
disableBackgroundBlur?: boolean;
|
||||||
|
onCreate?: () => void;
|
||||||
|
width?: number;
|
||||||
|
} & Pick<
|
||||||
|
SingleEntitySelectBaseProps<CustomEntityForSelect>,
|
||||||
|
| 'EmptyIcon'
|
||||||
|
| 'emptyLabel'
|
||||||
|
| 'entitiesToSelect'
|
||||||
|
| 'loading'
|
||||||
|
| 'onCancel'
|
||||||
|
| 'onEntitySelected'
|
||||||
|
| 'selectedEntity'
|
||||||
|
>;
|
||||||
|
|
||||||
export function SingleEntitySelect<
|
export function SingleEntitySelect<
|
||||||
CustomEntityForSelect extends EntityForSelect,
|
CustomEntityForSelect extends EntityForSelect,
|
||||||
>({
|
>({
|
||||||
entities,
|
|
||||||
onEntitySelected,
|
|
||||||
onCreate,
|
|
||||||
onCancel,
|
|
||||||
width,
|
|
||||||
disableBackgroundBlur = false,
|
disableBackgroundBlur = false,
|
||||||
noUser,
|
onCancel,
|
||||||
}: {
|
onCreate,
|
||||||
onCancel?: () => void;
|
width,
|
||||||
onCreate?: () => void;
|
...props
|
||||||
entities: EntitiesForSingleEntitySelect<CustomEntityForSelect>;
|
}: SingleEntitySelectProps<CustomEntityForSelect>) {
|
||||||
onEntitySelected: (entity: CustomEntityForSelect | null | undefined) => void;
|
|
||||||
disableBackgroundBlur?: boolean;
|
|
||||||
width?: number;
|
|
||||||
noUser?: CustomEntityForSelect;
|
|
||||||
}) {
|
|
||||||
const containerRef = useRef<HTMLDivElement>(null);
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
|
|
||||||
const { searchFilter, handleSearchFilterChange } = useEntitySelectSearch();
|
const { searchFilter, handleSearchFilterChange } = useEntitySelectSearch();
|
||||||
@ -64,12 +71,7 @@ export function SingleEntitySelect<
|
|||||||
autoFocus
|
autoFocus
|
||||||
/>
|
/>
|
||||||
<StyledDropdownMenuSeparator />
|
<StyledDropdownMenuSeparator />
|
||||||
<SingleEntitySelectBase
|
<SingleEntitySelectBase {...props} onCancel={onCancel} />
|
||||||
entities={entities}
|
|
||||||
onEntitySelected={onEntitySelected}
|
|
||||||
onCancel={onCancel}
|
|
||||||
noUser={noUser}
|
|
||||||
/>
|
|
||||||
{showCreateButton && (
|
{showCreateButton && (
|
||||||
<>
|
<>
|
||||||
<StyledDropdownMenuItemsContainer hasMaxHeight>
|
<StyledDropdownMenuItemsContainer hasMaxHeight>
|
||||||
|
|||||||
@ -2,49 +2,47 @@ import { useRef } from 'react';
|
|||||||
import { Key } from 'ts-key-enum';
|
import { Key } from 'ts-key-enum';
|
||||||
|
|
||||||
import { StyledDropdownMenuItemsContainer } from '@/ui/dropdown/components/StyledDropdownMenuItemsContainer';
|
import { StyledDropdownMenuItemsContainer } from '@/ui/dropdown/components/StyledDropdownMenuItemsContainer';
|
||||||
import { IconBuildingSkyscraper, IconUserCircle } from '@/ui/icon';
|
import type { IconComponent } from '@/ui/icon/types/IconComponent';
|
||||||
import { MenuItem } from '@/ui/menu-item/components/MenuItem';
|
import { MenuItem } from '@/ui/menu-item/components/MenuItem';
|
||||||
import { MenuItemSelectAvatar } from '@/ui/menu-item/components/MenuItemSelectAvatar';
|
import { MenuItemSelectAvatar } from '@/ui/menu-item/components/MenuItemSelectAvatar';
|
||||||
import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys';
|
import { useScopedHotkeys } from '@/ui/utilities/hotkey/hooks/useScopedHotkeys';
|
||||||
import { Avatar } from '@/users/components/Avatar';
|
import { Avatar } from '@/users/components/Avatar';
|
||||||
import { isDefined } from '~/utils/isDefined';
|
import { assertNotNull } from '~/utils/assert';
|
||||||
import { isNonEmptyString } from '~/utils/isNonEmptyString';
|
import { isNonEmptyString } from '~/utils/isNonEmptyString';
|
||||||
|
|
||||||
import { useEntitySelectScroll } from '../hooks/useEntitySelectScroll';
|
import { useEntitySelectScroll } from '../hooks/useEntitySelectScroll';
|
||||||
import { EntityForSelect } from '../types/EntityForSelect';
|
import { EntityForSelect } from '../types/EntityForSelect';
|
||||||
import { Entity } from '../types/EntityTypeForSelect';
|
|
||||||
import { RelationPickerHotkeyScope } from '../types/RelationPickerHotkeyScope';
|
import { RelationPickerHotkeyScope } from '../types/RelationPickerHotkeyScope';
|
||||||
|
|
||||||
import { DropdownMenuSkeletonItem } from './skeletons/DropdownMenuSkeletonItem';
|
import { DropdownMenuSkeletonItem } from './skeletons/DropdownMenuSkeletonItem';
|
||||||
|
|
||||||
export type EntitiesForSingleEntitySelect<
|
export type SingleEntitySelectBaseProps<
|
||||||
CustomEntityForSelect extends EntityForSelect,
|
CustomEntityForSelect extends EntityForSelect,
|
||||||
> = {
|
> = {
|
||||||
selectedEntity: CustomEntityForSelect;
|
EmptyIcon?: IconComponent;
|
||||||
|
emptyLabel?: string;
|
||||||
entitiesToSelect: CustomEntityForSelect[];
|
entitiesToSelect: CustomEntityForSelect[];
|
||||||
loading: boolean;
|
loading?: boolean;
|
||||||
|
onCancel?: () => void;
|
||||||
|
onEntitySelected: (entity?: CustomEntityForSelect) => void;
|
||||||
|
selectedEntity?: CustomEntityForSelect;
|
||||||
};
|
};
|
||||||
|
|
||||||
export function SingleEntitySelectBase<
|
export function SingleEntitySelectBase<
|
||||||
CustomEntityForSelect extends EntityForSelect,
|
CustomEntityForSelect extends EntityForSelect,
|
||||||
>({
|
>({
|
||||||
entities,
|
EmptyIcon,
|
||||||
onEntitySelected,
|
emptyLabel,
|
||||||
|
entitiesToSelect,
|
||||||
|
loading,
|
||||||
onCancel,
|
onCancel,
|
||||||
noUser,
|
onEntitySelected,
|
||||||
}: {
|
selectedEntity,
|
||||||
entities: EntitiesForSingleEntitySelect<CustomEntityForSelect>;
|
}: SingleEntitySelectBaseProps<CustomEntityForSelect>) {
|
||||||
onEntitySelected: (entity: CustomEntityForSelect | null | undefined) => void;
|
|
||||||
onCancel?: () => void;
|
|
||||||
noUser?: CustomEntityForSelect;
|
|
||||||
}) {
|
|
||||||
const containerRef = useRef<HTMLDivElement>(null);
|
const containerRef = useRef<HTMLDivElement>(null);
|
||||||
let entitiesInDropdown = isDefined(entities.selectedEntity)
|
const entitiesInDropdown = [selectedEntity, ...entitiesToSelect].filter(
|
||||||
? [entities.selectedEntity, ...(entities.entitiesToSelect ?? [])]
|
(entity): entity is CustomEntityForSelect =>
|
||||||
: entities.entitiesToSelect ?? [];
|
assertNotNull(entity) && isNonEmptyString(entity.name.trim()),
|
||||||
|
|
||||||
entitiesInDropdown = entitiesInDropdown.filter((entity) =>
|
|
||||||
isNonEmptyString(entity.name),
|
|
||||||
);
|
);
|
||||||
|
|
||||||
const { hoveredIndex, resetScroll } = useEntitySelectScroll({
|
const { hoveredIndex, resetScroll } = useEntitySelectScroll({
|
||||||
@ -71,25 +69,16 @@ export function SingleEntitySelectBase<
|
|||||||
[onCancel],
|
[onCancel],
|
||||||
);
|
);
|
||||||
|
|
||||||
entitiesInDropdown = entitiesInDropdown.filter((entity) =>
|
|
||||||
isNonEmptyString(entity.name.trim()),
|
|
||||||
);
|
|
||||||
|
|
||||||
const NoUserIcon =
|
|
||||||
noUser?.entityType === Entity.User
|
|
||||||
? IconUserCircle
|
|
||||||
: IconBuildingSkyscraper;
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<StyledDropdownMenuItemsContainer ref={containerRef} hasMaxHeight>
|
<StyledDropdownMenuItemsContainer ref={containerRef} hasMaxHeight>
|
||||||
{noUser && (
|
{emptyLabel && (
|
||||||
<MenuItem
|
<MenuItem
|
||||||
onClick={() => onEntitySelected(noUser)}
|
onClick={() => onEntitySelected()}
|
||||||
LeftIcon={NoUserIcon}
|
LeftIcon={EmptyIcon}
|
||||||
text={noUser.name}
|
text={emptyLabel}
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
{entities.loading ? (
|
{loading ? (
|
||||||
<DropdownMenuSkeletonItem />
|
<DropdownMenuSkeletonItem />
|
||||||
) : entitiesInDropdown.length === 0 ? (
|
) : entitiesInDropdown.length === 0 ? (
|
||||||
<MenuItem text="No result" />
|
<MenuItem text="No result" />
|
||||||
@ -98,7 +87,7 @@ export function SingleEntitySelectBase<
|
|||||||
<MenuItemSelectAvatar
|
<MenuItemSelectAvatar
|
||||||
key={entity.id}
|
key={entity.id}
|
||||||
testId="menu-item"
|
testId="menu-item"
|
||||||
selected={entities.selectedEntity?.id === entity.id}
|
selected={selectedEntity?.id === entity.id}
|
||||||
onClick={() => onEntitySelected(entity)}
|
onClick={() => onEntitySelected(entity)}
|
||||||
text={entity.name}
|
text={entity.name}
|
||||||
hovered={hoveredIndex === entitiesInDropdown.indexOf(entity)}
|
hovered={hoveredIndex === entitiesInDropdown.indexOf(entity)}
|
||||||
|
|||||||
@ -0,0 +1,81 @@
|
|||||||
|
import { expect } from '@storybook/jest';
|
||||||
|
import type { Meta, StoryObj } from '@storybook/react';
|
||||||
|
import { userEvent, within } from '@storybook/testing-library';
|
||||||
|
|
||||||
|
import { IconUserCircle } from '@/ui/icon';
|
||||||
|
import { useRecoilScopedValue } from '@/ui/utilities/recoil-scope/hooks/useRecoilScopedValue';
|
||||||
|
import { ComponentDecorator } from '~/testing/decorators/ComponentDecorator';
|
||||||
|
import { ComponentWithRecoilScopeDecorator } from '~/testing/decorators/ComponentWithRecoilScopeDecorator';
|
||||||
|
import { mockedPeopleData } from '~/testing/mock-data/people';
|
||||||
|
import { sleep } from '~/testing/sleep';
|
||||||
|
|
||||||
|
import { relationPickerSearchFilterScopedState } from '../../states/relationPickerSearchFilterScopedState';
|
||||||
|
import type { EntityForSelect } from '../../types/EntityForSelect';
|
||||||
|
import { Entity } from '../../types/EntityTypeForSelect';
|
||||||
|
import { SingleEntitySelect } from '../SingleEntitySelect';
|
||||||
|
|
||||||
|
const entities = mockedPeopleData.map<EntityForSelect>((person) => ({
|
||||||
|
id: person.id,
|
||||||
|
entityType: Entity.Person,
|
||||||
|
name: person.displayName,
|
||||||
|
}));
|
||||||
|
|
||||||
|
const meta: Meta<typeof SingleEntitySelect> = {
|
||||||
|
title: 'UI/Input/RelationPicker/SingleEntitySelect',
|
||||||
|
component: SingleEntitySelect,
|
||||||
|
decorators: [ComponentDecorator, ComponentWithRecoilScopeDecorator],
|
||||||
|
argTypes: {
|
||||||
|
selectedEntity: {
|
||||||
|
options: entities.map(({ name }) => name),
|
||||||
|
mapping: entities.reduce(
|
||||||
|
(result, entity) => ({ ...result, [entity.name]: entity }),
|
||||||
|
{},
|
||||||
|
),
|
||||||
|
},
|
||||||
|
},
|
||||||
|
render: function Render(args) {
|
||||||
|
const searchFilter = useRecoilScopedValue(
|
||||||
|
relationPickerSearchFilterScopedState,
|
||||||
|
);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<SingleEntitySelect
|
||||||
|
{...args}
|
||||||
|
entitiesToSelect={entities.filter(
|
||||||
|
(entity) =>
|
||||||
|
entity.id !== args.selectedEntity?.id &&
|
||||||
|
entity.name.includes(searchFilter),
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
);
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export default meta;
|
||||||
|
type Story = StoryObj<typeof SingleEntitySelect>;
|
||||||
|
|
||||||
|
export const Default: Story = {};
|
||||||
|
|
||||||
|
export const WithSelectedEntity: Story = {
|
||||||
|
args: { selectedEntity: entities[2] },
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WithEmptyOption: Story = {
|
||||||
|
args: {
|
||||||
|
EmptyIcon: IconUserCircle,
|
||||||
|
emptyLabel: 'Nobody',
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const WithSearchFilter: Story = {
|
||||||
|
play: async ({ canvasElement, step }) => {
|
||||||
|
const canvas = within(canvasElement);
|
||||||
|
const searchInput = canvas.getByRole('textbox');
|
||||||
|
|
||||||
|
await step('Enter search text', async () => {
|
||||||
|
await sleep(50);
|
||||||
|
await userEvent.type(searchInput, 'a');
|
||||||
|
await expect(searchInput).toHaveValue('a');
|
||||||
|
});
|
||||||
|
},
|
||||||
|
};
|
||||||
@ -35,9 +35,6 @@ export function GenericEditableRelationCellEditMode({
|
|||||||
const updateEntityField = useUpdateEntityField();
|
const updateEntityField = useUpdateEntityField();
|
||||||
|
|
||||||
function updateCachedPersonField(newFieldEntity: EntityForSelect | null) {
|
function updateCachedPersonField(newFieldEntity: EntityForSelect | null) {
|
||||||
if (newFieldEntity === null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setFieldValueEntity({
|
setFieldValueEntity({
|
||||||
avatarUrl: newFieldEntity?.avatarUrl ?? '',
|
avatarUrl: newFieldEntity?.avatarUrl ?? '',
|
||||||
entityType: Entity.Company,
|
entityType: Entity.Company,
|
||||||
@ -49,10 +46,6 @@ export function GenericEditableRelationCellEditMode({
|
|||||||
function updateCachedCompanyField(
|
function updateCachedCompanyField(
|
||||||
newFieldEntity: CompanyPickerSelectedCompany | null,
|
newFieldEntity: CompanyPickerSelectedCompany | null,
|
||||||
) {
|
) {
|
||||||
if (newFieldEntity === null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
setFieldValueEntity({
|
setFieldValueEntity({
|
||||||
id: newFieldEntity?.id ?? '',
|
id: newFieldEntity?.id ?? '',
|
||||||
name: newFieldEntity?.name ?? '',
|
name: newFieldEntity?.name ?? '',
|
||||||
@ -64,7 +57,6 @@ export function GenericEditableRelationCellEditMode({
|
|||||||
newFieldEntity: CompanyPickerSelectedCompany | null,
|
newFieldEntity: CompanyPickerSelectedCompany | null,
|
||||||
) {
|
) {
|
||||||
if (
|
if (
|
||||||
newFieldEntity &&
|
|
||||||
newFieldEntity?.id !== fieldValueEntity?.id &&
|
newFieldEntity?.id !== fieldValueEntity?.id &&
|
||||||
currentRowEntityId &&
|
currentRowEntityId &&
|
||||||
updateEntityField
|
updateEntityField
|
||||||
|
|||||||
@ -76,52 +76,39 @@ export function useUpdateEntityField() {
|
|||||||
>(
|
>(
|
||||||
currentEntityId: string,
|
currentEntityId: string,
|
||||||
columnDefinition: ColumnDefinition<MetadataType>,
|
columnDefinition: ColumnDefinition<MetadataType>,
|
||||||
newFieldValue: ValueType,
|
newFieldValue: ValueType | null,
|
||||||
) {
|
) {
|
||||||
const newFieldValueUnknown = newFieldValue as unknown;
|
|
||||||
// TODO: improve type guards organization, maybe with a common typeguard for all view fields
|
// TODO: improve type guards organization, maybe with a common typeguard for all view fields
|
||||||
// taking an object of options as parameter ?
|
// taking an object of options as parameter ?
|
||||||
//
|
//
|
||||||
// The goal would be to check that the view field value not only is valid,
|
// The goal would be to check that the view field value not only is valid,
|
||||||
// but also that it is validated against the corresponding view field type
|
// but also that it is validated against the corresponding view field type
|
||||||
|
|
||||||
// Relation
|
|
||||||
if (
|
if (
|
||||||
|
// Relation
|
||||||
isViewFieldRelation(columnDefinition) &&
|
isViewFieldRelation(columnDefinition) &&
|
||||||
isViewFieldRelationValue(newFieldValueUnknown)
|
isViewFieldRelationValue(newFieldValue)
|
||||||
) {
|
) {
|
||||||
const newSelectedEntity = newFieldValueUnknown;
|
updateEntity({
|
||||||
|
variables: {
|
||||||
|
where: { id: currentEntityId },
|
||||||
|
data: {
|
||||||
|
[columnDefinition.metadata.fieldName]:
|
||||||
|
!newFieldValue || newFieldValue.id === ''
|
||||||
|
? { disconnect: true }
|
||||||
|
: { connect: { id: newFieldValue.id } },
|
||||||
|
},
|
||||||
|
},
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
const fieldName = columnDefinition.metadata.fieldName;
|
if (
|
||||||
if (!newSelectedEntity || newSelectedEntity.id === '') {
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: {
|
|
||||||
[fieldName]: {
|
|
||||||
disconnect: true,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: {
|
|
||||||
[fieldName]: {
|
|
||||||
connect: { id: newSelectedEntity.id },
|
|
||||||
},
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
}
|
|
||||||
// Chip
|
// Chip
|
||||||
} else if (
|
|
||||||
isViewFieldChip(columnDefinition) &&
|
isViewFieldChip(columnDefinition) &&
|
||||||
isViewFieldChipValue(newFieldValueUnknown)
|
isViewFieldChipValue(newFieldValue)
|
||||||
) {
|
) {
|
||||||
const newContent = newFieldValueUnknown;
|
const newContent = newFieldValue;
|
||||||
|
|
||||||
updateEntity({
|
updateEntity({
|
||||||
variables: {
|
variables: {
|
||||||
@ -129,144 +116,60 @@ export function useUpdateEntityField() {
|
|||||||
data: { [columnDefinition.metadata.contentFieldName]: newContent },
|
data: { [columnDefinition.metadata.contentFieldName]: newContent },
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
// Text
|
// Text
|
||||||
} else if (
|
(isViewFieldText(columnDefinition) &&
|
||||||
isViewFieldText(columnDefinition) &&
|
isViewFieldTextValue(newFieldValue)) ||
|
||||||
isViewFieldTextValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Double text
|
|
||||||
} else if (
|
|
||||||
isViewFieldDoubleText(columnDefinition) &&
|
|
||||||
isViewFieldDoubleTextValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: {
|
|
||||||
[columnDefinition.metadata.firstValueFieldName]:
|
|
||||||
newContent.firstValue,
|
|
||||||
[columnDefinition.metadata.secondValueFieldName]:
|
|
||||||
newContent.secondValue,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Double Text Chip
|
|
||||||
} else if (
|
|
||||||
isViewFieldDoubleTextChip(columnDefinition) &&
|
|
||||||
isViewFieldDoubleTextChipValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: {
|
|
||||||
[columnDefinition.metadata.firstValueFieldName]:
|
|
||||||
newContent.firstValue,
|
|
||||||
[columnDefinition.metadata.secondValueFieldName]:
|
|
||||||
newContent.secondValue,
|
|
||||||
},
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Phone
|
// Phone
|
||||||
} else if (
|
(isViewFieldPhone(columnDefinition) &&
|
||||||
isViewFieldPhone(columnDefinition) &&
|
isViewFieldPhoneValue(newFieldValue)) ||
|
||||||
isViewFieldPhoneValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Email
|
// Email
|
||||||
} else if (
|
(isViewFieldEmail(columnDefinition) &&
|
||||||
isViewFieldEmail(columnDefinition) &&
|
isViewFieldEmailValue(newFieldValue)) ||
|
||||||
isViewFieldEmailValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// URL
|
// URL
|
||||||
} else if (
|
(isViewFieldURL(columnDefinition) &&
|
||||||
isViewFieldURL(columnDefinition) &&
|
isViewFieldURLValue(newFieldValue)) ||
|
||||||
isViewFieldURLValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Number
|
// Number
|
||||||
} else if (
|
(isViewFieldNumber(columnDefinition) &&
|
||||||
isViewFieldNumber(columnDefinition) &&
|
isViewFieldNumberValue(newFieldValue)) ||
|
||||||
isViewFieldNumberValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Boolean
|
// Boolean
|
||||||
} else if (
|
(isViewFieldBoolean(columnDefinition) &&
|
||||||
isViewFieldBoolean(columnDefinition) &&
|
isViewFieldBooleanValue(newFieldValue)) ||
|
||||||
isViewFieldBooleanValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
|
||||||
variables: {
|
|
||||||
where: { id: currentEntityId },
|
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
|
||||||
},
|
|
||||||
});
|
|
||||||
// Money
|
// Money
|
||||||
} else if (
|
(isViewFieldMoney(columnDefinition) &&
|
||||||
isViewFieldMoney(columnDefinition) &&
|
isViewFieldMoneyValue(newFieldValue)) ||
|
||||||
isViewFieldMoneyValue(newFieldValueUnknown)
|
// Date
|
||||||
|
(isViewFieldDate(columnDefinition) && isViewFieldDateValue(newFieldValue))
|
||||||
) {
|
) {
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
updateEntity({
|
updateEntity({
|
||||||
variables: {
|
variables: {
|
||||||
where: { id: currentEntityId },
|
where: { id: currentEntityId },
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
data: { [columnDefinition.metadata.fieldName]: newFieldValue },
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
// Date
|
return;
|
||||||
} else if (
|
}
|
||||||
isViewFieldDate(columnDefinition) &&
|
|
||||||
isViewFieldDateValue(newFieldValueUnknown)
|
|
||||||
) {
|
|
||||||
const newContent = newFieldValueUnknown;
|
|
||||||
|
|
||||||
|
if (
|
||||||
|
// Double text
|
||||||
|
(isViewFieldDoubleText(columnDefinition) &&
|
||||||
|
isViewFieldDoubleTextValue(newFieldValue)) ||
|
||||||
|
// Double Text Chip
|
||||||
|
(isViewFieldDoubleTextChip(columnDefinition) &&
|
||||||
|
isViewFieldDoubleTextChipValue(newFieldValue))
|
||||||
|
) {
|
||||||
updateEntity({
|
updateEntity({
|
||||||
variables: {
|
variables: {
|
||||||
where: { id: currentEntityId },
|
where: { id: currentEntityId },
|
||||||
data: { [columnDefinition.metadata.fieldName]: newContent },
|
data: {
|
||||||
|
[columnDefinition.metadata.firstValueFieldName]:
|
||||||
|
newFieldValue.firstValue,
|
||||||
|
[columnDefinition.metadata.secondValueFieldName]:
|
||||||
|
newFieldValue.secondValue,
|
||||||
|
},
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|||||||
@ -76,11 +76,9 @@ export function FilterDropdownEntitySearchSelect({
|
|||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<SingleEntitySelectBase
|
<SingleEntitySelectBase
|
||||||
entities={{
|
entitiesToSelect={entitiesForSelect.entitiesToSelect}
|
||||||
entitiesToSelect: entitiesForSelect.entitiesToSelect,
|
selectedEntity={entitiesForSelect.selectedEntities[0]}
|
||||||
selectedEntity: entitiesForSelect.selectedEntities[0],
|
loading={entitiesForSelect.loading}
|
||||||
loading: entitiesForSelect.loading,
|
|
||||||
}}
|
|
||||||
onEntitySelected={handleUserSelected}
|
onEntitySelected={handleUserSelected}
|
||||||
/>
|
/>
|
||||||
</>
|
</>
|
||||||
|
|||||||
@ -1,4 +1,5 @@
|
|||||||
import { useFilteredSearchEntityQuery } from '@/search/hooks/useFilteredSearchEntityQuery';
|
import { useFilteredSearchEntityQuery } from '@/search/hooks/useFilteredSearchEntityQuery';
|
||||||
|
import { IconUserCircle } from '@/ui/icon';
|
||||||
import { SingleEntitySelect } from '@/ui/input/relation-picker/components/SingleEntitySelect';
|
import { SingleEntitySelect } from '@/ui/input/relation-picker/components/SingleEntitySelect';
|
||||||
import { relationPickerSearchFilterScopedState } from '@/ui/input/relation-picker/states/relationPickerSearchFilterScopedState';
|
import { relationPickerSearchFilterScopedState } from '@/ui/input/relation-picker/states/relationPickerSearchFilterScopedState';
|
||||||
import { EntityForSelect } from '@/ui/input/relation-picker/types/EntityForSelect';
|
import { EntityForSelect } from '@/ui/input/relation-picker/types/EntityForSelect';
|
||||||
@ -51,24 +52,17 @@ export function UserPicker({
|
|||||||
) {
|
) {
|
||||||
onSubmit(selectedUser ?? null);
|
onSubmit(selectedUser ?? null);
|
||||||
}
|
}
|
||||||
const noUser: UserForSelect = {
|
|
||||||
entityType: Entity.User,
|
|
||||||
id: '',
|
|
||||||
name: 'No Owner',
|
|
||||||
avatarType: 'rounded',
|
|
||||||
avatarUrl: '',
|
|
||||||
};
|
|
||||||
return (
|
return (
|
||||||
<SingleEntitySelect
|
<SingleEntitySelect
|
||||||
width={width}
|
EmptyIcon={IconUserCircle}
|
||||||
onEntitySelected={handleEntitySelected}
|
emptyLabel="No Owner"
|
||||||
|
entitiesToSelect={users.entitiesToSelect}
|
||||||
|
loading={users.loading}
|
||||||
onCancel={onCancel}
|
onCancel={onCancel}
|
||||||
entities={{
|
onEntitySelected={handleEntitySelected}
|
||||||
loading: users.loading,
|
selectedEntity={users.selectedEntities[0]}
|
||||||
entitiesToSelect: users.entitiesToSelect,
|
width={width}
|
||||||
selectedEntity: users.selectedEntities[0],
|
|
||||||
}}
|
|
||||||
noUser={noUser}
|
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
Reference in New Issue
Block a user