# Introduction In this PR we've migrated `twenty-shared` from a `vite` app [libary-mode](https://vite.dev/guide/build#library-mode) to a [preconstruct](https://preconstruct.tools/) "atomic" application ( in the future would like to introduce preconstruct to handle of all our atomic dependencies such as `twenty-emails` `twenty-ui` etc it will be integrated at the monorepo's root directly, would be to invasive in the first, starting incremental via `twenty-shared`) For more information regarding the motivations please refer to nor: - https://github.com/twentyhq/core-team-issues/issues/587 - https://github.com/twentyhq/core-team-issues/issues/281#issuecomment-2630949682 close https://github.com/twentyhq/core-team-issues/issues/589 close https://github.com/twentyhq/core-team-issues/issues/590 ## How to test In order to ease the review this PR will ship all the codegen at the very end, the actual meaning full diff is `+2,411 −114` In order to migrate existing dependent packages to `twenty-shared` multi barrel new arch you need to run in local: ```sh yarn tsx packages/twenty-shared/scripts/migrateFromSingleToMultiBarrelImport.ts && \ npx nx run-many -t lint --fix -p twenty-front twenty-ui twenty-server twenty-emails twenty-shared twenty-zapier ``` Note that `migrateFromSingleToMultiBarrelImport` is idempotent, it's atm included in the PR but should not be merged. ( such as codegen will be added before merging this script will be removed ) ## Misc - related opened issue preconstruct https://github.com/preconstruct/preconstruct/issues/617 ## Closed related PR - https://github.com/twentyhq/twenty/pull/11028 - https://github.com/twentyhq/twenty/pull/10993 - https://github.com/twentyhq/twenty/pull/10960 ## Upcoming enhancement: ( in others dedicated PRs ) - 1/ refactor generate barrel to export atomic module instead of `*` - 2/ generate barrel own package with several files and tests - 3/ Migration twenty-ui the same way - 4/ Use `preconstruct` at monorepo global level ## Conclusion As always any suggestions are welcomed !
188 lines
5.9 KiB
TypeScript
188 lines
5.9 KiB
TypeScript
import styled from '@emotion/styled';
|
|
import { MouseEvent, useMemo, useRef, useState } from 'react';
|
|
import { IconComponent, MenuItem, MenuItemSelect } from 'twenty-ui';
|
|
|
|
import { Dropdown } from '@/ui/layout/dropdown/components/Dropdown';
|
|
import { DropdownMenuItemsContainer } from '@/ui/layout/dropdown/components/DropdownMenuItemsContainer';
|
|
import { DropdownMenuSearchInput } from '@/ui/layout/dropdown/components/DropdownMenuSearchInput';
|
|
import { DropdownMenuSeparator } from '@/ui/layout/dropdown/components/DropdownMenuSeparator';
|
|
import { useDropdown } from '@/ui/layout/dropdown/hooks/useDropdown';
|
|
|
|
import { SelectControl } from '@/ui/input/components/SelectControl';
|
|
import { SelectHotkeyScope } from '../types/SelectHotkeyScope';
|
|
import { isDefined } from 'twenty-shared/utils';
|
|
|
|
export type SelectOption<Value extends string | number | boolean | null> = {
|
|
value: Value;
|
|
label: string;
|
|
Icon?: IconComponent;
|
|
};
|
|
|
|
export type SelectSizeVariant = 'small' | 'default';
|
|
|
|
type CallToActionButton = {
|
|
text: string;
|
|
onClick: (event: MouseEvent<HTMLDivElement>) => void;
|
|
Icon?: IconComponent;
|
|
};
|
|
|
|
export type SelectValue = string | number | boolean | null;
|
|
|
|
export type SelectProps<Value extends SelectValue> = {
|
|
className?: string;
|
|
disabled?: boolean;
|
|
selectSizeVariant?: SelectSizeVariant;
|
|
dropdownId: string;
|
|
dropdownWidth?: `${string}px` | 'auto' | number;
|
|
dropdownWidthAuto?: boolean;
|
|
emptyOption?: SelectOption<Value>;
|
|
fullWidth?: boolean;
|
|
label?: string;
|
|
onChange?: (value: Value) => void;
|
|
onBlur?: () => void;
|
|
options: SelectOption<Value>[];
|
|
value?: Value;
|
|
withSearchInput?: boolean;
|
|
needIconCheck?: boolean;
|
|
callToActionButton?: CallToActionButton;
|
|
};
|
|
|
|
const StyledContainer = styled.div<{ fullWidth?: boolean }>`
|
|
width: ${({ fullWidth }) => (fullWidth ? '100%' : 'auto')};
|
|
`;
|
|
|
|
const StyledLabel = styled.span`
|
|
color: ${({ theme }) => theme.font.color.light};
|
|
display: block;
|
|
font-size: ${({ theme }) => theme.font.size.xs};
|
|
font-weight: ${({ theme }) => theme.font.weight.semiBold};
|
|
margin-bottom: ${({ theme }) => theme.spacing(1)};
|
|
`;
|
|
|
|
export const Select = <Value extends SelectValue>({
|
|
className,
|
|
disabled: disabledFromProps,
|
|
selectSizeVariant,
|
|
dropdownId,
|
|
dropdownWidth = 176,
|
|
dropdownWidthAuto = false,
|
|
emptyOption,
|
|
fullWidth,
|
|
label,
|
|
onChange,
|
|
onBlur,
|
|
options,
|
|
value,
|
|
withSearchInput,
|
|
needIconCheck,
|
|
callToActionButton,
|
|
}: SelectProps<Value>) => {
|
|
const selectContainerRef = useRef<HTMLDivElement>(null);
|
|
|
|
const [searchInputValue, setSearchInputValue] = useState('');
|
|
|
|
const selectedOption =
|
|
options.find(({ value: key }) => key === value) ||
|
|
emptyOption ||
|
|
options[0];
|
|
const filteredOptions = useMemo(
|
|
() =>
|
|
searchInputValue
|
|
? options.filter(({ label }) =>
|
|
label.toLowerCase().includes(searchInputValue.toLowerCase()),
|
|
)
|
|
: options,
|
|
[options, searchInputValue],
|
|
);
|
|
|
|
const isDisabled =
|
|
disabledFromProps ||
|
|
(options.length <= 1 &&
|
|
!isDefined(callToActionButton) &&
|
|
(!isDefined(emptyOption) || selectedOption !== emptyOption));
|
|
|
|
const { closeDropdown } = useDropdown(dropdownId);
|
|
|
|
const dropDownMenuWidth =
|
|
dropdownWidthAuto && selectContainerRef.current?.clientWidth
|
|
? selectContainerRef.current?.clientWidth
|
|
: dropdownWidth;
|
|
|
|
return (
|
|
<StyledContainer
|
|
className={className}
|
|
fullWidth={fullWidth}
|
|
tabIndex={0}
|
|
onBlur={onBlur}
|
|
ref={selectContainerRef}
|
|
>
|
|
{!!label && <StyledLabel>{label}</StyledLabel>}
|
|
{isDisabled ? (
|
|
<SelectControl
|
|
selectedOption={selectedOption}
|
|
isDisabled={isDisabled}
|
|
selectSizeVariant={selectSizeVariant}
|
|
/>
|
|
) : (
|
|
<Dropdown
|
|
dropdownId={dropdownId}
|
|
dropdownMenuWidth={dropDownMenuWidth}
|
|
dropdownPlacement="bottom-start"
|
|
clickableComponent={
|
|
<SelectControl
|
|
selectedOption={selectedOption}
|
|
isDisabled={isDisabled}
|
|
selectSizeVariant={selectSizeVariant}
|
|
/>
|
|
}
|
|
dropdownComponents={
|
|
<>
|
|
{!!withSearchInput && (
|
|
<DropdownMenuSearchInput
|
|
autoFocus
|
|
value={searchInputValue}
|
|
onChange={(event) => setSearchInputValue(event.target.value)}
|
|
/>
|
|
)}
|
|
{!!withSearchInput && !!filteredOptions.length && (
|
|
<DropdownMenuSeparator />
|
|
)}
|
|
{!!filteredOptions.length && (
|
|
<DropdownMenuItemsContainer hasMaxHeight>
|
|
{filteredOptions.map((option) => (
|
|
<MenuItemSelect
|
|
key={`${option.value}-${option.label}`}
|
|
LeftIcon={option.Icon}
|
|
text={option.label}
|
|
selected={selectedOption.value === option.value}
|
|
needIconCheck={needIconCheck}
|
|
onClick={() => {
|
|
onChange?.(option.value);
|
|
onBlur?.();
|
|
closeDropdown();
|
|
}}
|
|
/>
|
|
))}
|
|
</DropdownMenuItemsContainer>
|
|
)}
|
|
{!!callToActionButton && !!filteredOptions.length && (
|
|
<DropdownMenuSeparator />
|
|
)}
|
|
{!!callToActionButton && (
|
|
<DropdownMenuItemsContainer hasMaxHeight scrollable={false}>
|
|
<MenuItem
|
|
onClick={callToActionButton.onClick}
|
|
LeftIcon={callToActionButton.Icon}
|
|
text={callToActionButton.text}
|
|
/>
|
|
</DropdownMenuItemsContainer>
|
|
)}
|
|
</>
|
|
}
|
|
dropdownHotkeyScope={{ scope: SelectHotkeyScope.Select }}
|
|
/>
|
|
)}
|
|
</StyledContainer>
|
|
);
|
|
};
|