From 72575ff3e24975336a014b429090553aa8c776c2 Mon Sep 17 00:00:00 2001 From: snmln Date: Wed, 18 Dec 2024 13:30:19 -0500 Subject: [PATCH 1/2] bug correction --- .../nav/create-dynamic-nav-menu-list.tsx | 17 ++++++++++++++--- .../page-header/nav/nav-dropdown-button.tsx | 5 ++++- 2 files changed, 18 insertions(+), 4 deletions(-) diff --git a/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx b/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx index c946c349a..49d5539c7 100644 --- a/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx +++ b/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx @@ -12,6 +12,11 @@ export const createDynamicNavMenuList = ( isOpen?: boolean[], setIsOpen?: SetState ): JSX.Element[] => { + const setDropDownState = () => { + if (isOpen !== undefined && setIsOpen !== undefined) { + return setIsOpen(() => isOpen.fill(false)); + } + }; return navItems.map((item, index) => { switch (item.type) { case NavItemType.DROPDOWN: @@ -23,7 +28,8 @@ export const createDynamicNavMenuList = ( isOpen, setIsOpen, index, - linkProperties + linkProperties, + setDropDownState }} /> ); @@ -32,10 +38,15 @@ export const createDynamicNavMenuList = ( return ; case NavItemType.EXTERNAL_LINK: - return ; + return ( + + ); case NavItemType.ACTION: - return ; + return ; default: return <>; diff --git a/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx b/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx index f76d5f9b9..7d475e0f4 100644 --- a/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx +++ b/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx @@ -12,6 +12,7 @@ interface NavDropDownButtonProps { setIsOpen: SetState; index: number; linkProperties: LinkProperties; + setDropDownState: () => void; } export const NavDropDownButton = ({ @@ -19,7 +20,8 @@ export const NavDropDownButton = ({ isOpen, setIsOpen, index, - linkProperties + linkProperties, + setDropDownState }: NavDropDownButtonProps) => { const onToggle = (index: number, setIsOpen: SetState): void => { setIsOpen((prevIsOpen) => { @@ -47,6 +49,7 @@ export const NavDropDownButton = ({ items={submenuItems} isOpen={isOpen[index]} id={`${item.id}-dropdown`} + onClick={setDropDownState()} /> ); From 5acc803be9cfb27c5e96df70434ec5e062105153 Mon Sep 17 00:00:00 2001 From: snmln Date: Thu, 19 Dec 2024 07:59:57 -0500 Subject: [PATCH 2/2] implemenying custom hook --- .../nav/create-dynamic-nav-menu-list.tsx | 17 +++----------- .../page-header/nav/nav-dropdown-button.tsx | 23 ++++++++++++------- app/scripts/utils/use-click-outside.ts | 15 ++++++++++++ 3 files changed, 33 insertions(+), 22 deletions(-) create mode 100644 app/scripts/utils/use-click-outside.ts diff --git a/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx b/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx index 49d5539c7..c946c349a 100644 --- a/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx +++ b/app/scripts/components/common/page-header/nav/create-dynamic-nav-menu-list.tsx @@ -12,11 +12,6 @@ export const createDynamicNavMenuList = ( isOpen?: boolean[], setIsOpen?: SetState ): JSX.Element[] => { - const setDropDownState = () => { - if (isOpen !== undefined && setIsOpen !== undefined) { - return setIsOpen(() => isOpen.fill(false)); - } - }; return navItems.map((item, index) => { switch (item.type) { case NavItemType.DROPDOWN: @@ -28,8 +23,7 @@ export const createDynamicNavMenuList = ( isOpen, setIsOpen, index, - linkProperties, - setDropDownState + linkProperties }} /> ); @@ -38,15 +32,10 @@ export const createDynamicNavMenuList = ( return ; case NavItemType.EXTERNAL_LINK: - return ( - - ); + return ; case NavItemType.ACTION: - return ; + return ; default: return <>; diff --git a/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx b/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx index 7d475e0f4..5b7f1f5c6 100644 --- a/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx +++ b/app/scripts/components/common/page-header/nav/nav-dropdown-button.tsx @@ -1,10 +1,11 @@ -import React from 'react'; +import React, { useCallback } from 'react'; import { USWDSNavDropDownButton } from '../../uswds/header/nav-drop-down-button'; import { USWDSMenu } from '../../uswds/header/menu'; import { DropdownNavLink } from '../types'; import { createDynamicNavMenuList } from './create-dynamic-nav-menu-list'; import { SetState } from '$types/aliases'; import { LinkProperties } from '$types/veda'; +import { useClickOutside } from '$utils/use-click-outside'; interface NavDropDownButtonProps { item: DropdownNavLink; @@ -12,7 +13,6 @@ interface NavDropDownButtonProps { setIsOpen: SetState; index: number; linkProperties: LinkProperties; - setDropDownState: () => void; } export const NavDropDownButton = ({ @@ -20,8 +20,7 @@ export const NavDropDownButton = ({ isOpen, setIsOpen, index, - linkProperties, - setDropDownState + linkProperties }: NavDropDownButtonProps) => { const onToggle = (index: number, setIsOpen: SetState): void => { setIsOpen((prevIsOpen) => { @@ -34,11 +33,20 @@ export const NavDropDownButton = ({ return newIsOpen; }); }; - + const handleClickOutside = useCallback(() => { + if (isOpen[index]) { + setIsOpen((prevIsOpen) => { + const newIsOpen = [...prevIsOpen]; + newIsOpen[index] = false; + return newIsOpen; + }); + } + }, [index, isOpen, setIsOpen]); + const dropdownRef = useClickOutside(handleClickOutside); const submenuItems = createDynamicNavMenuList(item.children, linkProperties); return ( - +
onToggle(index, setIsOpen)} menuId={item.title} @@ -49,8 +57,7 @@ export const NavDropDownButton = ({ items={submenuItems} isOpen={isOpen[index]} id={`${item.id}-dropdown`} - onClick={setDropDownState()} /> - +
); }; diff --git a/app/scripts/utils/use-click-outside.ts b/app/scripts/utils/use-click-outside.ts new file mode 100644 index 000000000..4c0247903 --- /dev/null +++ b/app/scripts/utils/use-click-outside.ts @@ -0,0 +1,15 @@ +import { useEffect, useRef } from 'react'; +export const useClickOutside = (onClose: () => void) => { + const ref = useRef(null); + useEffect(() => { + const handleClickOutside = (event: MouseEvent) => { + if (ref.current && !ref.current.contains(event.target as Node)) { + onClose(); + } + }; + document.addEventListener('click', handleClickOutside, true); + return () => + document.removeEventListener('click', handleClickOutside, true); + }, [onClose]); + return ref; +}; \ No newline at end of file