From f4f8fbd97bc35f12a36fcb6e411abee9a8c1bbfd Mon Sep 17 00:00:00 2001 From: Charles Bochet Date: Fri, 16 Feb 2024 14:32:17 +0100 Subject: [PATCH 1/3] Resolve bugs tied to record creations on table --- .../components/RecordIndexContainer.tsx | 11 +++-- .../states/recordIndexViewTypeState.ts | 8 ++++ .../hooks/useCloseRecordTableCell.ts | 40 ----------------- .../pages/object-record/RecordIndexPage.tsx | 29 +++--------- .../object-record/RecordIndexPageHeader.tsx | 44 +++++++++++++++++++ 5 files changed, 62 insertions(+), 70 deletions(-) create mode 100644 packages/twenty-front/src/modules/object-record/record-index/states/recordIndexViewTypeState.ts create mode 100644 packages/twenty-front/src/pages/object-record/RecordIndexPageHeader.tsx diff --git a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx index 6c522797f1d7..81679b9c64f2 100644 --- a/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx +++ b/packages/twenty-front/src/modules/object-record/record-index/components/RecordIndexContainer.tsx @@ -1,6 +1,5 @@ -import { useState } from 'react'; import styled from '@emotion/styled'; -import { useRecoilCallback, useSetRecoilState } from 'recoil'; +import { useRecoilCallback, useRecoilState, useSetRecoilState } from 'recoil'; import { useColumnDefinitionsFromFieldMetadata } from '@/object-metadata/hooks/useColumnDefinitionsFromFieldMetadata'; import { useObjectMetadataItem } from '@/object-metadata/hooks/useObjectMetadataItem'; @@ -16,6 +15,7 @@ import { recordIndexFieldDefinitionsState } from '@/object-record/record-index/s import { recordIndexFiltersState } from '@/object-record/record-index/states/recordIndexFiltersState'; import { recordIndexIsCompactModeActiveState } from '@/object-record/record-index/states/recordIndexIsCompactModeActiveState'; import { recordIndexSortsState } from '@/object-record/record-index/states/recordIndexSortsState'; +import { recordIndexViewTypeState } from '@/object-record/record-index/states/recordIndexViewTypeState'; import { useRecordTable } from '@/object-record/record-table/hooks/useRecordTable'; import { SpreadsheetImportProvider } from '@/spreadsheet-import/provider/components/SpreadsheetImportProvider'; import { ViewBar } from '@/views/components/ViewBar'; @@ -46,10 +46,9 @@ export const RecordIndexContainer = ({ recordIndexId, objectNamePlural, }: RecordIndexContainerProps) => { - const [recordIndexViewType, setRecordIndexViewType] = useState< - ViewType | undefined - >(undefined); - + const [recordIndexViewType, setRecordIndexViewType] = useRecoilState( + recordIndexViewTypeState, + ); const { objectNameSingular } = useObjectNameSingularFromPlural({ objectNamePlural, }); diff --git a/packages/twenty-front/src/modules/object-record/record-index/states/recordIndexViewTypeState.ts b/packages/twenty-front/src/modules/object-record/record-index/states/recordIndexViewTypeState.ts new file mode 100644 index 000000000000..28dcdef038af --- /dev/null +++ b/packages/twenty-front/src/modules/object-record/record-index/states/recordIndexViewTypeState.ts @@ -0,0 +1,8 @@ +import { atom } from 'recoil'; + +import { ViewType } from '@/views/types/ViewType'; + +export const recordIndexViewTypeState = atom({ + key: 'recordIndexViewTypeState', + default: undefined, +}); diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCloseRecordTableCell.ts b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCloseRecordTableCell.ts index 55d5ac99dda4..9ba618747f1a 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCloseRecordTableCell.ts +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/useCloseRecordTableCell.ts @@ -1,59 +1,19 @@ -import { useContext } from 'react'; -import { useRecoilCallback, useRecoilValue } from 'recoil'; - -import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; -import { useRecordFieldInput } from '@/object-record/record-field/hooks/useRecordFieldInput'; -import { EntityDeleteContext } from '@/object-record/record-table/contexts/EntityDeleteHookContext'; -import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; -import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; import { useDragSelect } from '@/ui/utilities/drag-select/hooks/useDragSelect'; import { useSetHotkeyScope } from '@/ui/utilities/hotkey/hooks/useSetHotkeyScope'; -import { getSnapshotValue } from '@/ui/utilities/recoil-scope/utils/getSnapshotValue'; import { useCloseCurrentTableCellInEditMode } from '../../hooks/internal/useCloseCurrentTableCellInEditMode'; import { TableHotkeyScope } from '../../types/TableHotkeyScope'; export const useCloseRecordTableCell = () => { - const { getTableRowIdsState } = useRecordTableStates(); - const { columnIndex } = useContext(RecordTableCellContext); - const { entityId, fieldDefinition } = useContext(FieldContext); - const deleteOneRecord = useContext(EntityDeleteContext); - const setHotkeyScope = useSetHotkeyScope(); const { setDragSelectionStartEnabled } = useDragSelect(); const closeCurrentTableCellInEditMode = useCloseCurrentTableCellInEditMode(); - const { - getDraftValueSelector: getFieldInputDraftValueSelector, - isDraftValueEmpty: isCurrentFieldInputValueEmpty, - } = useRecordFieldInput( - `${entityId}-${fieldDefinition?.metadata?.fieldName}`, - ); - - const currentFieldInputDraftValue = useRecoilValue( - getFieldInputDraftValueSelector(), - ); - - const isFirstColumnCell = columnIndex === 0; - - const deleteRow = useRecoilCallback(({ snapshot }) => async () => { - const tableRowIds = getSnapshotValue(snapshot, getTableRowIdsState()); - - await deleteOneRecord(tableRowIds[0]); - }); - const closeTableCell = async () => { setDragSelectionStartEnabled(true); closeCurrentTableCellInEditMode(); setHotkeyScope(TableHotkeyScope.TableSoftFocus); - - if ( - isFirstColumnCell && - isCurrentFieldInputValueEmpty(currentFieldInputDraftValue) - ) { - await deleteRow(); - } }; return { diff --git a/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx b/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx index 2c87d997a780..61f719d43766 100644 --- a/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx +++ b/packages/twenty-front/src/pages/object-record/RecordIndexPage.tsx @@ -1,19 +1,15 @@ import { useParams } from 'react-router-dom'; import styled from '@emotion/styled'; -import { useObjectMetadataItemForSettings } from '@/object-metadata/hooks/useObjectMetadataItemForSettings'; import { useObjectNameSingularFromPlural } from '@/object-metadata/hooks/useObjectNameSingularFromPlural'; import { useCreateOneRecord } from '@/object-record/hooks/useCreateOneRecord'; import { RecordIndexContainer } from '@/object-record/record-index/components/RecordIndexContainer'; import { DEFAULT_CELL_SCOPE } from '@/object-record/record-table/record-table-cell/hooks/useOpenRecordTableCell'; import { useSelectedTableCellEditMode } from '@/object-record/record-table/record-table-cell/hooks/useSelectedTableCellEditMode'; -import { useIcons } from '@/ui/display/icon/hooks/useIcons'; -import { PageAddButton } from '@/ui/layout/page/PageAddButton'; import { PageBody } from '@/ui/layout/page/PageBody'; import { PageContainer } from '@/ui/layout/page/PageContainer'; -import { PageHeader } from '@/ui/layout/page/PageHeader'; -import { PageHotkeysEffect } from '@/ui/layout/page/PageHotkeysEffect'; import { useSetHotkeyScope } from '@/ui/utilities/hotkey/hooks/useSetHotkeyScope'; +import { RecordIndexPageHeader } from '~/pages/object-record/RecordIndexPageHeader'; const StyledIndexContainer = styled.div` display: flex; @@ -28,20 +24,11 @@ export const RecordIndexPage = () => { objectNamePlural, }); - const { findObjectMetadataItemByNamePlural } = - useObjectMetadataItemForSettings(); - - const { getIcon } = useIcons(); - const Icon = getIcon( - findObjectMetadataItemByNamePlural(objectNamePlural)?.icon, - ); - const { createOneRecord: createOneObject } = useCreateOneRecord({ objectNameSingular, }); const recordIndexId = objectNamePlural ?? ''; - const setHotkeyScope = useSetHotkeyScope(); const { setSelectedTableCellEditMode } = useSelectedTableCellEditMode({ @@ -49,7 +36,9 @@ export const RecordIndexPage = () => { }); const handleAddButtonClick = async () => { - await createOneObject?.({}); + await createOneObject?.({ + position: 0, + }); setSelectedTableCellEditMode(0, 0); setHotkeyScope(DEFAULT_CELL_SCOPE.scope, DEFAULT_CELL_SCOPE.customScopes); @@ -57,15 +46,7 @@ export const RecordIndexPage = () => { return ( - - - - + void; +}; + +export const RecordIndexPageHeader = ({ + createRecord, +}: RecordIndexPageHeaderProps) => { + const objectNamePlural = useParams().objectNamePlural ?? ''; + + const { findObjectMetadataItemByNamePlural } = + useObjectMetadataItemForSettings(); + + const { getIcon } = useIcons(); + const Icon = getIcon( + findObjectMetadataItemByNamePlural(objectNamePlural)?.icon, + ); + + const recordIndexViewType = useRecoilValue(recordIndexViewTypeState); + + return ( + + + {recordIndexViewType === ViewType.Table && ( + + )} + + ); +}; From 49661dd2323652fad002fe0e23e1c354848d9c59 Mon Sep 17 00:00:00 2001 From: Charles Bochet Date: Fri, 16 Feb 2024 14:52:02 +0100 Subject: [PATCH 2/3] Fix according to PR --- .../src/pages/object-record/RecordIndexPageHeader.tsx | 8 ++------ .../standard-objects-prefill-data/company.ts | 6 ++++++ .../standard-objects-prefill-data/person.ts | 6 ++++++ 3 files changed, 14 insertions(+), 6 deletions(-) diff --git a/packages/twenty-front/src/pages/object-record/RecordIndexPageHeader.tsx b/packages/twenty-front/src/pages/object-record/RecordIndexPageHeader.tsx index e389c3cbd1bb..fd24ab76f1b3 100644 --- a/packages/twenty-front/src/pages/object-record/RecordIndexPageHeader.tsx +++ b/packages/twenty-front/src/pages/object-record/RecordIndexPageHeader.tsx @@ -8,6 +8,7 @@ import { PageAddButton } from '@/ui/layout/page/PageAddButton'; import { PageHeader } from '@/ui/layout/page/PageHeader'; import { PageHotkeysEffect } from '@/ui/layout/page/PageHotkeysEffect'; import { ViewType } from '@/views/types/ViewType'; +import { capitalize } from '~/utils/string/capitalize'; type RecordIndexPageHeaderProps = { createRecord: () => void; @@ -29,12 +30,7 @@ export const RecordIndexPageHeader = ({ const recordIndexViewType = useRecoilValue(recordIndexViewTypeState); return ( - + {recordIndexViewType === ViewType.Table && ( diff --git a/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/company.ts b/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/company.ts index 814615457803..9899ffd1ca44 100644 --- a/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/company.ts +++ b/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/company.ts @@ -12,6 +12,7 @@ export const companyPrefillData = async ( 'domainName', 'address', 'employees', + 'position', ]) .orIgnore() .values([ @@ -20,30 +21,35 @@ export const companyPrefillData = async ( domainName: 'airbnb.com', address: 'San Francisco', employees: 5000, + position: 1, }, { name: 'Qonto', domainName: 'qonto.com', address: 'San Francisco', employees: 800, + position: 2, }, { name: 'Stripe', domainName: 'stripe.com', address: 'San Francisco', employees: 8000, + position: 3, }, { name: 'Figma', domainName: 'figma.com', address: 'San Francisco', employees: 800, + position: 4, }, { name: 'Notion', domainName: 'notion.com', address: 'San Francisco', employees: 400, + position: 5, }, ]) .returning('*') diff --git a/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/person.ts b/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/person.ts index 205ba5f43448..68d82ebb1eb9 100644 --- a/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/person.ts +++ b/packages/twenty-server/src/workspace/workspace-manager/standard-objects-prefill-data/person.ts @@ -13,6 +13,7 @@ export const personPrefillData = async ( 'city', 'email', 'avatarUrl', + 'position', ]) .orIgnore() .values([ @@ -23,6 +24,7 @@ export const personPrefillData = async ( email: 'chesky@airbnb.com', avatarUrl: '', + position: 1, }, { nameFirstName: 'Alexandre', @@ -31,6 +33,7 @@ export const personPrefillData = async ( email: 'prot@qonto.com', avatarUrl: '', + position: 2, }, { nameFirstName: 'Patrick', @@ -39,6 +42,7 @@ export const personPrefillData = async ( email: 'collison@stripe.com', avatarUrl: '', + position: 3, }, { nameFirstName: 'Dylan', @@ -47,6 +51,7 @@ export const personPrefillData = async ( email: 'field@figma.com', avatarUrl: '', + position: 4, }, { nameFirstName: 'Ivan', @@ -55,6 +60,7 @@ export const personPrefillData = async ( email: 'zhao@notion.com', avatarUrl: '', + position: 5, }, ]) .returning('*') From 325f3e0d99494c0d1642533c69af698500e59ee5 Mon Sep 17 00:00:00 2001 From: Charles Bochet Date: Fri, 16 Feb 2024 14:57:44 +0100 Subject: [PATCH 3/3] Fix tests --- .../hooks/__tests__/useCloseRecordTableCell.test.tsx | 7 +------ 1 file changed, 1 insertion(+), 6 deletions(-) diff --git a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useCloseRecordTableCell.test.tsx b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useCloseRecordTableCell.test.tsx index 5370062c90e8..c608e921e635 100644 --- a/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useCloseRecordTableCell.test.tsx +++ b/packages/twenty-front/src/modules/object-record/record-table/record-table-cell/hooks/__tests__/useCloseRecordTableCell.test.tsx @@ -3,7 +3,6 @@ import { RecoilRoot, useRecoilValue } from 'recoil'; import { textfieldDefinition } from '@/object-record/record-field/__mocks__/fieldDefinitions'; import { FieldContext } from '@/object-record/record-field/contexts/FieldContext'; -import { EntityDeleteContext } from '@/object-record/record-table/contexts/EntityDeleteHookContext'; import { RecordTableCellContext } from '@/object-record/record-table/contexts/RecordTableCellContext'; import { RecordTableRowContext } from '@/object-record/record-table/contexts/RecordTableRowContext'; import { useRecordTableStates } from '@/object-record/record-table/hooks/internal/useRecordTableStates'; @@ -23,7 +22,6 @@ jest.mock('@/ui/utilities/hotkey/hooks/useSetHotkeyScope', () => ({ })); const onColumnsChange = jest.fn(); -const deleteEntity = jest.fn(); const scopeId = 'scopeId'; const Wrapper = ({ children }: { children: React.ReactNode }) => ( @@ -44,9 +42,7 @@ const Wrapper = ({ children }: { children: React.ReactNode }) => ( - - {children} - + {children} @@ -86,6 +82,5 @@ describe('useCloseRecordTableCell', () => { expect(result.current.isDragSelectionStartEnabled()).toBe(true); expect(result.current.isTableCellInEditMode).toBe(false); expect(setHotkeyScope).toHaveBeenCalledWith('table-soft-focus'); - expect(deleteEntity).toHaveBeenCalled(); }); });