mirror of
https://github.com/lensapp/lens.git
synced 2025-05-20 05:10:56 +00:00
* fix parseApi not accepting versions that don't start with 'v' Signed-off-by: Sebastian Malton <smalton@mirantis.com> Co-authored-by: Sebastian Malton <smalton@mirantis.com>
125 lines
3.7 KiB
TypeScript
125 lines
3.7 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, OptionTypeBase } 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<OptionTypeBase>): 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<OptionTypeBase>) {
|
|
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}/>;
|
|
}
|
|
}
|