From 9568dc67bf43c4a6995f8c80fbdd14ccc8929c1f Mon Sep 17 00:00:00 2001 From: Josh Black Date: Fri, 26 Apr 2024 17:01:37 -0500 Subject: [PATCH 1/5] refactor(Banner): update region to use a dedicated aria-label --- packages/react/src/Banner/Banner.test.tsx | 34 ++++- packages/react/src/Banner/Banner.tsx | 129 +++++++++--------- .../Banner/__snapshots__/Banner.test.tsx.snap | 2 +- .../react/src/internal/hooks/useMergedRefs.ts | 16 +++ 4 files changed, 111 insertions(+), 70 deletions(-) create mode 100644 packages/react/src/internal/hooks/useMergedRefs.ts diff --git a/packages/react/src/Banner/Banner.test.tsx b/packages/react/src/Banner/Banner.test.tsx index 0da386c115e2..eead7e8ed388 100644 --- a/packages/react/src/Banner/Banner.test.tsx +++ b/packages/react/src/Banner/Banner.test.tsx @@ -25,12 +25,38 @@ describe('Banner', () => { it('should render as a region element', () => { render() - expect(screen.getByRole('region', {name: 'test'})).toBeInTheDocument() + expect(screen.getByRole('region', {name: 'Information'})).toBeInTheDocument() + expect(screen.getByRole('heading', {name: 'test'})).toBeInTheDocument() }) - it('should label the landmark element with the title prop', () => { + it('should label the landmark element with the corresponding variant label text', () => { render() - expect(screen.getByRole('region')).toEqual(screen.getByLabelText('test')) + expect(screen.getByRole('region')).toEqual(screen.getByLabelText('Information')) + }) + + it('should label the landmark element with the label for the critical variant', () => { + render() + expect(screen.getByRole('region')).toEqual(screen.getByLabelText('Critical')) + }) + + it('should label the landmark element with the label for the info variant', () => { + render() + expect(screen.getByRole('region')).toEqual(screen.getByLabelText('Information')) + }) + + it('should label the landmark element with the label for the success variant', () => { + render() + expect(screen.getByRole('region')).toEqual(screen.getByLabelText('Success')) + }) + + it('should label the landmark element with the label for the upsell variant', () => { + render() + expect(screen.getByRole('region')).toEqual(screen.getByLabelText('Recommendation')) + }) + + it('should label the landmark element with the label for the warning variant', () => { + render() + expect(screen.getByRole('region')).toEqual(screen.getByLabelText('Warning')) }) it('should default the title to a h2', () => { @@ -50,7 +76,7 @@ describe('Banner', () => { it('should rendering a description with the `description` prop', () => { render() expect(screen.getByText('test-description')).toBeInTheDocument() - expect(screen.getByRole('region', {name: 'test'})).toContainElement(screen.getByText('test-description')) + expect(screen.getByRole('region', {name: 'Information'})).toContainElement(screen.getByText('test-description')) }) it('should support a primary action', async () => { diff --git a/packages/react/src/Banner/Banner.tsx b/packages/react/src/Banner/Banner.tsx index 13fb6196dde6..2cb0e72c813f 100644 --- a/packages/react/src/Banner/Banner.tsx +++ b/packages/react/src/Banner/Banner.tsx @@ -1,10 +1,11 @@ import cx from 'clsx' -import React, {createContext, useContext, useEffect, useId, useMemo} from 'react' +import React, {useEffect} from 'react' import styled from 'styled-components' import {AlertIcon, InfoIcon, StopIcon, CheckCircleIcon, XIcon} from '@primer/octicons-react' import {Button, IconButton} from '../Button' import {get} from '../constants' import {VisuallyHidden} from '../internal/components/VisuallyHidden' +import {useMergedRefs} from '../internal/hooks/useMergedRefs' type BannerVariant = 'critical' | 'info' | 'success' | 'upsell' | 'warning' @@ -64,74 +65,84 @@ const iconForVariant: Record = { warning: , } +const labels: Record = { + critical: 'Critical', + info: 'Information', + success: 'Success', + upsell: 'Recommendation', + warning: 'Warning', +} + export const Banner = React.forwardRef(function Banner( {children, description, hideTitle, icon, onDismiss, primaryAction, secondaryAction, title, variant = 'info', ...rest}, - ref, + forwardRef, ) { - const titleId = useId() - const value = useMemo(() => { - return { - titleId, - } - }, [titleId]) const dismissible = variant !== 'critical' && onDismiss const hasActions = primaryAction || secondaryAction + const bannerRef = React.useRef(null) + const ref = useMergedRefs(forwardRef, bannerRef) if (__DEV__) { - // Note: __DEV__ will make it so that this hook is consistently called, or - // not called, depending on environment + // This hook is called consistently depending on the environment // eslint-disable-next-line react-hooks/rules-of-hooks useEffect(() => { - const title = document.getElementById(titleId) - if (!title) { + if (title) { + return + } + + const {current: banner} = bannerRef + if (!banner) { + return + } + + const hasTitle = banner.querySelector('[data-banner-title]') + if (!hasTitle) { throw new Error( - 'The Banner component requires a title to be provided as the `title` prop or through `Banner.Title`', + 'Expected a title to be provided to the component with the `title` prop or through `` but no title was found', ) } - }, [titleId]) + }, [title]) } return ( - - - -
{icon && variant === 'info' ? icon : iconForVariant[variant]}
-
-
- {title ? ( - hideTitle ? ( - - {title} - - ) : ( + + +
{icon && variant === 'info' ? icon : iconForVariant[variant]}
+
+
+ {title ? ( + hideTitle ? ( + {title} - ) - ) : null} - {description ? {description} : null} - {children} -
- {hasActions ? : null} + + ) : ( + {title} + ) + ) : null} + {description ? {description} : null} + {children}
- {dismissible ? ( - - ) : null} -
- + {hasActions ? : null} +
+ {dismissible ? ( + + ) : null} + ) }) @@ -342,9 +353,8 @@ export type BannerTitleProps = { export function BannerTitle(props: BannerTitleProps) { const {as: Heading = 'h2', className, children, ...rest} = props - const banner = useBanner() return ( - + {children} ) @@ -399,14 +409,3 @@ export function BannerSecondaryAction({children, className, ...rest}: BannerSeco ) } - -type BannerContextValue = {titleId: string} -const BannerContext = createContext(null) - -function useBanner(): BannerContextValue { - const value = useContext(BannerContext) - if (value) { - return value - } - throw new Error('Component must be used within a component') -} diff --git a/packages/react/src/Banner/__snapshots__/Banner.test.tsx.snap b/packages/react/src/Banner/__snapshots__/Banner.test.tsx.snap index 3a1a9768fbcc..34992a577069 100644 --- a/packages/react/src/Banner/__snapshots__/Banner.test.tsx.snap +++ b/packages/react/src/Banner/__snapshots__/Banner.test.tsx.snap @@ -1,3 +1,3 @@ // Jest Snapshot v1, https://goo.gl/fbAQLP -exports[`Banner should throw an error if no title is provided 1`] = `"The Banner component requires a title to be provided as the \`title\` prop or through \`Banner.Title\`"`; +exports[`Banner should throw an error if no title is provided 1`] = `"Expected a title to be provided to the component with the \`title\` prop or through \`\` but no title was found"`; diff --git a/packages/react/src/internal/hooks/useMergedRefs.ts b/packages/react/src/internal/hooks/useMergedRefs.ts new file mode 100644 index 000000000000..7c67f20fadc3 --- /dev/null +++ b/packages/react/src/internal/hooks/useMergedRefs.ts @@ -0,0 +1,16 @@ +import {useCallback} from 'react' + +export function useMergedRefs( + ...refs: Array | React.ForwardedRef | React.RefCallback> +): React.RefCallback { + return useCallback((instance: T) => { + for (const ref of refs) { + if (typeof ref === 'function') { + ref(instance) + } else if (ref) { + ref.current = instance + } + } + // eslint-disable-next-line react-hooks/exhaustive-deps + }, refs) +} From 6e1c409bf733d7ae504cf7cf1a7727fb02ddeaf9 Mon Sep 17 00:00:00 2001 From: Josh Black Date: Fri, 26 Apr 2024 17:02:55 -0500 Subject: [PATCH 2/5] chore: add changeset, update config to exclude codesandbox --- .changeset/cold-starfishes-shout.md | 5 +++++ .changeset/config.json | 2 +- 2 files changed, 6 insertions(+), 1 deletion(-) create mode 100644 .changeset/cold-starfishes-shout.md diff --git a/.changeset/cold-starfishes-shout.md b/.changeset/cold-starfishes-shout.md new file mode 100644 index 000000000000..ce8c32e1b712 --- /dev/null +++ b/.changeset/cold-starfishes-shout.md @@ -0,0 +1,5 @@ +--- +'@primer/react': minor +--- + +Update Banner to use an explicit aria-label instead of being labelled by Banner title diff --git a/.changeset/config.json b/.changeset/config.json index 7c4d724582b9..d826f57cc65c 100644 --- a/.changeset/config.json +++ b/.changeset/config.json @@ -6,5 +6,5 @@ "access": "public", "baseBranch": "main", "updateInternalDependencies": "patch", - "ignore": ["docs", "example-*"] + "ignore": ["docs", "example-*", "codesandbox"] } From e89bd6a17324e5549b9254cee325a2f9cfa0556e Mon Sep 17 00:00:00 2001 From: Josh Black Date: Mon, 29 Apr 2024 15:42:04 -0500 Subject: [PATCH 3/5] feat: add aria-label to Banner --- packages/react/src/Banner/Banner.docs.json | 5 +++++ packages/react/src/Banner/Banner.test.tsx | 5 +++++ packages/react/src/Banner/Banner.tsx | 22 ++++++++++++++++++++-- 3 files changed, 30 insertions(+), 2 deletions(-) diff --git a/packages/react/src/Banner/Banner.docs.json b/packages/react/src/Banner/Banner.docs.json index a12ab01b9849..fa4570ec9b09 100644 --- a/packages/react/src/Banner/Banner.docs.json +++ b/packages/react/src/Banner/Banner.docs.json @@ -6,6 +6,11 @@ "importPath": "@primer/react/experimental", "stories": [], "props": [ + { + "name": "aria-label", + "type": "string", + "description": "Provide an optional label to override the default name for the Banner landmark region" + }, { "name": "description", "type": "React.ReactNode", diff --git a/packages/react/src/Banner/Banner.test.tsx b/packages/react/src/Banner/Banner.test.tsx index eead7e8ed388..dee0b35ca014 100644 --- a/packages/react/src/Banner/Banner.test.tsx +++ b/packages/react/src/Banner/Banner.test.tsx @@ -59,6 +59,11 @@ describe('Banner', () => { expect(screen.getByRole('region')).toEqual(screen.getByLabelText('Warning')) }) + it('should support the `aria-label` prop to override the default label for the landmark', () => { + render() + expect(screen.getByRole('region')).toHaveAttribute('aria-label', 'test') + }); + it('should default the title to a h2', () => { render() expect(screen.getByRole('heading', {level: 2})).toBeInTheDocument() diff --git a/packages/react/src/Banner/Banner.tsx b/packages/react/src/Banner/Banner.tsx index 2cb0e72c813f..8f61ba1a1d1e 100644 --- a/packages/react/src/Banner/Banner.tsx +++ b/packages/react/src/Banner/Banner.tsx @@ -10,6 +10,12 @@ import {useMergedRefs} from '../internal/hooks/useMergedRefs' type BannerVariant = 'critical' | 'info' | 'success' | 'upsell' | 'warning' export type BannerProps = React.ComponentPropsWithoutRef<'section'> & { + /** + * Provide an optional label to override the default name for the Banner + * landmark region + */ + 'aria-label'?: string + /** * Provide an optional description for the Banner. This should provide * supplemental information about the Banner @@ -74,7 +80,19 @@ const labels: Record = { } export const Banner = React.forwardRef(function Banner( - {children, description, hideTitle, icon, onDismiss, primaryAction, secondaryAction, title, variant = 'info', ...rest}, + { + 'aria-label': label, + children, + description, + hideTitle, + icon, + onDismiss, + primaryAction, + secondaryAction, + title, + variant = 'info', + ...rest + }, forwardRef, ) { const dismissible = variant !== 'critical' && onDismiss @@ -107,7 +125,7 @@ export const Banner = React.forwardRef(function Banner return ( Date: Tue, 30 Apr 2024 11:12:55 -0500 Subject: [PATCH 4/5] Update packages/react/src/Banner/Banner.test.tsx Co-authored-by: Kate Higa <16447748+khiga8@users.noreply.github.com> --- packages/react/src/Banner/Banner.test.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/react/src/Banner/Banner.test.tsx b/packages/react/src/Banner/Banner.test.tsx index dee0b35ca014..592844d6ca4a 100644 --- a/packages/react/src/Banner/Banner.test.tsx +++ b/packages/react/src/Banner/Banner.test.tsx @@ -60,7 +60,7 @@ describe('Banner', () => { }) it('should support the `aria-label` prop to override the default label for the landmark', () => { - render() + render() expect(screen.getByRole('region')).toHaveAttribute('aria-label', 'test') }); From 7e133507482550cdc5f009a9dc24afaa3e2f5709 Mon Sep 17 00:00:00 2001 From: Josh Black Date: Tue, 30 Apr 2024 11:13:02 -0500 Subject: [PATCH 5/5] Update packages/react/src/Banner/Banner.test.tsx Co-authored-by: Kate Higa <16447748+khiga8@users.noreply.github.com> --- packages/react/src/Banner/Banner.test.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/react/src/Banner/Banner.test.tsx b/packages/react/src/Banner/Banner.test.tsx index 592844d6ca4a..88cc75d7ddc2 100644 --- a/packages/react/src/Banner/Banner.test.tsx +++ b/packages/react/src/Banner/Banner.test.tsx @@ -62,7 +62,7 @@ describe('Banner', () => { it('should support the `aria-label` prop to override the default label for the landmark', () => { render() expect(screen.getByRole('region')).toHaveAttribute('aria-label', 'test') - }); + }) it('should default the title to a h2', () => { render()