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:
@ -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>
|
||||||
|
|||||||
@ -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(
|
||||||
|
|||||||
@ -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({
|
||||||
|
|||||||
@ -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);
|
||||||
},
|
},
|
||||||
[
|
[
|
||||||
|
|||||||
@ -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,
|
|
||||||
});
|
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user