import React from 'react'; import { TextInputGroup, TextInputGroupMain, TextInputGroupUtilities, Button, Menu, MenuContent, MenuList, MenuItem, Popper, Chip, ChipGroup, Divider } from '@patternfly/react-core'; import SearchIcon from '@patternfly/react-icons/dist/esm/icons/search-icon'; import TimesIcon from '@patternfly/react-icons/dist/esm/icons/times-icon'; export const AttributeValueFiltering: React.FunctionComponent = () => { const [inputValue, setInputValue] = React.useState(''); const [selectedKey, setSelectedKey] = React.useState(''); const [menuIsOpen, setMenuIsOpen] = React.useState(false); const [currentChips, setCurrentChips] = React.useState([]); interface attributeValueData { [attribute: string]: string[]; } /** key and value data to be shown in the menu */ const data: attributeValueData = { Cluster: ['acmeqe-managed-1', 'local-cluster'], Kind: ['Template', 'ReplicationController', 'ReplicaSet', 'Deployment'], Label: ['release', 'environment', 'partition'], Name: ['backup-1', 'backup-2', 'production-1', 'production-2', 'testing'], Namespace: ['default', 'public'], Status: ['running', 'idle', 'stopped'] }; const keyNames = ['Cluster', 'Kind', 'Label', 'Name', 'Namespace', 'Status']; const [menuItemsText, setMenuItemsText] = React.useState(keyNames); const [menuItems, setMenuItems] = React.useState([]); /** refs used to detect when clicks occur inside vs outside of the textInputGroup and menu popper */ const menuRef = React.useRef(); const textInputGroupRef = React.useRef(); /** callback for updating the inputValue state in this component so that the input can be controlled */ const handleInputChange = (_event: React.FormEvent, value: string) => { setInputValue(value); }; /** callback for removing a chip from the chip selections */ const deleteChip = (chipToDelete: string) => { const newChips = currentChips.filter((chip) => !Object.is(chip, chipToDelete)); setCurrentChips(newChips); }; /** reset state hooks associated with key selection */ const clearSelectedKey = () => { setInputValue(''); setSelectedKey(''); setMenuItemsText(keyNames); }; /** callback for clearing all selected chips, the text input, and any selected keys */ const clearChipsAndInput = () => { setCurrentChips([]); clearSelectedKey(); }; React.useEffect(() => { /** in the menu only show items that include the text in the input */ const filteredMenuItems = menuItemsText .filter( (item) => !inputValue || item.toLowerCase().includes( inputValue .toString() .slice(selectedKey.length && selectedKey.length + 2) .toLowerCase() ) ) .map((currentValue, index) => ( {currentValue} )); /** in the menu show a disabled "no result" when all menu items are filtered out */ if (filteredMenuItems.length === 0) { const noResultItem = ( No results found ); setMenuItems([noResultItem]); return; } /** determine the menu heading text based on key selection; or lack thereof */ const headingItem = ( {selectedKey.length ? `${selectedKey} values` : 'Attributes'} ); const divider = ; setMenuItems([headingItem, divider, ...filteredMenuItems]); }, [inputValue]); /** add selected key/value pair as a chip in the chip group */ const selectValue = (selectedValue: string) => { setCurrentChips([...currentChips, `${selectedKey}: ${selectedValue}`]); clearSelectedKey(); }; /** update the input to show the selected key and the menu to show the values associated with that specific key */ const selectKey = (selectedText: string) => { setInputValue(`${selectedText}: `); setSelectedKey(selectedText); setMenuItemsText(data[selectedText]); }; const handleEnter = () => { /** do nothing if the menu contains no real results */ if (menuItems.length === 1) { return; } /** perform the appropriate action based on key selection state */ if (selectedKey.length) { selectValue(menuItems[2].props.children); } else { selectKey(menuItems[2].props.children); } }; /** allow the user to backspace at the selected key name to drop the currently selected key */ const handleBackspace = () => { if (selectedKey.length && inputValue === `${selectedKey}: `) { clearSelectedKey(); } }; /** allow the user to select a key by simply typing it and entering a colon, exact (case sensitive) matches only */ const handleColon = () => { if (!selectedKey.length && keyNames.includes(inputValue)) { selectKey(inputValue); event.preventDefault(); } }; /** allow the user to focus on the menu and navigate using the arrow keys */ const handleArrowKey = () => { if (menuRef.current) { const firstElement = menuRef.current.querySelector('li > button:not(:disabled)'); firstElement && firstElement.focus(); } }; /** enable keyboard only usage */ const handleTextInputKeyDown = (event: React.KeyboardEvent) => { switch (event.key) { case 'Enter': handleEnter(); break; case 'Escape': clearSelectedKey(); break; case 'Backspace': handleBackspace(); break; case ':': handleColon(); break; case 'ArrowUp': case 'ArrowDown': handleArrowKey(); break; } }; /** perform the proper key or value selection when a menu item is selected */ const onSelect = (event: React.MouseEvent, _itemId: string | number) => { const selectedText = (event.target as HTMLElement).innerText; if (selectedKey.length) { selectValue(selectedText); } else { selectKey(selectedText); } event.stopPropagation(); textInputGroupRef.current.querySelector('input').focus(); }; /** close the menu when a click occurs outside of the menu or text input group */ const handleClick = (event: MouseEvent) => { if ( menuRef.current && !menuRef.current.contains(event.target as HTMLElement) && !textInputGroupRef.current.contains(event.target as HTMLElement) ) { setMenuIsOpen(false); } }; /** show the search icon only when there are no chips to prevent the chips from being displayed behind the icon */ const showSearchIcon = !currentChips.length; /** only show the clear button when there is something that can be cleared */ const showClearButton = !!inputValue || !!currentChips.length; /** render the utilities component only when a component it contains is being rendered */ const showUtilities = showClearButton; const inputGroup = (
} value={inputValue} onChange={handleInputChange} onFocus={() => setMenuIsOpen(true)} onKeyDown={handleTextInputKeyDown} > {currentChips.map((currentChip) => ( deleteChip(currentChip)}> {currentChip} ))} {showUtilities && ( {showClearButton && ( )} )}
); const menu = (
{menuItems}
); return ( ); };