88 lines
2.6 KiB
TypeScript
88 lines
2.6 KiB
TypeScript
import type { GroupBase, InputProps, OptionProps, ControlProps } from "react-select";
|
|
import { components as reactSelectComponents } from "react-select";
|
|
|
|
import { classNames } from "@calcom/lib";
|
|
|
|
import { Icon } from "../../..";
|
|
import { UpgradeTeamsBadge } from "../../badge";
|
|
import type { SelectProps } from "./Select";
|
|
|
|
export const InputComponent = <
|
|
Option,
|
|
IsMulti extends boolean = false,
|
|
Group extends GroupBase<Option> = GroupBase<Option>
|
|
>({
|
|
inputClassName,
|
|
...props
|
|
}: InputProps<Option, IsMulti, Group>) => {
|
|
return (
|
|
<reactSelectComponents.Input
|
|
// disables our default form focus hightlight on the react-select input element
|
|
inputClassName={classNames(
|
|
"focus:ring-0 focus:ring-offset-0 dark:!text-darkgray-900 !text-emphasis",
|
|
inputClassName
|
|
)}
|
|
{...props}
|
|
/>
|
|
);
|
|
};
|
|
|
|
type ExtendedOption = {
|
|
value: string | number;
|
|
label: string;
|
|
needsTeamsUpgrade?: boolean;
|
|
};
|
|
|
|
export const OptionComponent = <
|
|
Option,
|
|
IsMulti extends boolean = false,
|
|
Group extends GroupBase<Option> = GroupBase<Option>
|
|
>({
|
|
...props
|
|
}: OptionProps<Option, IsMulti, Group>) => {
|
|
return (
|
|
// This gets styled in the select classNames prop now - handles overrides with styles vs className here doesnt
|
|
<reactSelectComponents.Option {...props}>
|
|
<div className="flex">
|
|
<span className="mr-auto" data-testid={`select-option-${(props as unknown as ExtendedOption).value}`}>
|
|
{props.label || <> </>}
|
|
</span>
|
|
{(props.data as unknown as ExtendedOption).needsTeamsUpgrade ? <UpgradeTeamsBadge /> : <></>}
|
|
{props.isSelected && <Icon name="check" className="ml-2 h-4 w-4" />}
|
|
</div>
|
|
</reactSelectComponents.Option>
|
|
);
|
|
};
|
|
|
|
export const ControlComponent = <
|
|
Option,
|
|
IsMulti extends boolean,
|
|
Group extends GroupBase<Option> = GroupBase<Option>
|
|
>(
|
|
controlProps: ControlProps<Option, IsMulti, Group> & {
|
|
selectProps: SelectProps<Option, IsMulti, Group>;
|
|
}
|
|
) => {
|
|
const dataTestId = controlProps.selectProps["data-testid"] ?? "select-control";
|
|
return (
|
|
<span data-testid={dataTestId}>
|
|
<reactSelectComponents.Control {...controlProps} />
|
|
</span>
|
|
);
|
|
};
|
|
|
|
// We need to override this component if we need a icon - we can't simpily override styles
|
|
type IconLeadingProps = {
|
|
icon: React.ReactNode;
|
|
children?: React.ReactNode;
|
|
} & React.ComponentProps<typeof reactSelectComponents.Control>;
|
|
|
|
export const IconLeading = ({ icon, children, ...props }: IconLeadingProps) => {
|
|
return (
|
|
<reactSelectComponents.Control {...props}>
|
|
{icon}
|
|
{children}
|
|
</reactSelectComponents.Control>
|
|
);
|
|
};
|