mirror of
https://github.com/fleetdm/fleet
synced 2026-04-21 13:37:30 +00:00
Implements patch policies #31914 - https://github.com/fleetdm/fleet/pull/40816 - https://github.com/fleetdm/fleet/pull/41248 - https://github.com/fleetdm/fleet/pull/41276 - https://github.com/fleetdm/fleet/pull/40948 - https://github.com/fleetdm/fleet/pull/40837 - https://github.com/fleetdm/fleet/pull/40956 - https://github.com/fleetdm/fleet/pull/41168 - https://github.com/fleetdm/fleet/pull/41171 - https://github.com/fleetdm/fleet/pull/40691 - https://github.com/fleetdm/fleet/pull/41524 - https://github.com/fleetdm/fleet/pull/41674 --------- Co-authored-by: Jonathan Katz <44128041+jkatz01@users.noreply.github.com> Co-authored-by: jkatz01 <yehonatankatz@gmail.com> Co-authored-by: RachelElysia <71795832+RachelElysia@users.noreply.github.com> Co-authored-by: Jahziel Villasana-Espinoza <jahziel@fleetdm.com>
537 lines
16 KiB
TypeScript
537 lines
16 KiB
TypeScript
/**
|
|
* This is a new component built off react-select 5.4
|
|
* meant to replace Dropdown.jsx built off react-select 1.3
|
|
*
|
|
* See storybook component for current functionality
|
|
*
|
|
* Prototyped on UserForm.tsx but added and tested the following:
|
|
* Options: text, disabled, option helptext, option tooltip
|
|
* Other: label text, dropdown help text, dropdown error
|
|
*/
|
|
|
|
import classnames from "classnames";
|
|
import React from "react";
|
|
import Select, {
|
|
components,
|
|
DropdownIndicatorProps,
|
|
GroupBase,
|
|
OptionProps,
|
|
PropsValue,
|
|
SingleValue,
|
|
StylesConfig,
|
|
ValueContainerProps,
|
|
} from "react-select-5";
|
|
|
|
import { COLORS } from "styles/var/colors";
|
|
import { PADDING } from "styles/var/padding";
|
|
|
|
import FormField from "components/forms/FormField";
|
|
import DropdownOptionTooltipWrapper from "components/forms/fields/Dropdown/DropdownOptionTooltipWrapper";
|
|
import Icon from "components/Icon";
|
|
import { IconNames } from "components/icons";
|
|
import { TooltipContent } from "interfaces/dropdownOption";
|
|
|
|
interface CustomOptionProps
|
|
extends Omit<OptionProps<CustomOptionType, false>, "data"> {
|
|
data: CustomOptionType;
|
|
}
|
|
|
|
const baseClass = "dropdown-wrapper";
|
|
|
|
const CustomOption = (props: CustomOptionProps) => {
|
|
const { data, ...rest } = props;
|
|
|
|
const optionContent = (
|
|
<div className={`${baseClass}__option`} data-testid="dropdown-option">
|
|
{data.label}
|
|
{data.helpText && (
|
|
<span className={`${baseClass}__help-text`}>{data.helpText}</span>
|
|
)}
|
|
</div>
|
|
);
|
|
|
|
return (
|
|
<components.Option {...rest} data={data}>
|
|
{data.tooltipContent ? (
|
|
<DropdownOptionTooltipWrapper tipContent={data.tooltipContent}>
|
|
{optionContent}
|
|
</DropdownOptionTooltipWrapper>
|
|
) : (
|
|
optionContent
|
|
)}
|
|
</components.Option>
|
|
);
|
|
};
|
|
|
|
export const CustomDropdownIndicator = (
|
|
props: DropdownIndicatorProps<
|
|
CustomOptionType,
|
|
false,
|
|
GroupBase<CustomOptionType>
|
|
>
|
|
) => {
|
|
const { isFocused, selectProps } = props;
|
|
const color =
|
|
isFocused || selectProps.menuIsOpen
|
|
? "ui-fleet-black-75-over"
|
|
: "ui-fleet-black-75";
|
|
|
|
return (
|
|
<components.DropdownIndicator
|
|
{...props}
|
|
className={`${baseClass}__indicator`}
|
|
>
|
|
<Icon
|
|
name="chevron-down"
|
|
color={color}
|
|
className={`${baseClass}__icon`}
|
|
/>
|
|
</components.DropdownIndicator>
|
|
);
|
|
};
|
|
|
|
export interface CustomOptionType {
|
|
label: React.ReactNode;
|
|
value: string;
|
|
tooltipContent?: TooltipContent;
|
|
helpText?: React.ReactNode;
|
|
isDisabled?: boolean;
|
|
iconName?: IconNames;
|
|
}
|
|
|
|
type DropdownWrapperVariant = "table-filter" | "button";
|
|
|
|
export interface IDropdownWrapper {
|
|
options: CustomOptionType[];
|
|
value?: PropsValue<CustomOptionType> | string; // Future: Handle number types, cascade of type checking will be needed
|
|
onChange: (newValue: SingleValue<CustomOptionType>) => void;
|
|
name: string;
|
|
className?: string;
|
|
wrapperClassname?: string;
|
|
labelClassname?: string;
|
|
error?: string;
|
|
label?: JSX.Element | string;
|
|
helpText?: JSX.Element | string;
|
|
isSearchable?: boolean;
|
|
isDisabled?: boolean;
|
|
iconName?: IconNames;
|
|
placeholder?: string;
|
|
/** E.g. scroll to view dropdown menu in a scrollable parent container */
|
|
onMenuOpen?: () => void;
|
|
/** Table filter dropdowns have filter icon and height: 40px
|
|
* Button dropdowns have hover/active state, padding, height matching actual buttons, and no selected option styling */
|
|
variant?: DropdownWrapperVariant;
|
|
/** This makes the menu fit all text without wrapping,
|
|
* aligning right to fit text on screen */
|
|
nowrapMenu?: boolean;
|
|
customNoOptionsMessage?: string;
|
|
}
|
|
|
|
const getOptionBackgroundColor = (
|
|
state: OptionProps<CustomOptionType, false>
|
|
) => {
|
|
return state.isFocused ? COLORS["ui-fleet-black-5"] : "transparent";
|
|
};
|
|
|
|
const getOptionFontWeight = (
|
|
state: OptionProps<CustomOptionType, false>,
|
|
variant?: DropdownWrapperVariant
|
|
) => {
|
|
// For "button" dropdowns, selected options are not styled differently
|
|
if (variant === "button") {
|
|
return "normal";
|
|
}
|
|
|
|
// For other variants, selected options are bold
|
|
return state.isSelected ? "600" : "normal";
|
|
};
|
|
|
|
/** generates the default custom styles for the dropdown component.
|
|
* NOTE: we export this from DropdownWrapper components so that other more
|
|
* customisable dropdown components can use this for consistency in styling */
|
|
export const generateCustomDropdownStyles = (
|
|
variant?: DropdownWrapperVariant,
|
|
isDisabled = false,
|
|
nowrapMenu = false
|
|
): StylesConfig<CustomOptionType, false> => {
|
|
return {
|
|
container: (provided) => {
|
|
const buttonVariantContainer = {
|
|
borderRadius: "6px",
|
|
"&:active": {
|
|
backgroundColor: "rgba(25, 33, 71, 0.05)",
|
|
},
|
|
height: "38px",
|
|
};
|
|
|
|
return {
|
|
...provided,
|
|
width: "100%",
|
|
height: "36px",
|
|
...(variant === "button" && buttonVariantContainer),
|
|
};
|
|
},
|
|
|
|
control: (provided, state) => {
|
|
if (variant === "button") {
|
|
return {
|
|
backgroundColor: "initial",
|
|
borderColor: "none",
|
|
display: "flex",
|
|
flexDirection: "row",
|
|
width: "max-content",
|
|
padding: PADDING["pad-small"],
|
|
border: 0,
|
|
borderRadius: "6px",
|
|
boxShadow: "none",
|
|
cursor: "pointer",
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75"],
|
|
},
|
|
"&:hover": {
|
|
backgroundColor: "rgba(25, 33, 71, 0.05)",
|
|
boxShadow: "none",
|
|
".dropdown-wrapper__placeholder": {
|
|
color: COLORS["ui-fleet-black-75-over"],
|
|
},
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75-over"],
|
|
},
|
|
},
|
|
".react-select__control--is-focused": {
|
|
backgroundColor: "rgba(25, 33, 71, 0.05)",
|
|
boxShadow: "none",
|
|
".dropdown-wrapper__placeholder": {
|
|
color: COLORS["ui-fleet-black-75-down"],
|
|
},
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75-down"],
|
|
},
|
|
},
|
|
...(state.isFocused && {
|
|
backgroundColor: "rgba(25, 33, 71, 0.05)",
|
|
".dropdown-wrapper__placeholder": {
|
|
color: COLORS["ui-fleet-black-75-down"],
|
|
},
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75-down"],
|
|
},
|
|
}),
|
|
// TODO: Figure out a way to apply separate &:focus-visible styling
|
|
// Currently only relying on &:focus styling for tabbing through app
|
|
...(state.menuIsOpen && {
|
|
".dropdown-wrapper__indicator svg": {
|
|
transform: "rotate(180deg)",
|
|
transition: "transform 0.25s ease",
|
|
},
|
|
}),
|
|
...(variant === "button" && { height: "22px" }),
|
|
};
|
|
}
|
|
|
|
return {
|
|
...provided,
|
|
display: "flex",
|
|
flexDirection: "row",
|
|
width: "100%",
|
|
backgroundColor: COLORS["core-fleet-white"],
|
|
paddingLeft: "8px", // TODO: Update to match styleguide of (16px) when updating rest of UI (8px)
|
|
paddingRight: "8px",
|
|
cursor: "pointer",
|
|
boxShadow: "none",
|
|
borderRadius: "4px",
|
|
borderColor: state.isFocused
|
|
? COLORS["core-fleet-black"]
|
|
: COLORS["ui-fleet-black-10"],
|
|
"&:hover": {
|
|
boxShadow: "none",
|
|
borderColor: COLORS["ui-fleet-black-50"],
|
|
".dropdown-wrapper__single-value": {
|
|
color: COLORS["ui-fleet-black-75"],
|
|
},
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75"],
|
|
},
|
|
".filter-icon path": {
|
|
fill: COLORS["ui-fleet-black-75"],
|
|
},
|
|
},
|
|
// When tabbing
|
|
// Relies on --is-focused for styling as &:focus-visible cannot be applied
|
|
"&.react-select__control--is-focused": {
|
|
borderColor: state.isFocused
|
|
? COLORS["core-fleet-black"]
|
|
: COLORS["ui-fleet-black-25"],
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75"],
|
|
},
|
|
".filter-icon path": {
|
|
fill: COLORS["ui-fleet-black-75"],
|
|
},
|
|
},
|
|
...(state.isFocused && {
|
|
".dropdown-wrapper__placeholder": {
|
|
color: COLORS["ui-fleet-black-75"],
|
|
},
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75"],
|
|
},
|
|
}),
|
|
...(state.isDisabled && {
|
|
".dropdown-wrapper__single-value": {
|
|
color: COLORS["ui-fleet-black-50"],
|
|
},
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-50"],
|
|
},
|
|
".filter-icon path": {
|
|
fill: COLORS["ui-fleet-black-50"],
|
|
},
|
|
}),
|
|
"&:active": {
|
|
".dropdown-wrapper__single-value": {
|
|
color: COLORS["ui-fleet-black-75"],
|
|
},
|
|
".dropdown-wrapper__indicator path": {
|
|
stroke: COLORS["ui-fleet-black-75"],
|
|
},
|
|
".filter-icon path": {
|
|
fill: COLORS["ui-fleet-black-75"],
|
|
},
|
|
},
|
|
...(state.menuIsOpen && {
|
|
".dropdown-wrapper__indicator svg": {
|
|
transform: "rotate(180deg)",
|
|
transition: "transform 0.25s ease",
|
|
},
|
|
}),
|
|
};
|
|
},
|
|
placeholder: (provided, state) => {
|
|
const buttonVariantPlaceholder = {
|
|
color: state.isFocused
|
|
? COLORS["ui-fleet-black-75-over"]
|
|
: COLORS["ui-fleet-black-75"],
|
|
fontSize: "13px",
|
|
fontWeight: "600",
|
|
lineHeight: "normal",
|
|
paddingLeft: 0,
|
|
opacity: isDisabled ? 0.5 : 1,
|
|
marginTop: variant === "button" ? "-1px" : "1px", // TODO: Figure out vertical centering to not need pixel fix
|
|
};
|
|
|
|
return {
|
|
...provided,
|
|
fontSize: "13px",
|
|
...(variant === "button" && buttonVariantPlaceholder),
|
|
};
|
|
},
|
|
singleValue: (provided) => ({
|
|
...provided,
|
|
fontSize: "13px",
|
|
margin: 0,
|
|
padding: 0,
|
|
}),
|
|
dropdownIndicator: (provided) => ({
|
|
...provided,
|
|
display: "flex",
|
|
padding: "2px",
|
|
svg: {
|
|
transition: "transform 0.25s ease",
|
|
},
|
|
opacity: isDisabled ? 0.5 : 1,
|
|
}),
|
|
menu: (provided) => ({
|
|
...provided,
|
|
boxShadow: "0 2px 6px rgba(0, 0, 0, 0.1)",
|
|
borderRadius: "4px",
|
|
zIndex: 6,
|
|
overflow: "hidden",
|
|
border: 0,
|
|
marginTop: "3px",
|
|
left: 0,
|
|
maxHeight: "none",
|
|
position: "absolute",
|
|
animation: "fade-in 150ms ease-out",
|
|
...(nowrapMenu && {
|
|
width: "fit-content",
|
|
left: "auto",
|
|
right: "0",
|
|
}),
|
|
}),
|
|
menuList: (provided) => ({
|
|
...provided,
|
|
padding: PADDING["pad-small"],
|
|
maxHeight: "none",
|
|
...(nowrapMenu && { width: "fit-content" }),
|
|
}),
|
|
valueContainer: (provided) => ({
|
|
...provided,
|
|
padding: 0,
|
|
display: "flex",
|
|
gap: PADDING[variant === "button" ? "pad-xsmall" : "pad-small"],
|
|
flexWrap: "nowrap", // This ensures the value is on a single line and truncated
|
|
}),
|
|
option: (provided, state) => ({
|
|
...provided,
|
|
padding: "10px 8px",
|
|
fontSize: "13px",
|
|
borderRadius: "4px",
|
|
backgroundColor: getOptionBackgroundColor(state),
|
|
fontWeight: getOptionFontWeight(state, variant),
|
|
color: COLORS["core-fleet-black"],
|
|
"&:hover": {
|
|
backgroundColor: state.isDisabled
|
|
? "transparent"
|
|
: COLORS["ui-fleet-black-5"],
|
|
cursor: state.isDisabled ? "not-allowed" : "pointer",
|
|
},
|
|
"&:active": {
|
|
backgroundColor: state.isDisabled
|
|
? "transparent"
|
|
: COLORS["ui-fleet-black-5"],
|
|
},
|
|
...(state.isDisabled && {
|
|
color: COLORS["ui-fleet-black-50"],
|
|
fontStyle: "italic",
|
|
cursor: "not-allowed",
|
|
}),
|
|
// Styles for custom option
|
|
".dropdown-wrapper__option": {
|
|
display: "flex",
|
|
flexDirection: "column",
|
|
gap: "8px",
|
|
width: "100%",
|
|
whiteSpace: nowrapMenu ? "nowrap" : "normal",
|
|
},
|
|
".dropdown-wrapper__help-text": {
|
|
fontSize: "12px",
|
|
width: "100%",
|
|
whiteSpace: nowrapMenu ? "nowrap" : "normal",
|
|
color: state.isDisabled
|
|
? COLORS["ui-fleet-black-50"]
|
|
: COLORS["ui-fleet-black-75"],
|
|
fontStyle: "italic",
|
|
fontWeight: "normal",
|
|
},
|
|
}),
|
|
menuPortal: (base) => ({ ...base, zIndex: 999 }), // Not hidden beneath scrollable sections
|
|
noOptionsMessage: (provided) => ({
|
|
...provided,
|
|
textAlign: "left",
|
|
fontSize: "13px",
|
|
padding: "10px 8px",
|
|
}),
|
|
};
|
|
};
|
|
|
|
const DropdownWrapper = ({
|
|
options,
|
|
value,
|
|
onChange,
|
|
name,
|
|
className,
|
|
labelClassname,
|
|
wrapperClassname,
|
|
error,
|
|
label,
|
|
helpText,
|
|
isSearchable = false,
|
|
isDisabled = false,
|
|
iconName,
|
|
placeholder,
|
|
onMenuOpen,
|
|
variant,
|
|
nowrapMenu,
|
|
customNoOptionsMessage,
|
|
}: IDropdownWrapper) => {
|
|
const wrapperClassNames = classnames(baseClass, className, {
|
|
[`${baseClass}__table-filter`]: variant === "table-filter",
|
|
[`${wrapperClassname}`]: !!wrapperClassname,
|
|
});
|
|
|
|
const handleChange = (newValue: SingleValue<CustomOptionType>) => {
|
|
onChange(newValue);
|
|
};
|
|
|
|
// Ability to handle value of type string or CustomOptionType
|
|
const getCurrentValue = () => {
|
|
if (typeof value === "string") {
|
|
return options.find((option) => option.value === value) || null;
|
|
}
|
|
return value;
|
|
};
|
|
|
|
const ValueContainer = ({
|
|
children,
|
|
...props
|
|
}: ValueContainerProps<CustomOptionType, false>) => {
|
|
const iconToDisplay =
|
|
iconName || (variant === "table-filter" ? "filter" : null);
|
|
|
|
return (
|
|
components.ValueContainer && (
|
|
<components.ValueContainer {...props}>
|
|
{!!children && iconToDisplay && (
|
|
<Icon name={iconToDisplay} className="filter-icon" />
|
|
)}
|
|
{children}
|
|
</components.ValueContainer>
|
|
)
|
|
);
|
|
};
|
|
|
|
const renderLabel = () => {
|
|
const labelWrapperClasses = classnames(
|
|
`${baseClass}__label`,
|
|
labelClassname,
|
|
{
|
|
[`${baseClass}__label--error`]: !!error,
|
|
[`${baseClass}__label--disabled`]: isDisabled,
|
|
}
|
|
);
|
|
|
|
if (!label) {
|
|
return "";
|
|
}
|
|
|
|
return (
|
|
<label className={labelWrapperClasses} htmlFor={name}>
|
|
{error || label}
|
|
</label>
|
|
);
|
|
};
|
|
|
|
return (
|
|
<FormField
|
|
name={name}
|
|
label={renderLabel()}
|
|
helpText={helpText}
|
|
type="dropdown"
|
|
className={wrapperClassNames}
|
|
>
|
|
<Select<CustomOptionType, false>
|
|
classNamePrefix="react-select"
|
|
isSearchable={isSearchable}
|
|
styles={generateCustomDropdownStyles(variant, isDisabled, nowrapMenu)}
|
|
options={options}
|
|
components={{
|
|
Option: CustomOption,
|
|
DropdownIndicator: CustomDropdownIndicator,
|
|
IndicatorSeparator: () => null,
|
|
ValueContainer,
|
|
}}
|
|
value={getCurrentValue()}
|
|
onChange={handleChange}
|
|
isDisabled={isDisabled}
|
|
noOptionsMessage={() => customNoOptionsMessage ?? "No results found"}
|
|
tabIndex={isDisabled ? -1 : 0} // Ensures disabled dropdown has no keyboard accessibility
|
|
placeholder={placeholder}
|
|
onMenuOpen={onMenuOpen}
|
|
controlShouldRenderValue={variant !== "button"} // Control doesn't change placeholder to selected value
|
|
/>
|
|
</FormField>
|
|
);
|
|
};
|
|
|
|
export default DropdownWrapper;
|