Skip to content

Commit

Permalink
Accessibility engineering review for Link component. (#3317)
Browse files Browse the repository at this point in the history
* Remove as prop from Link.

* Underline Link when inside of a paragraph tag, hover behavior should match focus behavior.

* Create curly-otters-repeat.md

* Soft deprecate as prop for Link so we don't break compatibility.

* Fix issues with Link-dependent components when as prop is not accepted.

* Remove ReactRouterLikeLink function from NavList tests.

* Remove references to as prop from Link docs.

* Update generated/components.json

* Remove reference to as prop in ActionList.LinkItem docs.

* Update generated/components.json

* Add link variant to button to match PVC implementation.

* Update generated/components.json

* Allow up to 4 levels of nesting for the NavList (#3343)

* allows up to 4 levels of nesting for the NavList

* adds changeset

* Update src/NavList/NavList.tsx

Co-authored-by: Cole Bemis <[email protected]>

* updates snapshots

---------

Co-authored-by: Cole Bemis <[email protected]>

* Refactor(ActionList): ActionList.Item should render content as a button if parent is not interactive. (#3284)

* Using button as the underlying tag for ActionList.Item content.

* Fix subnav layout.

* Changing ActionList content to button if the top-level is not a button. ActionMenu is busted.

* Fix issue where ActionList.Item content would render as a button inside ActionMenu, breaking keyboard navigation.

* Create purple-crabs-matter.md

* test(vrt): update snapshots

* Fix bug with MenuContext not being exported properly.

* Fix color violation in ActionList.Item.

* Formatting, update snapshots.

* test(vrt): update snapshots

* Fix a11y violation in ActionList story.

* Formatting.

* Fix ts-ignore comment.

* Adjust line-height when button is rendered.

* Revert "test(vrt): update snapshots"

This reverts commit ba6d863.

* Revert "test(vrt): update snapshots"

This reverts commit 31d5358.

* Update snapshots.

* Remove flexGrow from ActionList.Item to remove extra 1px vertical height.

* Set padding to 0 and put flexGrow back to fix padding issue for ActionList buttons.

* Fix ActionList text wrap story.

* Fix underlinenav story by checking ActionList.Item if as prop is a.

* Update snapshots and formatting.

* Update snapshots.

* If ActionList.Item content is rendered as a button, remove tabIndex from li and fix padding.

* Fix various layout edge cases.

* ts-ignore event handlers on ItemWrapper.

* Update snapshots.

* Revert fontWeight config that differed from production docs.

* Pass selectionVariant prop illegally in UnderlineNav story to fix issue with Selection spans showing up.

* fix: prevent closing menu when `event.preventDefault()` is called on `ActionList.Item`'s `onSelect` handler by @gr2m (#3346)

* fix: prevent closing menu when `event.preventDefault()` is called on `ActionList.Item`'s `onSelect` handler (#3163)

* test: prevent closing menu when `event.preventDefault()` is called on `ActionList.Item`'s `onSelect` handler

Failing test for #3162

* fix: prevent closing menu when `event.preventDefault()` is called on `ActionList.Item`ߴs `onSelect` handler

* add storybook example: Delayed Menu Close

* update docs

* docs: changeset

* Update changelog

---------

Co-authored-by: Siddharth Kshetrapal <[email protected]>

* Update generated/components.json

---------

Co-authored-by: Gregor Martynus <[email protected]>
Co-authored-by: siddharthkp <[email protected]>

* Check for tabIndex value in isTopLevelInteractive.

* Reference styles in updated menuProps.

* Updated snapshots.

* Fix padding setting instead of using values from styles, which are inverted.

* Update snapshots.

---------

Co-authored-by: radglob <[email protected]>
Co-authored-by: Gregor Martynus <[email protected]>
Co-authored-by: siddharthkp <[email protected]>

* Rollback attempt at underlying links inside other bits of text.

* Update src/Button/styles.ts

Co-authored-by: Katie Langerman <[email protected]>

---------

Co-authored-by: radglob <[email protected]>
Co-authored-by: Mike Perrotti <[email protected]>
Co-authored-by: Cole Bemis <[email protected]>
Co-authored-by: Gregor Martynus <[email protected]>
Co-authored-by: siddharthkp <[email protected]>
Co-authored-by: Katie Langerman <[email protected]>
  • Loading branch information
7 people authored Jun 22, 2023
1 parent 44ccc90 commit cc235fd
Show file tree
Hide file tree
Showing 18 changed files with 97 additions and 320 deletions.
6 changes: 6 additions & 0 deletions .changeset/curly-otters-repeat.md
Original file line number Diff line number Diff line change
@@ -0,0 +1,6 @@
---
"@primer/react": minor
---

Link components no longer accept an `as` prop. If you need link-like styling, you should use a different component and style accordingly.
Button has a `link` variant which can be used for this purpose.
4 changes: 0 additions & 4 deletions docs/content/Link.mdx
Original file line number Diff line number Diff line change
Expand Up @@ -10,10 +10,6 @@ import data from '../../src/Link/Link.docs.json'

The Link component styles anchor tags with default hyperlink color cues and hover text decoration. `Link` is used for destinations, or moving from one page to another.

In special cases where you'd like a `<button>` styled like a `Link`, use `<Link as='button'>`. Make sure to provide a click handler with `onClick`.

**Important:** When using the `as` prop, be sure to always render an accessible element type, like `a`, `button`, `input`, or `summary`.

## Examples

```jsx live
Expand Down
14 changes: 2 additions & 12 deletions generated/components.json
Original file line number Diff line number Diff line change
Expand Up @@ -215,11 +215,6 @@
"name": "ref",
"type": "React.RefObject<HTMLAnchorElement>"
},
{
"name": "as",
"type": "React.ElementType",
"defaultValue": "\"a\""
},
{
"name": "sx",
"type": "SystemStyleObject"
Expand Down Expand Up @@ -900,7 +895,7 @@
},
{
"name": "variant",
"type": "| 'default'\n| 'primary'\n| 'danger'\n| 'outline'\n| 'invisible'",
"type": "| 'default'\n| 'primary'\n| 'danger'\n| 'outline'\n| 'invisible'\n| 'link'",
"defaultValue": "'default'",
"description": "Change the visual style of the button."
},
Expand Down Expand Up @@ -2062,7 +2057,7 @@
"name": "href",
"type": "string",
"defaultValue": "",
"description": "URL to be used for the Link. (The `href` is passed to the underlying `<a>` element. If `as` is specified, the link behavior may need different props)."
"description": "URL to be used for the Link. (The `href` is passed to the underlying `<a>` element)."
},
{
"name": "muted",
Expand All @@ -2086,11 +2081,6 @@
"name": "ref",
"type": "React.RefObject<HTMLAnchorElement>"
},
{
"name": "as",
"type": "React.ElementType",
"defaultValue": "\"a\""
},
{
"name": "sx",
"type": "SystemStyleObject"
Expand Down
5 changes: 0 additions & 5 deletions src/ActionList/ActionList.docs.json
Original file line number Diff line number Diff line change
Expand Up @@ -114,11 +114,6 @@
"name": "ref",
"type": "React.RefObject<HTMLAnchorElement>"
},
{
"name": "as",
"type": "React.ElementType",
"defaultValue": "\"a\""
},
{
"name": "sx",
"type": "SystemStyleObject"
Expand Down
14 changes: 5 additions & 9 deletions src/ActionList/LinkItem.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import {ActionListItemProps} from './shared'
type LinkProps = {
download?: string
href?: string
to?: string
hrefLang?: string
media?: string
ping?: string
Expand All @@ -21,7 +22,7 @@ type LinkProps = {
// LinkItem does not support selected, variants, etc.
export type ActionListLinkItemProps = Pick<ActionListItemProps, 'active' | 'children' | 'sx'> & LinkProps

export const LinkItem = React.forwardRef(({sx = {}, active, as: Component, ...props}, forwardedRef) => {
export const LinkItem = React.forwardRef(({sx = {}, active, ...props}, forwardedRef) => {
const styles = {
// occupy full size of Item
paddingX: 2,
Expand All @@ -35,6 +36,8 @@ export const LinkItem = React.forwardRef(({sx = {}, active, as: Component, ...pr
'&:hover': {color: 'inherit', textDecoration: 'none'},
}

if (props.href === undefined && props.to !== undefined) props.href = props.to

return (
<Item
active={active}
Expand All @@ -45,14 +48,7 @@ export const LinkItem = React.forwardRef(({sx = {}, active, as: Component, ...pr
props.onClick && props.onClick(event as React.MouseEvent<HTMLAnchorElement>)
}
return (
<Link
as={Component}
sx={merge(styles, sx as SxProp)}
{...rest}
{...props}
onClick={clickHandler}
ref={forwardedRef}
>
<Link sx={merge(styles, sx as SxProp)} {...rest} {...props} onClick={clickHandler} ref={forwardedRef}>
{children}
</Link>
)
Expand Down
4 changes: 2 additions & 2 deletions src/Button/Button.docs.json
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@
},
{
"name": "variant",
"type": "| 'default'\n| 'primary'\n| 'danger'\n| 'outline'\n| 'invisible'",
"type": "| 'default'\n| 'primary'\n| 'danger'\n| 'outline'\n| 'invisible'\n| 'link'",
"defaultValue": "'default'",
"description": "Change the visual style of the button."
},
Expand Down Expand Up @@ -67,4 +67,4 @@
]
}
]
}
}
2 changes: 1 addition & 1 deletion src/Button/Button.stories.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -22,7 +22,7 @@ export default {
control: {
type: 'radio',
},
options: ['default', 'primary', 'danger', 'invisible', 'outline'],
options: ['default', 'primary', 'danger', 'invisible', 'outline', 'link'],
},
alignContent: {
control: {
Expand Down
19 changes: 19 additions & 0 deletions src/Button/styles.ts
Original file line number Diff line number Diff line change
Expand Up @@ -186,6 +186,25 @@ export const getVariantStyles = (variant: VariantType = 'default', theme?: Theme
borderColor: 'btn.outline.selectedBorder',
},
},
link: {
color: 'accent.fg',
backgroundColor: 'transparent',
display: 'inline-flex',
border: 'none',
height: 'unset',
padding: 0,
'&:hover:not(:disabled)': {
textDecoration: 'underline',
},
'&:focus-visible, &:focus': {
outlineOffset: '2px',
},
'&:disabled, &[aria-disabled=true]': {
color: 'primer.fg.disabled',
backgroundColor: 'transparent',
borderColor: 'transparent',
},
},
}
return style[variant]
}
Expand Down
2 changes: 1 addition & 1 deletion src/Button/types.ts
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@ export const StyledButton = styled.button<SxProp>`
${sx};
`

export type VariantType = 'default' | 'primary' | 'invisible' | 'danger' | 'outline'
export type VariantType = 'default' | 'primary' | 'invisible' | 'danger' | 'outline' | 'link'

export type Size = 'small' | 'medium' | 'large'

Expand Down
9 changes: 2 additions & 7 deletions src/Link/Link.docs.json
Original file line number Diff line number Diff line change
Expand Up @@ -9,7 +9,7 @@
"name": "href",
"type": "string",
"defaultValue": "",
"description": "URL to be used for the Link. (The `href` is passed to the underlying `<a>` element. If `as` is specified, the link behavior may need different props)."
"description": "URL to be used for the Link. (The `href` is passed to the underlying `<a>` element)."
},
{
"name": "muted",
Expand All @@ -33,15 +33,10 @@
"name": "ref",
"type": "React.RefObject<HTMLAnchorElement>"
},
{
"name": "as",
"type": "React.ElementType",
"defaultValue": "\"a\""
},
{
"name": "sx",
"type": "SystemStyleObject"
}
],
"subcomponents": []
}
}
20 changes: 19 additions & 1 deletion src/Link/Link.features.stories.tsx
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
import Link from '../Link'
import {Meta} from '@storybook/react'
import Box from '../Box'
import {Meta, StoryFn} from '@storybook/react'
import React from 'react'
import {ComponentProps} from '../utils/types'

Expand All @@ -19,3 +20,20 @@ export const Underline = () => (
Link
</Link>
)

export const WithinText: StoryFn<typeof Link> = args => (
<Box as="p">
This{' '}
<Link href="#" {...args}>
link
</Link>{' '}
is inside of other text.
</Box>
)

WithinText.args = {
muted: false,
underline: false,
}

WithinText.argTypes = {}
46 changes: 9 additions & 37 deletions src/Link/Link.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import React, {forwardRef, useEffect} from 'react'
import React, {forwardRef} from 'react'
import styled from 'styled-components'
import {system} from 'styled-system'
import {get} from '../constants'
Expand All @@ -23,55 +23,27 @@ const hoverColor = system({
const StyledLink = styled.a<StyledLinkProps>`
color: ${props => (props.muted ? get('colors.fg.muted')(props) : get('colors.accent.fg')(props))};
text-decoration: ${props => (props.underline ? 'underline' : 'none')};
&:hover {
&:hover,
&:focus {
text-decoration: ${props => (props.muted ? 'none' : 'underline')};
${props => (props.hoverColor ? hoverColor : props.muted ? `color: ${get('colors.accent.fg')(props)}` : '')};
}
&:is(button) {
display: inline-block;
padding: 0;
font-size: inherit;
white-space: nowrap;
cursor: pointer;
user-select: none;
background-color: transparent;
border: 0;
appearance: none;
}
${sx};
`

const Link = forwardRef(({as: Component = 'a', ...props}, forwardedRef) => {
const Link = forwardRef(({as, ...props}, forwardedRef) => {
const innerRef = React.useRef<HTMLAnchorElement>(null)
useRefObjectAsForwardedRef(forwardedRef, innerRef)

if (__DEV__) {
/**
* The Linter yells because it thinks this conditionally calls an effect,
* but since this is a compile-time flag and not a runtime conditional
* this is safe, and ensures the entire effect is kept out of prod builds
* shaving precious bytes from the output, and avoiding mounting a noop effect
*/
// eslint-disable-next-line react-hooks/rules-of-hooks
useEffect(() => {
if (
innerRef.current &&
!(innerRef.current instanceof HTMLButtonElement) &&
!(innerRef.current instanceof HTMLAnchorElement)
) {
// eslint-disable-next-line no-console
console.error(
'Error: Found `Link` component that renders an inaccessible element',
innerRef.current,
'Please ensure `Link` always renders as <a> or <button>',
)
}
}, [innerRef])
if (as !== undefined) {
// eslint-disable-next-line no-console
console.warn(
'Links no longer accept an as prop. If you need to style another tag as a link, you should use a different component and apply appropriate styling.',
)
}

return (
<StyledLink
as={Component}
{...props}
// @ts-ignore shh
ref={innerRef}
Expand Down
15 changes: 1 addition & 14 deletions src/Link/__tests__/Link.test.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@ import {render as HTMLRender} from '@testing-library/react'
import {axe} from 'jest-axe'

describe('Link', () => {
behavesAsComponent({Component: Link})
behavesAsComponent({Component: Link, options: {skipAs: true}})

checkExports('Link', {
default: Link,
Expand All @@ -29,24 +29,11 @@ describe('Link', () => {
expect(render(<Link sx={{fontStyle: 'italic'}} />)).toHaveStyleRule('font-style', 'italic')
})

it('applies button styles when rendering a button element', () => {
expect(render(<Link as="button" />)).toMatchSnapshot()
})

it('respects the "muted" prop', () => {
expect(render(<Link muted />)).toMatchSnapshot()
})

it('respects the "sx" prop when "muted" prop is also passed', () => {
expect(render(<Link muted sx={{color: 'fg.onEmphasis'}} />)).toMatchSnapshot()
})

it('logs a warning when trying to render invalid "as" prop', () => {
const consoleSpy = jest.spyOn(global.console, 'error').mockImplementation()

HTMLRender(<Link as="i" />)
expect(consoleSpy).toHaveBeenCalled()

consoleSpy.mockRestore()
})
})
Loading

0 comments on commit cc235fd

Please sign in to comment.