From 1e7acc35fc7840c5fc321be690c55ffe449d2d94 Mon Sep 17 00:00:00 2001 From: adamviktora Date: Wed, 27 Sep 2023 14:23:05 +0200 Subject: [PATCH] docs(custom menu): add example with search input inline filtering --- .../src/demos/CustomMenus/CustomMenus.md | 6 ++ .../CustomMenus/examples/FilterMenuDemo.tsx | 102 ++++++++++++++++++ 2 files changed, 108 insertions(+) create mode 100644 packages/react-core/src/demos/CustomMenus/examples/FilterMenuDemo.tsx diff --git a/packages/react-core/src/demos/CustomMenus/CustomMenus.md b/packages/react-core/src/demos/CustomMenus/CustomMenus.md index ca5f8642dfd..d79aff84b28 100644 --- a/packages/react-core/src/demos/CustomMenus/CustomMenus.md +++ b/packages/react-core/src/demos/CustomMenus/CustomMenus.md @@ -49,6 +49,12 @@ Additionally, menu components may be connected to each other manually through ou ``` +### With search input filtering + +```ts file="./examples/FilterMenuDemo.tsx" + +``` + ### Tree view menu When rendering a menu-like element that does not contain `` components, [panel](/components/panel) allows more flexible control and customization. diff --git a/packages/react-core/src/demos/CustomMenus/examples/FilterMenuDemo.tsx b/packages/react-core/src/demos/CustomMenus/examples/FilterMenuDemo.tsx new file mode 100644 index 00000000000..3f243dffc21 --- /dev/null +++ b/packages/react-core/src/demos/CustomMenus/examples/FilterMenuDemo.tsx @@ -0,0 +1,102 @@ +import React from 'react'; +import { + Menu, + MenuList, + MenuContent, + MenuSearch, + MenuSearchInput, + Divider, + SearchInput, + SelectOption, + MenuToggle, + MenuContainer +} from '@patternfly/react-core'; + +export const FilterMenuDemo: React.FunctionComponent = () => { + const [activeItem, setActiveItem] = React.useState(0); + const [input, setInput] = React.useState(''); + const [isOpen, setIsOpen] = React.useState(false); + const toggleRef = React.useRef(); + const menuRef = React.useRef(); + + const onSelect = (_event: React.MouseEvent | undefined, itemId: number | string | undefined) => { + const item = itemId as number; + // eslint-disable-next-line no-console + console.log(`clicked ${itemId}`); + setActiveItem(item); + }; + + const handleTextInputChange = (value: string) => { + if (!isOpen) { + setIsOpen(true); + } + setInput(value); + }; + + const menuListItemsText = [ + 'Action 1', + 'Action 2', + 'Action 3', + 'My project', + 'OpenShift cluster', + 'Production Ansible', + 'AWS', + 'Azure', + 'My project 2', + 'OpenShift cluster ', + 'Production Ansible 2 ', + 'AWS 2', + 'Azure 2' + ]; + + const menuListItems = menuListItemsText + .filter((item) => !input || item.toLowerCase().includes(input.toString().toLowerCase())) + .map((currentValue, index) => ( + + {currentValue} + + )); + if (input && menuListItems.length === 0) { + menuListItems.push( + + No results found + + ); + } + + const toggle = ( + setIsOpen(!isOpen)} isExpanded={isOpen}> + {isOpen ? 'Expanded' : 'Collapsed'} + + ); + + const menu = ( + + + + handleTextInputChange(value)} + /> + + + + + {menuListItems} + + + ); + + return ( + setIsOpen(isOpen)} + onOpenChangeKeys={['Escape']} + /> + ); +};