* feat: wip import csv * feat: start implementing twenty UI * feat: new radio button component * feat: use new radio button component and fix scroll issue * fix: max height modal * feat: wip try to customize react-data-grid to match design * feat: wip match columns * feat: wip match column selection * feat: match column * feat: clean heading component & try to fix scroll in last step * feat: validation step * fix: small cleaning and remove unused component * feat: clean folder architecture * feat: remove translations * feat: remove chackra theme * feat: remove unused libraries * feat: use option button to open spreadsheet & fix stories * Fix lint and fix imports --------- Co-authored-by: Charles Bochet <charles@twenty.com>
286 lines
8.2 KiB
TypeScript
286 lines
8.2 KiB
TypeScript
import {
|
|
type FormEvent,
|
|
useCallback,
|
|
useEffect,
|
|
useRef,
|
|
useState,
|
|
} from 'react';
|
|
import { useTheme } from '@emotion/react';
|
|
import { useRecoilState, useRecoilValue } from 'recoil';
|
|
import { v4 } from 'uuid';
|
|
|
|
import { useSpreadsheetImport } from '@/spreadsheet-import/hooks/useSpreadsheetImport';
|
|
import { IconButton } from '@/ui/button/components/IconButton';
|
|
import { DropdownMenuHeader } from '@/ui/dropdown/components/DropdownMenuHeader';
|
|
import { DropdownMenuInput } from '@/ui/dropdown/components/DropdownMenuInput';
|
|
import { DropdownMenuItem } from '@/ui/dropdown/components/DropdownMenuItem';
|
|
import { DropdownMenuItemsContainer } from '@/ui/dropdown/components/DropdownMenuItemsContainer';
|
|
import { DropdownMenuSeparator } from '@/ui/dropdown/components/DropdownMenuSeparator';
|
|
import type {
|
|
ViewFieldDefinition,
|
|
ViewFieldMetadata,
|
|
} from '@/ui/editable-field/types/ViewField';
|
|
import DropdownButton from '@/ui/filter-n-sort/components/DropdownButton';
|
|
import {
|
|
IconChevronLeft,
|
|
IconFileImport,
|
|
IconMinus,
|
|
IconPlus,
|
|
IconTag,
|
|
} from '@/ui/icon';
|
|
import {
|
|
hiddenTableColumnsState,
|
|
tableColumnsState,
|
|
visibleTableColumnsState,
|
|
} from '@/ui/table/states/tableColumnsState';
|
|
import { usePreviousHotkeyScope } from '@/ui/utilities/hotkey/hooks/usePreviousHotkeyScope';
|
|
import { useRecoilScopedState } from '@/ui/utilities/recoil-scope/hooks/useRecoilScopedState';
|
|
import { useRecoilScopedValue } from '@/ui/utilities/recoil-scope/hooks/useRecoilScopedValue';
|
|
|
|
import { TableRecoilScopeContext } from '../../states/recoil-scope-contexts/TableRecoilScopeContext';
|
|
import {
|
|
type TableView,
|
|
tableViewEditModeState,
|
|
tableViewsByIdState,
|
|
tableViewsState,
|
|
} from '../../states/tableViewsState';
|
|
import { TableOptionsHotkeyScope } from '../../types/TableOptionsHotkeyScope';
|
|
|
|
import { TableOptionsDropdownSection } from './TableOptionsDropdownSection';
|
|
|
|
type TableOptionsDropdownButtonProps = {
|
|
onColumnsChange?: (columns: ViewFieldDefinition<ViewFieldMetadata>[]) => void;
|
|
onViewsChange?: (views: TableView[]) => void;
|
|
HotkeyScope: TableOptionsHotkeyScope;
|
|
};
|
|
|
|
enum Option {
|
|
Properties = 'Properties',
|
|
}
|
|
|
|
export const TableOptionsDropdownButton = ({
|
|
onColumnsChange,
|
|
onViewsChange,
|
|
HotkeyScope,
|
|
}: TableOptionsDropdownButtonProps) => {
|
|
const theme = useTheme();
|
|
|
|
const { openSpreadsheetImport } = useSpreadsheetImport();
|
|
|
|
const [isUnfolded, setIsUnfolded] = useState(false);
|
|
const [selectedOption, setSelectedOption] = useState<Option | undefined>(
|
|
undefined,
|
|
);
|
|
|
|
const viewEditInputRef = useRef<HTMLInputElement>(null);
|
|
|
|
const [columns, setColumns] = useRecoilState(tableColumnsState);
|
|
const [viewEditMode, setViewEditMode] = useRecoilState(
|
|
tableViewEditModeState,
|
|
);
|
|
const [views, setViews] = useRecoilScopedState(
|
|
tableViewsState,
|
|
TableRecoilScopeContext,
|
|
);
|
|
const visibleColumns = useRecoilValue(visibleTableColumnsState);
|
|
const hiddenColumns = useRecoilValue(hiddenTableColumnsState);
|
|
const viewsById = useRecoilScopedValue(
|
|
tableViewsByIdState,
|
|
TableRecoilScopeContext,
|
|
);
|
|
|
|
const {
|
|
goBackToPreviousHotkeyScope,
|
|
setHotkeyScopeAndMemorizePreviousScope,
|
|
} = usePreviousHotkeyScope();
|
|
|
|
function handleImport() {
|
|
openSpreadsheetImport({
|
|
onSubmit: (datam, file) => {
|
|
console.log('datam', datam);
|
|
console.log('file', file);
|
|
},
|
|
fields: [],
|
|
});
|
|
}
|
|
|
|
const handleColumnVisibilityChange = useCallback(
|
|
(columnId: string, nextIsVisible: boolean) => {
|
|
const nextColumns = columns.map((column) =>
|
|
column.id === columnId
|
|
? { ...column, isVisible: nextIsVisible }
|
|
: column,
|
|
);
|
|
|
|
setColumns(nextColumns);
|
|
onColumnsChange?.(nextColumns);
|
|
},
|
|
[columns, onColumnsChange, setColumns],
|
|
);
|
|
|
|
const renderFieldActions = useCallback(
|
|
(column: ViewFieldDefinition<ViewFieldMetadata>) =>
|
|
// Do not allow hiding last visible column
|
|
!column.isVisible || visibleColumns.length > 1 ? (
|
|
<IconButton
|
|
icon={
|
|
column.isVisible ? (
|
|
<IconMinus size={theme.icon.size.sm} />
|
|
) : (
|
|
<IconPlus size={theme.icon.size.sm} />
|
|
)
|
|
}
|
|
onClick={() =>
|
|
handleColumnVisibilityChange(column.id, !column.isVisible)
|
|
}
|
|
/>
|
|
) : undefined,
|
|
[handleColumnVisibilityChange, theme.icon.size.sm, visibleColumns.length],
|
|
);
|
|
|
|
const resetViewEditMode = useCallback(() => {
|
|
setViewEditMode({ mode: undefined, viewId: undefined });
|
|
|
|
if (viewEditInputRef.current) {
|
|
viewEditInputRef.current.value = '';
|
|
}
|
|
}, [setViewEditMode]);
|
|
|
|
const handleViewNameSubmit = useCallback(
|
|
(event?: FormEvent) => {
|
|
event?.preventDefault();
|
|
|
|
if (viewEditMode.mode && viewEditInputRef.current?.value) {
|
|
const name = viewEditInputRef.current.value;
|
|
const nextViews =
|
|
viewEditMode.mode === 'create'
|
|
? [...views, { id: v4(), name }]
|
|
: views.map((view) =>
|
|
view.id === viewEditMode.viewId ? { ...view, name } : view,
|
|
);
|
|
|
|
(onViewsChange ?? setViews)(nextViews);
|
|
}
|
|
|
|
resetViewEditMode();
|
|
},
|
|
[
|
|
onViewsChange,
|
|
resetViewEditMode,
|
|
setViews,
|
|
viewEditMode.mode,
|
|
viewEditMode.viewId,
|
|
views,
|
|
],
|
|
);
|
|
|
|
const handleSelectOption = useCallback(
|
|
(option: Option) => {
|
|
handleViewNameSubmit();
|
|
setIsUnfolded(true);
|
|
setSelectedOption(option);
|
|
},
|
|
[handleViewNameSubmit],
|
|
);
|
|
|
|
const resetSelectedOption = useCallback(() => {
|
|
setSelectedOption(undefined);
|
|
}, []);
|
|
|
|
const handleUnfoldedChange = useCallback(
|
|
(nextIsUnfolded: boolean) => {
|
|
setIsUnfolded(nextIsUnfolded);
|
|
|
|
if (!nextIsUnfolded) {
|
|
handleViewNameSubmit();
|
|
resetSelectedOption();
|
|
}
|
|
},
|
|
[handleViewNameSubmit, resetSelectedOption],
|
|
);
|
|
|
|
useEffect(() => {
|
|
isUnfolded || viewEditMode.mode
|
|
? setHotkeyScopeAndMemorizePreviousScope(HotkeyScope)
|
|
: goBackToPreviousHotkeyScope();
|
|
}, [
|
|
HotkeyScope,
|
|
goBackToPreviousHotkeyScope,
|
|
isUnfolded,
|
|
setHotkeyScopeAndMemorizePreviousScope,
|
|
viewEditMode.mode,
|
|
]);
|
|
|
|
return (
|
|
<DropdownButton
|
|
label="Options"
|
|
isActive={false}
|
|
isUnfolded={isUnfolded || !!viewEditMode.mode}
|
|
onIsUnfoldedChange={handleUnfoldedChange}
|
|
HotkeyScope={HotkeyScope}
|
|
>
|
|
{!selectedOption && (
|
|
<>
|
|
{!!viewEditMode.mode ? (
|
|
<DropdownMenuInput
|
|
ref={viewEditInputRef}
|
|
autoFocus
|
|
placeholder={
|
|
viewEditMode.mode === 'create' ? 'New view' : 'View name'
|
|
}
|
|
defaultValue={
|
|
viewEditMode.viewId
|
|
? viewsById[viewEditMode.viewId]?.name
|
|
: undefined
|
|
}
|
|
/>
|
|
) : (
|
|
<DropdownMenuHeader>View settings</DropdownMenuHeader>
|
|
)}
|
|
<DropdownMenuSeparator />
|
|
<DropdownMenuItemsContainer>
|
|
<DropdownMenuItem
|
|
onClick={() => handleSelectOption(Option.Properties)}
|
|
>
|
|
<IconTag size={theme.icon.size.md} />
|
|
Properties
|
|
</DropdownMenuItem>
|
|
{false && (
|
|
<DropdownMenuItem onClick={handleImport}>
|
|
<IconFileImport size={theme.icon.size.md} />
|
|
Import
|
|
</DropdownMenuItem>
|
|
)}
|
|
</DropdownMenuItemsContainer>
|
|
</>
|
|
)}
|
|
{selectedOption === Option.Properties && (
|
|
<>
|
|
<DropdownMenuHeader
|
|
startIcon={<IconChevronLeft size={theme.icon.size.md} />}
|
|
onClick={resetSelectedOption}
|
|
>
|
|
Properties
|
|
</DropdownMenuHeader>
|
|
<DropdownMenuSeparator />
|
|
<TableOptionsDropdownSection
|
|
renderActions={renderFieldActions}
|
|
title="Visible"
|
|
columns={visibleColumns}
|
|
/>
|
|
{hiddenColumns.length > 0 && (
|
|
<>
|
|
<DropdownMenuSeparator />
|
|
<TableOptionsDropdownSection
|
|
renderActions={renderFieldActions}
|
|
title="Hidden"
|
|
columns={hiddenColumns}
|
|
/>
|
|
</>
|
|
)}
|
|
</>
|
|
)}
|
|
</DropdownButton>
|
|
);
|
|
};
|