Fix 5598 - View field creation (#5732)

- Fix duplicate view field creation
- Fix redirect to proper settings data model page
- Refetch view fields after field creation (temporary solution)

Fixes  https://github.com/twentyhq/twenty/issues/5598
This commit is contained in:
Lucas Bordeau
2024-06-04 15:10:56 +02:00
committed by GitHub
parent c5d5d18347
commit 5e32cb215e
5 changed files with 63 additions and 132 deletions

View File

@ -1,10 +1,11 @@
import { useCallback } from 'react'; import { useCallback, useContext } from 'react';
import { Link } from 'react-router-dom'; import { Link } from 'react-router-dom';
import styled from '@emotion/styled'; import styled from '@emotion/styled';
import { useRecoilValue } from 'recoil'; import { useRecoilValue } from 'recoil';
import { IconSettings, useIcons } from 'twenty-ui'; import { IconSettings, useIcons } from 'twenty-ui';
import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata'; import { FieldMetadata } from '@/object-record/record-field/types/FieldMetadata';
import { RecordTableContext } from '@/object-record/record-table/contexts/RecordTableContext';
import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates';
import { useTableColumns } from '@/object-record/record-table/hooks/useTableColumns'; import { useTableColumns } from '@/object-record/record-table/hooks/useTableColumns';
import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition'; import { ColumnDefinition } from '@/object-record/record-table/types/ColumnDefinition';
@ -14,6 +15,7 @@ import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown';
import { MenuItem } from '@/ui/navigation/menu-item/components/MenuItem'; import { MenuItem } from '@/ui/navigation/menu-item/components/MenuItem';
export const RecordTableHeaderPlusButtonContent = () => { export const RecordTableHeaderPlusButtonContent = () => {
const { objectMetadataItem } = useContext(RecordTableContext);
const { closeDropdown } = useDropdown(); const { closeDropdown } = useDropdown();
const { hiddenTableColumnsSelector } = useRecordTableStates(); const { hiddenTableColumnsSelector } = useRecordTableStates();
@ -54,7 +56,9 @@ export const RecordTableHeaderPlusButtonContent = () => {
</> </>
)} )}
<DropdownMenuItemsContainer> <DropdownMenuItemsContainer>
<StyledMenuItemLink to="/settings/objects"> <StyledMenuItemLink
to={`/settings/objects/${objectMetadataItem.namePlural}`}
>
<MenuItem LeftIcon={IconSettings} text="Customize fields" /> <MenuItem LeftIcon={IconSettings} text="Customize fields" />
</StyledMenuItemLink> </StyledMenuItemLink>
</DropdownMenuItemsContainer> </DropdownMenuItemsContainer>

View File

