From b11091bfee62cd3fc066760707c768ec67faecb3 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Micha=C5=82=20Grabowski?= Date: Thu, 31 Oct 2024 11:56:28 +0100 Subject: [PATCH] improve search; fix linter --- .../modules/common/popup-menu/popup.menu.js | 121 ++++++++++-------- 1 file changed, 68 insertions(+), 53 deletions(-) diff --git a/src/bundle/ui-dev/src/modules/common/popup-menu/popup.menu.js b/src/bundle/ui-dev/src/modules/common/popup-menu/popup.menu.js index 87479271bc..3256a6260c 100644 --- a/src/bundle/ui-dev/src/modules/common/popup-menu/popup.menu.js +++ b/src/bundle/ui-dev/src/modules/common/popup-menu/popup.menu.js @@ -1,23 +1,14 @@ -import React, { useState, useEffect, useRef, useLayoutEffect } from 'react'; -import ReactDOM from 'react-dom'; +import React, { useState, useEffect, useRef, useMemo } from 'react'; import PropTypes from 'prop-types'; -import { getRootDOMElement, getTranslator } from '@ibexa-admin-ui/src/bundle/Resources/public/js/scripts/helpers/context.helper'; +import { getTranslator } from '@ibexa-admin-ui/src/bundle/Resources/public/js/scripts/helpers/context.helper'; +import { createCssClassNames } from '@ibexa-admin-ui/src/bundle/ui-dev/src/modules/common/helpers/css.class.names'; import Icon from '@ibexa-admin-ui/src/bundle/ui-dev/src/modules/common/icon/icon'; -import { createCssClassNames } from '../helpers/css.class.names'; -// import Icon from '../icon/icon'; -// import { getTranslator } from '@ibexa-admin-ui/src/bundle/Resources/public/js/scripts/helpers/context.helper'; - -const { document } = window; const MIN_SEARCH_ITEMS_DEFAULT = 5; const MIN_ITEMS_LIST_HEIGHT = 150; -const ITEMS_LIST_WIDGET_MARGIN = 8; -const ITEMS_LIST_SITE_MARGIN = ITEMS_LIST_WIDGET_MARGIN + 4; -const RESTRICTED_AREA_ITEMS_CONTAINER = 190; -const PopupMenu = ({ positionOffset, items, scrollContainer, referenceElement, onItemClick, footer, extraClasses }) => { - const rootDOMElement = getRootDOMElement(); +const PopupMenu = ({ extraClasses, footer, items, onItemClick, positionOffset, referenceElement, scrollContainer }) => { const Translator = getTranslator(); const containerRef = useRef(); const [isVisible, setIsVisible] = useState(true); @@ -26,6 +17,7 @@ const PopupMenu = ({ positionOffset, items, scrollContainer, referenceElement, o top: 0, }); const [filterText, setFilterText] = useState(''); + const numberOfItems = useMemo(() => items.reduce((sum, group) => sum + group.items.length, 0), [items]); const popupMenuClassName = createCssClassNames({ 'c-popup-menu': true, 'c-popup-menu--visible': isVisible, @@ -45,6 +37,12 @@ const PopupMenu = ({ positionOffset, items, scrollContainer, referenceElement, o return itemLabelLowerCase.indexOf(filterTextLowerCase) === 0; }; const renderGroup = (group) => { + const isAnyItemVisible = group.items.some(showItem); + + if (!isAnyItemVisible) { + return null; + } + const groupClassName = createCssClassNames({ 'c-popup-menu__group': true, }); @@ -85,8 +83,6 @@ const PopupMenu = ({ positionOffset, items, scrollContainer, referenceElement, o } else { const { x: offsetX, y: offsetY } = positionOffset(referenceElement, 'top'); - console.log(offsetY); - itemsStyles.top = referenceTop + offsetY; itemsStyles.left = referenceLeft + offsetX; itemsStyles.maxHeight = referenceTop; @@ -95,35 +91,12 @@ const PopupMenu = ({ positionOffset, items, scrollContainer, referenceElement, o setItemsListStyles(itemsStyles); }; - - useEffect(() => { - calculateAndSetItemsListStyles(); - - if (!isVisible) { - return; + const renderSearch = () => { + if (numberOfItems < MIN_SEARCH_ITEMS_DEFAULT) { + return null; } - const onInteractionOutside = (event) => { - if (containerRef.current.contains(event.target) || referenceElement.contains(event.target)) { - return; - } - - setIsVisible(false); - }; - - window.document.body.addEventListener('click', onInteractionOutside, false); - scrollContainer.addEventListener('scroll', calculateAndSetItemsListStyles, false); - - return () => { - window.document.body.removeEventListener('click', onInteractionOutside); - scrollContainer.removeEventListener('scroll', calculateAndSetItemsListStyles); - - setItemsListStyles({}); - }; - }, [isVisible]); - - return ( -
+ return (
+ ); + }; + const renderFooter = () => { + if (!footer) { + return null; + } + + return
{footer}
; + }; + + useEffect(() => { + calculateAndSetItemsListStyles(); + + if (!isVisible) { + return; + } + + const onInteractionOutside = (event) => { + if (containerRef.current.contains(event.target) || referenceElement.contains(event.target)) { + return; + } + + setIsVisible(false); + }; + + window.document.body.addEventListener('click', onInteractionOutside, false); + scrollContainer.addEventListener('scroll', calculateAndSetItemsListStyles, false); + + return () => { + window.document.body.removeEventListener('click', onInteractionOutside); + scrollContainer.removeEventListener('scroll', calculateAndSetItemsListStyles); + + setItemsListStyles({}); + }; + }, [isVisible]); + + return ( +
+ {renderSearch()}
{items.map(renderGroup)}
- { footer && ( - - ) } + {renderFooter()}
); }; PopupMenu.propTypes = { referenceElement: PropTypes.isRequired, + extraClasses: PropTypes.string, + footer: PropTypes.node, + items: PropTypes.arrayOf({ + items: PropTypes.shapeOf({ + value: PropTypes.oneOf([PropTypes.string, PropTypes.number]), + label: PropTypes.string, + }), + }), + onItemClick: PropTypes.func, positionOffset: PropTypes.func, scrollContainer: PropTypes.node, - onItemClick: PropTypes.func, - footer: PropTypes.node, - extraClasses: PropTypes.string, }; PopupMenu.defaultProps = { + extraClasses: '', + footer: null, + items: [], + onItemClick: () => {}, positionOffset: () => ({ x: 0, y: 0 }), scrollContainer: window.document.body, - onItemClick: () => {}, - footer: null, - extraClasses: '', }; export default PopupMenu;