* add fieldDefinition * update naming * use a unique contextProvider for editable fields * remove EntityUpdateMutationHookContext.Provider usage in CompanyBoardCard * add fieldDefinitionState * remove unnecessary refetchQueries to avoid re-render * add FieldMetadata * add type guards and update useUpdateGenericEntityField * restore refetchQueries
109 lines
3.3 KiB
TypeScript
109 lines
3.3 KiB
TypeScript
import { useContext } from 'react';
|
|
import styled from '@emotion/styled';
|
|
import { useRecoilState } from 'recoil';
|
|
|
|
import { PeoplePicker } from '@/people/components/PeoplePicker';
|
|
import {
|
|
ViewFieldRelationMetadata,
|
|
ViewFieldRelationValue,
|
|
} from '@/ui/editable-field/types/ViewField';
|
|
import { EntityForSelect } from '@/ui/input/relation-picker/types/EntityForSelect';
|
|
import { Entity } from '@/ui/input/relation-picker/types/EntityTypeForSelect';
|
|
|
|
import { useEditableField } from '../hooks/useEditableField';
|
|
import { useUpdateGenericEntityField } from '../hooks/useUpdateGenericEntityField';
|
|
import { EditableFieldContext } from '../states/EditableFieldContext';
|
|
import { genericEntityFieldFamilySelector } from '../states/genericEntityFieldFamilySelector';
|
|
import { FieldDefinition } from '../types/FieldDefinition';
|
|
import { FieldRelationMetadata } from '../types/FieldMetadata';
|
|
|
|
const RelationPickerContainer = styled.div`
|
|
left: 0px;
|
|
position: absolute;
|
|
top: -8px;
|
|
`;
|
|
|
|
function RelationPicker({
|
|
fieldDefinition,
|
|
fieldValue,
|
|
handleEntitySubmit,
|
|
handleCancel,
|
|
}: {
|
|
fieldDefinition: FieldDefinition<FieldRelationMetadata>;
|
|
fieldValue: ViewFieldRelationValue;
|
|
handleEntitySubmit: (newRelationId: EntityForSelect | null) => void;
|
|
handleCancel: () => void;
|
|
}) {
|
|
switch (fieldDefinition.metadata.relationType) {
|
|
case Entity.Person: {
|
|
return (
|
|
<PeoplePicker
|
|
personId={fieldValue?.id ?? null}
|
|
onSubmit={handleEntitySubmit}
|
|
onCancel={handleCancel}
|
|
/>
|
|
);
|
|
}
|
|
default:
|
|
console.warn(
|
|
`Unknown relation type: "${fieldDefinition.metadata.relationType}" in GenericEditableRelationField`,
|
|
);
|
|
return <> </>;
|
|
}
|
|
}
|
|
|
|
export function GenericEditableRelationFieldEditMode() {
|
|
const currentEditableField = useContext(EditableFieldContext);
|
|
const currentEditableFieldEntityId = currentEditableField.entityId;
|
|
const currentEditableFieldDefinition =
|
|
currentEditableField.fieldDefinition as FieldDefinition<ViewFieldRelationMetadata>;
|
|
|
|
// TODO: we could use a hook that would return the field value with the right type
|
|
const [fieldValue, setFieldValue] = useRecoilState<any | null>(
|
|
genericEntityFieldFamilySelector({
|
|
entityId: currentEditableFieldEntityId ?? '',
|
|
fieldName: currentEditableFieldDefinition
|
|
? currentEditableFieldDefinition.metadata.fieldName
|
|
: '',
|
|
}),
|
|
);
|
|
|
|
const updateField = useUpdateGenericEntityField();
|
|
const { closeEditableField } = useEditableField();
|
|
|
|
function handleSubmit(newRelation: EntityForSelect | null) {
|
|
if (newRelation?.id === fieldValue?.id) return;
|
|
|
|
setFieldValue({
|
|
id: newRelation?.id ?? null,
|
|
displayName: newRelation?.name ?? null,
|
|
avatarUrl: newRelation?.avatarUrl ?? null,
|
|
});
|
|
|
|
if (currentEditableFieldEntityId && updateField) {
|
|
updateField(
|
|
currentEditableFieldEntityId,
|
|
currentEditableFieldDefinition,
|
|
newRelation,
|
|
);
|
|
}
|
|
|
|
closeEditableField();
|
|
}
|
|
|
|
function handleCancel() {
|
|
closeEditableField();
|
|
}
|
|
|
|
return (
|
|
<RelationPickerContainer>
|
|
<RelationPicker
|
|
fieldDefinition={currentEditableFieldDefinition}
|
|
fieldValue={fieldValue}
|
|
handleEntitySubmit={handleSubmit}
|
|
handleCancel={handleCancel}
|
|
/>
|
|
</RelationPickerContainer>
|
|
);
|
|
}
|