From 6207def838a71ca8e07c7bdb7c85e7ac29ede821 Mon Sep 17 00:00:00 2001 From: Braden MacDonald Date: Mon, 17 Jun 2024 13:14:43 -0700 Subject: [PATCH 1/5] feat: add typings for hooks --- ...sx => useIndexOfLastVisibleChild.test.tsx} | 2 +- ...{useToggle.test.jsx => useToggle.test.tsx} | 3 +- ...owSize.test.jsx => useWindowSize.test.tsx} | 0 ...vigation.jsx => useArrowKeyNavigation.tsx} | 34 ++++++++++++----- ...ild.jsx => useIndexOfLastVisibleChild.tsx} | 34 +++++++++-------- .../{useIsVisible.jsx => useIsVisible.tsx} | 9 +++-- src/hooks/useToggle.jsx | 37 ------------------ src/hooks/useToggle.tsx | 38 +++++++++++++++++++ .../{useWindowSize.jsx => useWindowSize.tsx} | 9 ++++- src/index.d.ts | 10 ++--- src/index.js | 10 ++--- 11 files changed, 106 insertions(+), 80 deletions(-) rename src/hooks/tests/{useIndexOfLastVisibleChild.test.jsx => useIndexOfLastVisibleChild.test.tsx} (97%) rename src/hooks/tests/{useToggle.test.jsx => useToggle.test.tsx} (96%) rename src/hooks/tests/{useWindowSize.test.jsx => useWindowSize.test.tsx} (100%) rename src/hooks/{useArrowKeyNavigation.jsx => useArrowKeyNavigation.tsx} (69%) rename src/hooks/{useIndexOfLastVisibleChild.jsx => useIndexOfLastVisibleChild.tsx} (70%) rename src/hooks/{useIsVisible.jsx => useIsVisible.tsx} (74%) delete mode 100644 src/hooks/useToggle.jsx create mode 100644 src/hooks/useToggle.tsx rename src/hooks/{useWindowSize.jsx => useWindowSize.tsx} (81%) diff --git a/src/hooks/tests/useIndexOfLastVisibleChild.test.jsx b/src/hooks/tests/useIndexOfLastVisibleChild.test.tsx similarity index 97% rename from src/hooks/tests/useIndexOfLastVisibleChild.test.jsx rename to src/hooks/tests/useIndexOfLastVisibleChild.test.tsx index 9a104f1589..d47377072d 100644 --- a/src/hooks/tests/useIndexOfLastVisibleChild.test.jsx +++ b/src/hooks/tests/useIndexOfLastVisibleChild.test.tsx @@ -12,7 +12,7 @@ window.ResizeObserver = window.ResizeObserver })); function TestComponent() { - const [containerElementRef, setContainerElementRef] = React.useState(null); + const [containerElementRef, setContainerElementRef] = React.useState(null); const overflowElementRef = React.useRef(null); const indexOfLastVisibleChild = useIndexOfLastVisibleChild(containerElementRef, overflowElementRef.current); diff --git a/src/hooks/tests/useToggle.test.jsx b/src/hooks/tests/useToggle.test.tsx similarity index 96% rename from src/hooks/tests/useToggle.test.jsx rename to src/hooks/tests/useToggle.test.tsx index c1fd08fdc5..0c457617f1 100644 --- a/src/hooks/tests/useToggle.test.jsx +++ b/src/hooks/tests/useToggle.test.tsx @@ -4,6 +4,7 @@ import { render, screen } from '@testing-library/react'; import userEvent from '@testing-library/user-event'; import { useToggle } from '../..'; +import { ToggleHandlers } from '../useToggle'; const TOGGLE_IS_ON = 'on'; const TOGGLE_IS_OFF = 'off'; @@ -19,7 +20,7 @@ const resetHandlerMocks = () => { }; // eslint-disable-next-line react/prop-types -function FakeComponent({ defaultIsOn, handlers }) { +function FakeComponent({ defaultIsOn, handlers }: { defaultIsOn: boolean, handlers: ToggleHandlers }) { const [isOn, setOn, setOff, toggle] = useToggle(defaultIsOn, handlers); return ( diff --git a/src/hooks/tests/useWindowSize.test.jsx b/src/hooks/tests/useWindowSize.test.tsx similarity index 100% rename from src/hooks/tests/useWindowSize.test.jsx rename to src/hooks/tests/useWindowSize.test.tsx diff --git a/src/hooks/useArrowKeyNavigation.jsx b/src/hooks/useArrowKeyNavigation.tsx similarity index 69% rename from src/hooks/useArrowKeyNavigation.jsx rename to src/hooks/useArrowKeyNavigation.tsx index bdfe258b2f..862d3f8d14 100644 --- a/src/hooks/useArrowKeyNavigation.jsx +++ b/src/hooks/useArrowKeyNavigation.tsx @@ -4,13 +4,21 @@ import { useRef, useEffect } from 'react'; * A React hook to enable arrow key navigation on a component. */ -function handleEnter({ event, currentIndex, activeElement }) { +function handleEnter( + { event, currentIndex, activeElement }: { event: KeyboardEvent, currentIndex: number, activeElement: HTMLElement }, +) { if (currentIndex === -1) { return; } activeElement.click(); event.preventDefault(); } -function handleArrowKey({ event, currentIndex, availableElements }) { +function handleArrowKey( + { event, currentIndex, availableElements }: { + event: KeyboardEvent, + currentIndex: number, + availableElements: NodeListOf, + }, +) { // If the focus isn't in the container, focus on the first thing if (currentIndex === -1) { availableElements[0].focus(); } @@ -44,7 +52,7 @@ function handleEvents({ ignoredKeys = [], parentNode, selectors = 'a,button,input', -}) { +}: { event: KeyboardEvent, ignoredKeys?: string[], parentNode: HTMLElement | undefined, selectors?: string }) { if (!parentNode) { return; } const { key } = event; @@ -60,7 +68,7 @@ function handleEvents({ if (!parentNode.contains(activeElement)) { return; } // Get the list of elements we're allowed to scroll through - const availableElements = parentNode.querySelectorAll(selectors); + const availableElements = parentNode.querySelectorAll(selectors); // No elements are available to loop through. if (!availableElements.length) { return; } @@ -70,18 +78,24 @@ function handleEvents({ (availableElement) => availableElement === activeElement, ); - if (key === 'Enter') { - handleEnter({ event, currentIndex, activeElement }); + if (key === 'Enter' && activeElement) { + handleEnter({ event, currentIndex, activeElement: activeElement as HTMLElement }); } handleArrowKey({ event, currentIndex, availableElements }); } -export default function useArrowKeyNavigation(props) { - const { selectors, ignoredKeys } = props || {}; - const parentNode = useRef(); +export interface ArrowKeyNavProps { + /** e.g. 'a,button,input' */ + selectors?: string; + ignoredKeys?: string[]; +} + +export default function useArrowKeyNavigation(props: ArrowKeyNavProps = {}) { + const { selectors, ignoredKeys } = props; + const parentNode = useRef(); useEffect(() => { - const eventHandler = (event) => { + const eventHandler = (event: KeyboardEvent) => { handleEvents({ event, ignoredKeys, parentNode: parentNode.current, selectors, }); diff --git a/src/hooks/useIndexOfLastVisibleChild.jsx b/src/hooks/useIndexOfLastVisibleChild.tsx similarity index 70% rename from src/hooks/useIndexOfLastVisibleChild.jsx rename to src/hooks/useIndexOfLastVisibleChild.tsx index 786375cdbf..0428ef97b5 100644 --- a/src/hooks/useIndexOfLastVisibleChild.jsx +++ b/src/hooks/useIndexOfLastVisibleChild.tsx @@ -5,25 +5,32 @@ import { useLayoutEffect, useState } from 'react'; * that fits within its bounding rectangle. This is done by summing the widths * of the children until they exceed the width of the container. * - * @param {Element} containerElementRef - container element - * @param {Element} overflowElementRef - overflow element - * * The hook returns the index of the last visible child. + * + * @param containerElementRef - container element + * @param overflowElementRef - overflow element */ -const useIndexOfLastVisibleChild = (containerElementRef, overflowElementRef) => { +const useIndexOfLastVisibleChild = ( + containerElementRef: Element | null, + overflowElementRef: Element | null, +): number => { const [indexOfLastVisibleChild, setIndexOfLastVisibleChild] = useState(-1); useLayoutEffect(() => { + if (!containerElementRef) { + return undefined; + } + function updateLastVisibleChildIndex() { // Get array of child nodes from NodeList form - const childNodesArr = Array.prototype.slice.call(containerElementRef.children); + const childNodesArr = Array.prototype.slice.call(containerElementRef!.children); const { nextIndexOfLastVisibleChild } = childNodesArr // filter out the overflow element .filter(childNode => childNode !== overflowElementRef) // sum the widths to find the last visible element's index .reduce((acc, childNode, index) => { acc.sumWidth += childNode.getBoundingClientRect().width; - if (acc.sumWidth <= containerElementRef.getBoundingClientRect().width) { + if (acc.sumWidth <= containerElementRef!.getBoundingClientRect().width) { acc.nextIndexOfLastVisibleChild = index; } return acc; @@ -32,23 +39,18 @@ const useIndexOfLastVisibleChild = (containerElementRef, overflowElementRef) => // sometimes we'll show a dropdown with one item in it when it would fit, // but allowing this case dramatically simplifies the calculations we need // to do above. - sumWidth: overflowElementRef ? overflowElementRef.getBoundingClientRect().width : 0, + sumWidth: overflowElementRef?.getBoundingClientRect().width ?? 0, nextIndexOfLastVisibleChild: -1, }); setIndexOfLastVisibleChild(nextIndexOfLastVisibleChild); } - if (containerElementRef) { - updateLastVisibleChildIndex(); - - const resizeObserver = new ResizeObserver(() => updateLastVisibleChildIndex()); - resizeObserver.observe(containerElementRef); - - return () => resizeObserver.disconnect(); - } + updateLastVisibleChildIndex(); - return undefined; + const resizeObserver = new ResizeObserver(() => updateLastVisibleChildIndex()); + resizeObserver.observe(containerElementRef); + return () => resizeObserver.disconnect(); }, [containerElementRef, overflowElementRef]); return indexOfLastVisibleChild; diff --git a/src/hooks/useIsVisible.jsx b/src/hooks/useIsVisible.tsx similarity index 74% rename from src/hooks/useIsVisible.jsx rename to src/hooks/useIsVisible.tsx index c9f661af86..0ac26a9f86 100644 --- a/src/hooks/useIsVisible.jsx +++ b/src/hooks/useIsVisible.tsx @@ -1,7 +1,10 @@ -import { useRef, useState, useEffect } from 'react'; +import React, { useRef, useState, useEffect } from 'react'; -const useIsVisible = (defaultIsVisible = true) => { - const sentinelRef = useRef(); +const useIsVisible = (defaultIsVisible = true): [ + isVisible: boolean, + sentinelRef: React.MutableRefObject, +] => { + const sentinelRef = useRef(null); const [isVisible, setIsVisible] = useState(defaultIsVisible); useEffect(() => { diff --git a/src/hooks/useToggle.jsx b/src/hooks/useToggle.jsx deleted file mode 100644 index 731d50e07d..0000000000 --- a/src/hooks/useToggle.jsx +++ /dev/null @@ -1,37 +0,0 @@ -import { useState, useCallback } from 'react'; - -export default function useToggle(defaultIsOn, handlers = {}) { - const { handleToggleOn, handleToggleOff, handleToggle } = handlers; - const [isOn, setIsOn] = useState(defaultIsOn || false); - - const setOn = useCallback(() => { - setIsOn(true); - // istanbul ignore else - if (handleToggleOn) { - handleToggleOn(); - } - // istanbul ignore else - if (handleToggle) { - handleToggle(true); - } - }, [handleToggleOn, handleToggle]); - - const setOff = useCallback(() => { - setIsOn(false); - // istanbul ignore else - if (handleToggleOff) { - handleToggleOff(); - } - // istanbul ignore else - if (handleToggle) { - handleToggle(false); - } - }, [handleToggleOff, handleToggle]); - - const toggle = useCallback(() => { - const doToggle = isOn ? setOff : setOn; - doToggle(); - }, [isOn, setOn, setOff]); - - return [isOn, setOn, setOff, toggle]; -} diff --git a/src/hooks/useToggle.tsx b/src/hooks/useToggle.tsx new file mode 100644 index 0000000000..20614dcf44 --- /dev/null +++ b/src/hooks/useToggle.tsx @@ -0,0 +1,38 @@ +import { useState, useCallback } from 'react'; + +export type Toggler = [ + isOn: boolean, + setOn: () => void, + setOff: () => void, + toggle: () => void, +]; + +export interface ToggleHandlers { + handleToggleOn?: () => void; + handleToggleOff?: () => void; + handleToggle?: (newStatus: boolean) => void; +} + +export default function useToggle(defaultIsOn = false, handlers: ToggleHandlers = {}): Toggler { + const { handleToggleOn, handleToggleOff, handleToggle } = handlers; + const [isOn, setIsOn] = useState(defaultIsOn); + + const setOn = useCallback(() => { + setIsOn(true); + handleToggleOn?.(); + handleToggle?.(true); + }, [handleToggleOn, handleToggle]); + + const setOff = useCallback(() => { + setIsOn(false); + handleToggleOff?.(); + handleToggle?.(false); + }, [handleToggleOff, handleToggle]); + + const toggle = useCallback(() => { + const doToggle = isOn ? setOff : setOn; + doToggle(); + }, [isOn, setOn, setOff]); + + return [isOn, setOn, setOff, toggle]; +} diff --git a/src/hooks/useWindowSize.jsx b/src/hooks/useWindowSize.tsx similarity index 81% rename from src/hooks/useWindowSize.jsx rename to src/hooks/useWindowSize.tsx index 68229c7b01..6053563e0d 100644 --- a/src/hooks/useWindowSize.jsx +++ b/src/hooks/useWindowSize.tsx @@ -1,9 +1,14 @@ import { useState, useLayoutEffect } from 'react'; -function useWindowSize() { +export interface WindowSizeData { + width: number | undefined; + height: number | undefined; +} + +function useWindowSize(): WindowSizeData { // Initialize state with undefined width/height so server and client renders match // Learn more here: https://joshwcomeau.com/react/the-perils-of-rehydration/ - const [windowSize, setWindowSize] = useState({ + const [windowSize, setWindowSize] = useState({ width: undefined, height: undefined, }); diff --git a/src/index.d.ts b/src/index.d.ts index b72210599c..883fc323e0 100644 --- a/src/index.d.ts +++ b/src/index.d.ts @@ -40,6 +40,11 @@ export { default as ModalLayer } from './Modal/ModalLayer'; export { default as Overlay, OverlayTrigger } from './Overlay'; export { default as Portal } from './Modal/Portal'; export { default as Tooltip } from './Tooltip'; +export { default as useWindowSize, type WindowSizeData } from './hooks/useWindowSize'; +export { default as useToggle, type Toggler, type ToggleHandlers } from './hooks/useToggle'; +export { default as useArrowKeyNavigation, type ArrowKeyNavProps } from './hooks/useArrowKeyNavigation'; +export { default as useIndexOfLastVisibleChild } from './hooks/useIndexOfLastVisibleChild'; +export { default as useIsVisible } from './hooks/useIsVisible'; // // // // // // // // // // // // // // // // // // // // // // // // // // // // Things that don't have types @@ -187,11 +192,6 @@ export const Sticky: any; // from './Sticky'; export const SelectableBox: any; // from './SelectableBox'; export const breakpoints: any; // from './utils/breakpoints'; export const Variant: any; // from './utils/constants'; -export const useWindowSize: any; // from './hooks/useWindowSize'; -export const useToggle: any; // from './hooks/useToggle'; -export const useArrowKeyNavigation: any; // from './hooks/useArrowKeyNavigation'; -export const useIndexOfLastVisibleChild: any; // from './hooks/useIndexOfLastVisibleChild'; -export const useIsVisible: any; // from './hooks/useIsVisible'; export const OverflowScrollContext: any, OverflowScroll: any, diff --git a/src/index.js b/src/index.js index f9d846ad87..7b52e0f891 100644 --- a/src/index.js +++ b/src/index.js @@ -40,6 +40,11 @@ export { default as ModalLayer } from './Modal/ModalLayer'; export { default as Overlay, OverlayTrigger } from './Overlay'; export { default as Portal } from './Modal/Portal'; export { default as Tooltip } from './Tooltip'; +export { default as useWindowSize } from './hooks/useWindowSize'; +export { default as useToggle } from './hooks/useToggle'; +export { default as useArrowKeyNavigation } from './hooks/useArrowKeyNavigation'; +export { default as useIndexOfLastVisibleChild } from './hooks/useIndexOfLastVisibleChild'; +export { default as useIsVisible } from './hooks/useIsVisible'; // // // // // // // // // // // // // // // // // // // // // // // // // // // // Things that don't have types @@ -187,11 +192,6 @@ export { default as Sticky } from './Sticky'; export { default as SelectableBox } from './SelectableBox'; export { default as breakpoints } from './utils/breakpoints'; export { default as Variant } from './utils/constants'; -export { default as useWindowSize } from './hooks/useWindowSize'; -export { default as useToggle } from './hooks/useToggle'; -export { default as useArrowKeyNavigation } from './hooks/useArrowKeyNavigation'; -export { default as useIndexOfLastVisibleChild } from './hooks/useIndexOfLastVisibleChild'; -export { default as useIsVisible } from './hooks/useIsVisible'; export { OverflowScrollContext, OverflowScroll, From 0c637e578e64205b0a57bb474b03848a1c490efb Mon Sep 17 00:00:00 2001 From: Braden MacDonald Date: Mon, 8 Jul 2024 16:18:37 +0200 Subject: [PATCH 2/5] fix: wrong return type specified in www Navbar --- www/src/components/header/Navbar.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/www/src/components/header/Navbar.tsx b/www/src/components/header/Navbar.tsx index ffef619923..1652cb843a 100644 --- a/www/src/components/header/Navbar.tsx +++ b/www/src/components/header/Navbar.tsx @@ -15,7 +15,7 @@ import Search from '../Search'; export interface INavbar { siteTitle: string, - onMenuClick: () => boolean, + onMenuClick: () => void, setTarget: React.Dispatch>, onSettingsClick?: () => void, menuIsOpen?: boolean, From a8882006d3e2750e550c6883ee578d38748c1eb6 Mon Sep 17 00:00:00 2001 From: Braden MacDonald Date: Tue, 9 Jul 2024 10:39:26 +0200 Subject: [PATCH 3/5] fix: useIsVisible() can take any element, not just
--- src/hooks/useIsVisible.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/hooks/useIsVisible.tsx b/src/hooks/useIsVisible.tsx index 0ac26a9f86..314b988c39 100644 --- a/src/hooks/useIsVisible.tsx +++ b/src/hooks/useIsVisible.tsx @@ -2,9 +2,9 @@ import React, { useRef, useState, useEffect } from 'react'; const useIsVisible = (defaultIsVisible = true): [ isVisible: boolean, - sentinelRef: React.MutableRefObject, + sentinelRef: React.MutableRefObject, ] => { - const sentinelRef = useRef(null); + const sentinelRef = useRef(null); const [isVisible, setIsVisible] = useState(defaultIsVisible); useEffect(() => { From 23ee64b177fc70f838d166022780aa0f6655f2c3 Mon Sep 17 00:00:00 2001 From: Braden MacDonald Date: Mon, 16 Dec 2024 13:46:38 -0800 Subject: [PATCH 4/5] fix: gatsby site wasn't building correctly after some .jsx -> .tsx changes --- www/gatsby-config.js | 1 + 1 file changed, 1 insertion(+) diff --git a/www/gatsby-config.js b/www/gatsby-config.js index e3976a0b2c..c0ae5209a2 100644 --- a/www/gatsby-config.js +++ b/www/gatsby-config.js @@ -34,6 +34,7 @@ const plugins = [ }, }, 'gatsby-plugin-react-helmet', + 'gatsby-plugin-typescript', { resolve: 'gatsby-plugin-manifest', options: { From a3fea9e291c5b3fc28ccd4be154d5d5695cd8009 Mon Sep 17 00:00:00 2001 From: Braden MacDonald Date: Mon, 16 Dec 2024 14:00:58 -0800 Subject: [PATCH 5/5] refactor: cleanups suggested by Adam --- .../tests/useIndexOfLastVisibleChild.test.tsx | 2 +- src/hooks/useArrowKeyNavigation.tsx | 38 ++++++++++++------- src/hooks/useIndexOfLastVisibleChild.tsx | 4 +- 3 files changed, 27 insertions(+), 17 deletions(-) diff --git a/src/hooks/tests/useIndexOfLastVisibleChild.test.tsx b/src/hooks/tests/useIndexOfLastVisibleChild.test.tsx index d47377072d..4e4dc7e728 100644 --- a/src/hooks/tests/useIndexOfLastVisibleChild.test.tsx +++ b/src/hooks/tests/useIndexOfLastVisibleChild.test.tsx @@ -13,7 +13,7 @@ window.ResizeObserver = window.ResizeObserver function TestComponent() { const [containerElementRef, setContainerElementRef] = React.useState(null); - const overflowElementRef = React.useRef(null); + const overflowElementRef = React.useRef(null); const indexOfLastVisibleChild = useIndexOfLastVisibleChild(containerElementRef, overflowElementRef.current); return ( diff --git a/src/hooks/useArrowKeyNavigation.tsx b/src/hooks/useArrowKeyNavigation.tsx index 862d3f8d14..1700bbafbc 100644 --- a/src/hooks/useArrowKeyNavigation.tsx +++ b/src/hooks/useArrowKeyNavigation.tsx @@ -1,24 +1,24 @@ import { useRef, useEffect } from 'react'; -/** - * A React hook to enable arrow key navigation on a component. - */ +interface HandleEnterArgs { + event: KeyboardEvent; + currentIndex: number; + activeElement: HTMLElement; +} -function handleEnter( - { event, currentIndex, activeElement }: { event: KeyboardEvent, currentIndex: number, activeElement: HTMLElement }, -) { +function handleEnter({ event, currentIndex, activeElement }: HandleEnterArgs) { if (currentIndex === -1) { return; } activeElement.click(); event.preventDefault(); } -function handleArrowKey( - { event, currentIndex, availableElements }: { - event: KeyboardEvent, - currentIndex: number, - availableElements: NodeListOf, - }, -) { +interface HandleArrowKeyArgs { + event: KeyboardEvent; + currentIndex: number; + availableElements: NodeListOf; +} + +function handleArrowKey({ event, currentIndex, availableElements }: HandleArrowKeyArgs) { // If the focus isn't in the container, focus on the first thing if (currentIndex === -1) { availableElements[0].focus(); } @@ -44,6 +44,13 @@ function handleArrowKey( event.preventDefault(); } +interface HandleEventsArgs { + event: KeyboardEvent; + ignoredKeys?: string[]; + parentNode: HTMLElement | undefined; + selectors?: string; +} + /** * Implement arrow key navigation for the given parentNode */ @@ -52,7 +59,7 @@ function handleEvents({ ignoredKeys = [], parentNode, selectors = 'a,button,input', -}: { event: KeyboardEvent, ignoredKeys?: string[], parentNode: HTMLElement | undefined, selectors?: string }) { +}: HandleEventsArgs) { if (!parentNode) { return; } const { key } = event; @@ -90,6 +97,9 @@ export interface ArrowKeyNavProps { ignoredKeys?: string[]; } +/** + * A React hook to enable arrow key navigation on a component. + */ export default function useArrowKeyNavigation(props: ArrowKeyNavProps = {}) { const { selectors, ignoredKeys } = props; const parentNode = useRef(); diff --git a/src/hooks/useIndexOfLastVisibleChild.tsx b/src/hooks/useIndexOfLastVisibleChild.tsx index 0428ef97b5..2111152981 100644 --- a/src/hooks/useIndexOfLastVisibleChild.tsx +++ b/src/hooks/useIndexOfLastVisibleChild.tsx @@ -11,8 +11,8 @@ import { useLayoutEffect, useState } from 'react'; * @param overflowElementRef - overflow element */ const useIndexOfLastVisibleChild = ( - containerElementRef: Element | null, - overflowElementRef: Element | null, + containerElementRef: HTMLElement | null, + overflowElementRef: HTMLElement | null, ): number => { const [indexOfLastVisibleChild, setIndexOfLastVisibleChild] = useState(-1);