@ -43,21 +43,23 @@ export const useTableColumns = (props?: useRecordTableProps) => {
async ( async (
viewField: Omit<ColumnDefinition<FieldMetadata>, 'size' | 'position'>, viewField: Omit<ColumnDefinition<FieldMetadata>, 'size' | 'position'>,
) => { ) => {
const isNewColumn = !tableColumns.some( const shouldShowColumn = !visibleTableColumns.some(
(tableColumns) => (visibleColumn) =>
tableColumns.fieldMetadataId === viewField.fieldMetadataId, visibleColumn.fieldMetadataId === viewField.fieldMetadataId,
); );
const lastTableColumnPosition = [...tableColumns]
const tableColumnPositions = [...tableColumns]
.sort((a, b) => b.position - a.position) .sort((a, b) => b.position - a.position)
.map((column) => column.position); .map((column) => column.position);
const lastPosition = lastTableColumnPosition[0] ?? 0; const lastPosition = tableColumnPositions[0] ?? 0;
if (isNewColumn) { if (shouldShowColumn) {
const newColumn = availableTableColumns.find( const newColumn = availableTableColumns.find(
(availableTableColumn) => (availableTableColumn) =>
availableTableColumn.fieldMetadataId === viewField.fieldMetadataId, availableTableColumn.fieldMetadataId === viewField.fieldMetadataId,
); );
if (!newColumn) return; if (!newColumn) return;
const nextColumns = [ const nextColumns = [
@ -67,7 +69,7 @@ export const useTableColumns = (props?: useRecordTableProps) => {
await handleColumnsChange(nextColumns); await handleColumnsChange(nextColumns);
} else { } else {
const nextColumns = tableColumns.map((previousColumn) => const nextColumns = visibleTableColumns.map((previousColumn) =>
previousColumn.fieldMetadataId === viewField.fieldMetadataId previousColumn.fieldMetadataId === viewField.fieldMetadataId
? { ...previousColumn, isVisible: !viewField.isVisible } ? { ...previousColumn, isVisible: !viewField.isVisible }
: previousColumn, : previousColumn,
@ -76,7 +78,12 @@ export const useTableColumns = (props?: useRecordTableProps) => {
await handleColumnsChange(nextColumns); await handleColumnsChange(nextColumns);
} }
}, },
[tableColumns, availableTableColumns, handleColumnsChange], [
tableColumns,
availableTableColumns,
handleColumnsChange,
visibleTableColumns,
],
); );
const handleMoveTableColumn = useCallback( const handleMoveTableColumn = useCallback(

View File

@ -78,6 +78,7 @@ export const usePersistViewFieldRecords = () => {
const updateViewFieldRecords = useCallback( const updateViewFieldRecords = useCallback(
(viewFieldsToUpdate: ViewField[]) => { (viewFieldsToUpdate: ViewField[]) => {
if (!viewFieldsToUpdate.length) return; if (!viewFieldsToUpdate.length) return;
return Promise.all( return Promise.all(
viewFieldsToUpdate.map((viewField) => viewFieldsToUpdate.map((viewField) =>
apolloClient.mutate({ apolloClient.mutate({

View File

@ -19,7 +19,7 @@ export const useSaveCurrentViewFields = (viewBarComponentId?: string) => {
const saveViewFields = useRecoilCallback( const saveViewFields = useRecoilCallback(
({ set, snapshot }) => ({ set, snapshot }) =>
async (fields: ViewField[]) => { async (viewFieldsToSave: ViewField[]) => {
const currentViewId = snapshot const currentViewId = snapshot
.getLoadable(currentViewIdState) .getLoadable(currentViewIdState)
.getValue(); .getValue();
@ -29,21 +29,27 @@ export const useSaveCurrentViewFields = (viewBarComponentId?: string) => {
} }
set(isPersistingViewFieldsState, true); set(isPersistingViewFieldsState, true);
const view = await getViewFromCache(currentViewId); const view = await getViewFromCache(currentViewId);
if (isUndefinedOrNull(view)) { if (isUndefinedOrNull(view)) {
return; return;
} }
const viewFieldsToUpdate = fields const currentViewFields = view.viewFields;
.map((field) => {
const existingField = view.viewFields.find( const viewFieldsToUpdate = viewFieldsToSave
(viewField) => viewField.id === field.id, .map((viewFieldToSave) => {
const existingField = currentViewFields.find(
(currentViewField) =>
currentViewField.fieldMetadataId ===
viewFieldToSave.fieldMetadataId,
); );
if (isUndefinedOrNull(existingField)) { if (isUndefinedOrNull(existingField)) {
return undefined; return undefined;
} }
if ( if (
isDeeplyEqual( isDeeplyEqual(
{ {
@ -52,24 +58,33 @@ export const useSaveCurrentViewFields = (viewBarComponentId?: string) => {
isVisible: existingField.isVisible, isVisible: existingField.isVisible,
}, },
{ {
position: field.position, position: viewFieldToSave.position,
size: field.size, size: viewFieldToSave.size,
isVisible: field.isVisible, isVisible: viewFieldToSave.isVisible,
}, },
) )
) { ) {
return undefined; return undefined;
} }
return field;
return { ...viewFieldToSave, id: existingField.id };
}) })
.filter(isDefined); .filter(isDefined);
const viewFieldsToCreate = fields.filter((field) => !field.id); const viewFieldsToCreate = viewFieldsToSave.filter(
(viewFieldToSave) =>
!currentViewFields.some(
(currentViewField) =>
currentViewField.fieldMetadataId ===
viewFieldToSave.fieldMetadataId,
),
);
await Promise.all([ await Promise.all([
createViewFieldRecords(viewFieldsToCreate, view), createViewFieldRecords(viewFieldsToCreate, view),
updateViewFieldRecords(viewFieldsToUpdate), updateViewFieldRecords(viewFieldsToUpdate),
]); ]);
set(isPersistingViewFieldsState, false); set(isPersistingViewFieldsState, false);
}, },
[ [

View File

@ -1,7 +1,7 @@
import { useEffect, useState } from 'react'; import { useEffect, useState } from 'react';
import { FormProvider, useForm } from 'react-hook-form'; import { FormProvider, useForm } from 'react-hook-form';
import { useNavigate, useParams } from 'react-router-dom'; import { useNavigate, useParams } from 'react-router-dom';
import { Reference, useApolloClient } from '@apollo/client'; import { useApolloClient } from '@apollo/client';
import styled from '@emotion/styled'; import styled from '@emotion/styled';
import { zodResolver } from '@hookform/resolvers/zod'; import { zodResolver } from '@hookform/resolvers/zod';
import pick from 'lodash.pick'; import pick from 'lodash.pick';
@ -11,10 +11,7 @@ import { z } from 'zod';
import { useCreateOneRelationMetadataItem } from '@/object-metadata/hooks/useCreateOneRelationMetadataItem'; import { useCreateOneRelationMetadataItem } from '@/object-metadata/hooks/useCreateOneRelationMetadataItem';
import { useFieldMetadataItem } from '@/object-metadata/hooks/useFieldMetadataItem'; import { useFieldMetadataItem } from '@/object-metadata/hooks/useFieldMetadataItem';
import { useFilteredObjectMetadataItems } from '@/object-metadata/hooks/useFilteredObjectMetadataItems'; import { useFilteredObjectMetadataItems } from '@/object-metadata/hooks/useFilteredObjectMetadataItems';
import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem';
import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular'; import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular';
import { RecordGqlRefEdge } from '@/object-record/cache/types/RecordGqlRefEdge';
import { modifyRecordFromCache } from '@/object-record/cache/utils/modifyRecordFromCache';
import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords'; import { useFindManyRecords } from '@/object-record/hooks/useFindManyRecords';
import { SaveAndCancelButtons } from '@/settings/components/SaveAndCancelButtons/SaveAndCancelButtons'; import { SaveAndCancelButtons } from '@/settings/components/SaveAndCancelButtons/SaveAndCancelButtons';
import { SettingsHeaderContainer } from '@/settings/components/SettingsHeaderContainer'; import { SettingsHeaderContainer } from '@/settings/components/SettingsHeaderContainer';
@ -51,13 +48,12 @@ export const SettingsObjectNewFieldStep2 = () => {
const { objectSlug = '' } = useParams(); const { objectSlug = '' } = useParams();
const { enqueueSnackBar } = useSnackBar(); const { enqueueSnackBar } = useSnackBar();
const { findActiveObjectMetadataItemBySlug, findObjectMetadataItemById } = const { findActiveObjectMetadataItemBySlug } =
useFilteredObjectMetadataItems(); useFilteredObjectMetadataItems();
const activeObjectMetadataItem = const activeObjectMetadataItem =
findActiveObjectMetadataItemBySlug(objectSlug); findActiveObjectMetadataItemBySlug(objectSlug);
const { createMetadataField } = useFieldMetadataItem(); const { createMetadataField } = useFieldMetadataItem();
const cache = useApolloClient().cache;
const formConfig = useForm<SettingsDataModelNewFieldFormValues>({ const formConfig = useForm<SettingsDataModelNewFieldFormValues>({
mode: 'onTouched', mode: 'onTouched',
@ -70,12 +66,8 @@ export const SettingsObjectNewFieldStep2 = () => {
} }
}, [activeObjectMetadataItem, navigate]); }, [activeObjectMetadataItem, navigate]);
const [objectViews, setObjectViews] = useState<View[]>([]); const [, setObjectViews] = useState<View[]>([]);
const [relationObjectViews, setRelationObjectViews] = useState<View[]>([]); const [, setRelationObjectViews] = useState<View[]>([]);
const { objectMetadataItem: viewObjectMetadataItem } = useObjectMetadataItem({
objectNameSingular: CoreObjectNameSingular.View,
});
useFindManyRecords<View>({ useFindManyRecords<View>({
objectNameSingular: CoreObjectNameSingular.View, objectNameSingular: CoreObjectNameSingular.View,
@ -111,6 +103,8 @@ export const SettingsObjectNewFieldStep2 = () => {
const { createOneRelationMetadataItem: createOneRelationMetadata } = const { createOneRelationMetadataItem: createOneRelationMetadata } =
useCreateOneRelationMetadataItem(); useCreateOneRelationMetadataItem();
const apolloClient = useApolloClient();
if (!activeObjectMetadataItem) return null; if (!activeObjectMetadataItem) return null;
const canSave = const canSave =
@ -126,7 +120,7 @@ export const SettingsObjectNewFieldStep2 = () => {
) { ) {
const { relation: relationFormValues, ...fieldFormValues } = formValues; const { relation: relationFormValues, ...fieldFormValues } = formValues;
const createdRelation = await createOneRelationMetadata({ await createOneRelationMetadata({
relationType: relationFormValues.type, relationType: relationFormValues.type,
field: pick(fieldFormValues, ['icon', 'label', 'description']), field: pick(fieldFormValues, ['icon', 'label', 'description']),
objectMetadataId: activeObjectMetadataItem.id, objectMetadataId: activeObjectMetadataItem.id,
@ -139,111 +133,21 @@ export const SettingsObjectNewFieldStep2 = () => {
}, },
}); });
const relationObjectMetadataItem = findObjectMetadataItemById( // TODO: fix optimistic update logic
relationFormValues.objectMetadataId, // Forcing a refetch for now but it's not ideal
); await apolloClient.refetchQueries({
include: ['FindManyViews', 'CombinedFindManyRecords'],
objectViews.map(async (view) => {
const viewFieldToCreate = {
viewId: view.id,
fieldMetadataId:
relationFormValues.type === 'MANY_TO_ONE'
? createdRelation.data?.createOneRelation.toFieldMetadataId
: createdRelation.data?.createOneRelation.fromFieldMetadataId,
position: activeObjectMetadataItem.fields.length,
isVisible: true,
size: 100,
};
modifyRecordFromCache({
objectMetadataItem: viewObjectMetadataItem,
cache: cache,
fieldModifiers: {
viewFields: (viewFieldsRef, { readField }) => {
const edges = readField<{ node: Reference }[]>(
'edges',
viewFieldsRef,
);
if (!edges) return viewFieldsRef;
return {
...viewFieldsRef,
edges: [...edges, { node: viewFieldToCreate }],
};
},
},
recordId: view.id,
});
relationObjectViews.map(async (view) => {
const viewFieldToCreate = {
viewId: view.id,
fieldMetadataId:
relationFormValues.type === 'MANY_TO_ONE'
? createdRelation.data?.createOneRelation.fromFieldMetadataId
: createdRelation.data?.createOneRelation.toFieldMetadataId,
position: relationObjectMetadataItem?.fields.length,
isVisible: true,
size: 100,
};
modifyRecordFromCache({
objectMetadataItem: viewObjectMetadataItem,
cache: cache,
fieldModifiers: {
viewFields: (viewFieldsRef, { readField }) => {
const edges = readField<{ node: Reference }[]>(
'edges',
viewFieldsRef,
);
if (!edges) return viewFieldsRef;
return {
...viewFieldsRef,
edges: [...edges, { node: viewFieldToCreate }],
};
},
},
recordId: view.id,
});
});
}); });
} else { } else {
const createdMetadataField = await createMetadataField({ await createMetadataField({
...formValues, ...formValues,
objectMetadataId: activeObjectMetadataItem.id, objectMetadataId: activeObjectMetadataItem.id,
}); });
objectViews.map(async (view) => { // TODO: fix optimistic update logic
const viewFieldToCreate = { // Forcing a refetch for now but it's not ideal
viewId: view.id, await apolloClient.refetchQueries({
fieldMetadataId: createdMetadataField.data?.createOneField.id, include: ['FindManyViews', 'CombinedFindManyRecords'],
position: activeObjectMetadataItem.fields.length,
isVisible: true,
size: 100,
};
modifyRecordFromCache({
objectMetadataItem: viewObjectMetadataItem,
cache: cache,
fieldModifiers: {
viewFields: (cachedViewFieldsConnection, { readField }) => {
const edges = readField<RecordGqlRefEdge[]>(
'edges',
cachedViewFieldsConnection,
);
if (!edges) return cachedViewFieldsConnection;
return {
...cachedViewFieldsConnection,
edges: [...edges, { node: viewFieldToCreate }],
};
},
},
recordId: view.id,
});
}); });
} }