1
0
mirror of https://github.com/lensapp/lens.git synced 2025-05-20 05:10:56 +00:00
lens/dashboard/client/components/select/select.tsx
Jari Kolehmainen 1d0815abd2
Lens app source code (#119)
Signed-off-by: Jari Kolehmainen <jari.kolehmainen@gmail.com>
2020-03-15 09:52:02 +02:00

125 lines
3.6 KiB
TypeScript

// Wrapper for "react-select" component
// API docs: https://react-select.com/
import "./select.scss";
import React, { ReactNode } from "react";
import { computed } from "mobx";
import { observer } from "mobx-react";
import { autobind, cssNames } from "../../utils";
import ReactSelect, { components as ReactSelectComponents } from "react-select"
import { Props as ReactSelectProps } from "react-select/base"
import Creatable, { CreatableProps } from "react-select/creatable"
import { StylesConfig } from "react-select/src/styles"
import { ActionMeta } from "react-select/src/types"
import { themeStore } from "../../theme.store";
export { ReactSelectComponents }
export interface GroupSelectOption<T extends SelectOption = SelectOption> {
label: ReactNode;
options: T[];
}
export interface SelectOption<T = any> {
value: T;
label?: React.ReactNode;
}
export interface SelectProps<T = any> extends ReactSelectProps<T>, CreatableProps<T> {
value?: T;
themeName?: "dark" | "light" | "outlined";
menuClass?: string;
isCreatable?: boolean;
autoConvertOptions?: boolean; // to internal format (i.e. {value: T, label: string}[]), not working with groups
onChange?(option: T, meta?: ActionMeta): void;
}
@observer
export class Select extends React.Component<SelectProps> {
static defaultProps: SelectProps = {
autoConvertOptions: true,
menuPortalTarget: document.body,
}
private theme = this.props.themeName || themeStore.activeTheme.type;
private styles: StylesConfig = {
menuPortal: styles => ({
...styles,
zIndex: "auto"
}),
};
protected isValidOption(opt: SelectOption | any) {
return typeof opt === "object" && opt.value !== undefined;
}
@computed get selectedOption() {
const { value, isMulti } = this.props;
if (isMulti) {
return this.options.filter(opt => {
const values = value ? [].concat(value) : [];
return values.includes(opt) || values.includes(opt.value);
});
}
return this.options.find(opt => opt === value || opt.value === value);
}
@computed get options(): SelectOption[] {
const { autoConvertOptions, options } = this.props;
if (autoConvertOptions && Array.isArray(options)) {
return options.map(opt => {
return this.isValidOption(opt) ? opt : { value: opt, label: String(opt) };
})
}
return options as SelectOption[];
}
@autobind()
onChange(value: SelectOption, meta: ActionMeta) {
if (this.props.onChange) {
this.props.onChange(value, meta);
}
}
@autobind()
onKeyDown(evt: React.KeyboardEvent<HTMLElement>) {
if (this.props.onKeyDown) {
this.props.onKeyDown(evt);
}
const escapeKey = evt.nativeEvent.code === "Escape";
if (escapeKey) evt.stopPropagation(); // don't close the <Dialog/>
}
render() {
const {
className, menuClass, isCreatable, autoConvertOptions,
value, options, components = {}, ...props
} = this.props;
const themeClass = `theme-${this.theme}`;
const selectProps: Partial<SelectProps> = {
...props,
styles: this.styles,
value: autoConvertOptions ? this.selectedOption : value,
options: autoConvertOptions ? this.options : options,
onChange: this.onChange,
onKeyDown: this.onKeyDown,
className: cssNames("Select", themeClass, className),
classNamePrefix: "Select",
components: {
...components,
Menu: props => (
<ReactSelectComponents.Menu
{...props}
className={cssNames(menuClass, themeClass)}
/>
),
}
}
return isCreatable
? <Creatable {...selectProps}/>
: <ReactSelect {...selectProps}/>;
}
}