Closes #8752, #8753, #8754 Implements usage of aggregate queries in kanban views. https://github.com/user-attachments/assets/732590ca-2785-4c57-82d5-d999a2279e92 TO DO 1. write tests + storybook 2. Fix values displayed should have the same format as defined in number fields + Fix display for amountMicros --------- Co-authored-by: Weiko <corentin@twenty.com>
191 lines
7.1 KiB
TypeScript
191 lines
7.1 KiB
TypeScript
import { CoreObjectNameSingular } from '@/object-metadata/types/CoreObjectNameSingular';
|
|
import { useCreateOneRecord } from '@/object-record/hooks/useCreateOneRecord';
|
|
import { RecordIndexRootPropsContext } from '@/object-record/record-index/contexts/RecordIndexRootPropsContext';
|
|
import { getSnapshotValue } from '@/ui/utilities/recoil-scope/utils/getSnapshotValue';
|
|
import { useRecoilComponentCallbackStateV2 } from '@/ui/utilities/state/component-state/hooks/useRecoilComponentCallbackStateV2';
|
|
import { usePersistViewFieldRecords } from '@/views/hooks/internal/usePersistViewFieldRecords';
|
|
import { usePersistViewFilterGroupRecords } from '@/views/hooks/internal/usePersistViewFilterGroupRecords';
|
|
import { usePersistViewFilterRecords } from '@/views/hooks/internal/usePersistViewFilterRecords';
|
|
import { usePersistViewGroupRecords } from '@/views/hooks/internal/usePersistViewGroupRecords';
|
|
import { usePersistViewSortRecords } from '@/views/hooks/internal/usePersistViewSortRecords';
|
|
import { useGetViewFilterGroupsCombined } from '@/views/hooks/useGetCombinedViewFilterGroups';
|
|
import { useGetViewFiltersCombined } from '@/views/hooks/useGetCombinedViewFilters';
|
|
import { useGetViewSortsCombined } from '@/views/hooks/useGetCombinedViewSorts';
|
|
import { useGetViewFromCache } from '@/views/hooks/useGetViewFromCache';
|
|
import { currentViewIdComponentState } from '@/views/states/currentViewIdComponentState';
|
|
import { isPersistingViewFieldsComponentState } from '@/views/states/isPersistingViewFieldsComponentState';
|
|
import { GraphQLView } from '@/views/types/GraphQLView';
|
|
import { View } from '@/views/types/View';
|
|
import { ViewGroup } from '@/views/types/ViewGroup';
|
|
import { ViewType } from '@/views/types/ViewType';
|
|
import { useContext } from 'react';
|
|
import { useRecoilCallback } from 'recoil';
|
|
import { isDefined } from 'twenty-ui';
|
|
import { v4 } from 'uuid';
|
|
import { isUndefinedOrNull } from '~/utils/isUndefinedOrNull';
|
|
|
|
export const useCreateViewFromCurrentView = (viewBarComponentId?: string) => {
|
|
const currentViewIdCallbackState = useRecoilComponentCallbackStateV2(
|
|
currentViewIdComponentState,
|
|
viewBarComponentId,
|
|
);
|
|
|
|
const isPersistingViewFieldsCallbackState = useRecoilComponentCallbackStateV2(
|
|
isPersistingViewFieldsComponentState,
|
|
viewBarComponentId,
|
|
);
|
|
|
|
const { getViewFromCache } = useGetViewFromCache();
|
|
|
|
const { createOneRecord } = useCreateOneRecord<View>({
|
|
objectNameSingular: CoreObjectNameSingular.View,
|
|
});
|
|
|
|
const { createViewFieldRecords } = usePersistViewFieldRecords();
|
|
|
|
const { getViewSortsCombined } = useGetViewSortsCombined(viewBarComponentId);
|
|
const { getViewFiltersCombined } =
|
|
useGetViewFiltersCombined(viewBarComponentId);
|
|
const { getViewFilterGroupsCombined } =
|
|
useGetViewFilterGroupsCombined(viewBarComponentId);
|
|
|
|
const { createViewSortRecords } = usePersistViewSortRecords();
|
|
|
|
const { createViewGroupRecords } = usePersistViewGroupRecords();
|
|
|
|
const { createViewFilterRecords } = usePersistViewFilterRecords();
|
|
|
|
const { createViewFilterGroupRecords } = usePersistViewFilterGroupRecords();
|
|
|
|
const { objectMetadataItem } = useContext(RecordIndexRootPropsContext);
|
|
|
|
const createViewFromCurrentView = useRecoilCallback(
|
|
({ snapshot, set }) =>
|
|
async (
|
|
{
|
|
id,
|
|
name,
|
|
icon,
|
|
kanbanFieldMetadataId,
|
|
type,
|
|
}: Partial<
|
|
Pick<
|
|
GraphQLView,
|
|
'id' | 'name' | 'icon' | 'kanbanFieldMetadataId' | 'type'
|
|
>
|
|
>,
|
|
shouldCopyFiltersAndSortsAndAggregate?: boolean,
|
|
) => {
|
|
const currentViewId = getSnapshotValue(
|
|
snapshot,
|
|
currentViewIdCallbackState,
|
|
);
|
|
|
|
if (!isDefined(currentViewId)) {
|
|
return;
|
|
}
|
|
|
|
// Here we might instead want to get view from unsaved filters ?
|
|
const sourceView = await getViewFromCache(currentViewId);
|
|
|
|
if (!isDefined(sourceView)) {
|
|
return;
|
|
}
|
|
|
|
set(isPersistingViewFieldsCallbackState, true);
|
|
|
|
const newView = await createOneRecord({
|
|
id: id ?? v4(),
|
|
name: name ?? sourceView.name,
|
|
icon: icon ?? sourceView.icon,
|
|
key: null,
|
|
kanbanFieldMetadataId:
|
|
kanbanFieldMetadataId ?? sourceView.kanbanFieldMetadataId,
|
|
kanbanAggregateOperation: shouldCopyFiltersAndSortsAndAggregate
|
|
? sourceView.kanbanAggregateOperation
|
|
: undefined,
|
|
kanbanAggregateOperationFieldMetadataId:
|
|
shouldCopyFiltersAndSortsAndAggregate
|
|
? sourceView.kanbanAggregateOperationFieldMetadataId
|
|
: undefined,
|
|
type: type ?? sourceView.type,
|
|
objectMetadataId: sourceView.objectMetadataId,
|
|
});
|
|
|
|
if (isUndefinedOrNull(newView)) {
|
|
throw new Error('Failed to create view');
|
|
}
|
|
|
|
await createViewFieldRecords(sourceView.viewFields, newView);
|
|
|
|
if (type === ViewType.Kanban) {
|
|
if (!isDefined(kanbanFieldMetadataId)) {
|
|
throw new Error('Kanban view must have a kanban field');
|
|
}
|
|
|
|
const viewGroupsToCreate =
|
|
objectMetadataItem?.fields
|
|
?.find((field) => field.id === kanbanFieldMetadataId)
|
|
?.options?.map(
|
|
(option, index) =>
|
|
({
|
|
id: v4(),
|
|
__typename: 'ViewGroup',
|
|
fieldMetadataId: kanbanFieldMetadataId,
|
|
fieldValue: option.value,
|
|
isVisible: true,
|
|
position: index,
|
|
}) satisfies ViewGroup,
|
|
) ?? [];
|
|
|
|
viewGroupsToCreate.push({
|
|
__typename: 'ViewGroup',
|
|
id: v4(),
|
|
fieldValue: '',
|
|
position: viewGroupsToCreate.length,
|
|
isVisible: true,
|
|
fieldMetadataId: kanbanFieldMetadataId,
|
|
} satisfies ViewGroup);
|
|
|
|
await createViewGroupRecords(viewGroupsToCreate, newView);
|
|
}
|
|
|
|
if (shouldCopyFiltersAndSortsAndAggregate === true) {
|
|
const sourceViewCombinedFilterGroups = getViewFilterGroupsCombined(
|
|
sourceView.id,
|
|
);
|
|
const sourceViewCombinedFilters = getViewFiltersCombined(
|
|
sourceView.id,
|
|
);
|
|
const sourceViewCombinedSorts = getViewSortsCombined(sourceView.id);
|
|
|
|
await createViewSortRecords(sourceViewCombinedSorts, newView);
|
|
await createViewFilterRecords(sourceViewCombinedFilters, newView);
|
|
await createViewFilterGroupRecords(
|
|
sourceViewCombinedFilterGroups,
|
|
newView,
|
|
);
|
|
}
|
|
|
|
set(isPersistingViewFieldsCallbackState, false);
|
|
},
|
|
[
|
|
objectMetadataItem,
|
|
createViewSortRecords,
|
|
createViewFilterRecords,
|
|
createOneRecord,
|
|
createViewFieldRecords,
|
|
getViewSortsCombined,
|
|
getViewFiltersCombined,
|
|
getViewFilterGroupsCombined,
|
|
currentViewIdCallbackState,
|
|
getViewFromCache,
|
|
isPersistingViewFieldsCallbackState,
|
|
createViewGroupRecords,
|
|
createViewFilterGroupRecords,
|
|
],
|
|
);
|
|
|
|
return { createViewFromCurrentView };
|
|
};
|