Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Support links in collection components #4993

Merged
merged 22 commits into from
Sep 14, 2023
Merged

Support links in collection components #4993

merged 22 commits into from
Sep 14, 2023

Conversation

devongovett
Copy link
Member

@devongovett devongovett commented Aug 30, 2023

Closes #1244, fixes #2595

This adds support for links in all collection components. The <Item> component supports href and other link-related DOM props. When an href is provided, the item becomes a link and is rendered as an <a> element when possible. There are three behaviors for links in collections:

  • In some collections such as ListView and TableView, links behave like our existing onAction prop. In the default checkbox selection mode, that means clicking an item navigates by default, and if you click the checkbox you start selecting. In highlight selection mode, clicking once selects and double clicking navigates. Links are basically implemented as a default action.
  • In some components like tabs, links behave like selection. That means any interaction that would move the selection would trigger a navigation, e.g. using the arrow keys. This would typically be used if you were controlling the selected key based on the url.
  • Finally, in some components like menus, tags, and listbox, links override other interactions. Link items may be mixed with non link items, but links are never selectable. As @dannify says, "if in doubt, links win out". 😄

Components using the aria grid pattern do not use native <a> elements because role="row" is not allowed on an <a> element according to the HTML spec. In this case, we put the link attributes on the regular row element as data attributes, and trigger the link using JavaScript on click. You lose the ability to right click and perform link actions, but until the spec is changed, this is the best we can do. w3c/html-aria#473

To integrate with client side routers, there is a RouterProvider component. This accepts a navigate prop that you provide a function from your router (e.g. react-router, next.js, etc.) to go to a new url. All components that support being links use this provider to do their navigation. If one isn't provided, native browser navigation is performed by simulating a click on the link element. That is also done when modifier keys are pressed. This approach means you only need to provide the router provider in one place at the top of your app, and we don't need to open up the API to pass in custom Link components which comes with a host of potential issues. In the future, RouterProvide may also allow us to implement other features such as automatically determining the selected item (e.g. tab) based on the current URL so you don't need to pass a selectedKey at all.

Open questions

  • In what package should we put RouterProvider?
  • Should we integrate RouterProvider into the Spectrum Provider component?

@rspbot

This comment was marked as outdated.

@rspbot
Copy link

rspbot commented Aug 30, 2023

Copy link
Member

@snowystinger snowystinger left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Will continue reviewing tomorrow

packages/@react-aria/interactions/src/usePress.ts Outdated Show resolved Hide resolved
packages/@react-aria/utils/src/openLink.tsx Show resolved Hide resolved

return (
<RouterProvider navigate={setUrl}>
<Tabs selectedKey={url}>
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

just clarifying
this is controlled, but we don't listen for onSelectionChange, instead, we listen to navigate on a parent component? if someone hooks up an onSelectionChange, is that a big deal? What happens if some aren't links?

snowystinger
snowystinger previously approved these changes Aug 31, 2023
Copy link
Member

@snowystinger snowystinger left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I'm probably just mixing things up, but https://reactspectrum.blob.core.windows.net/reactspectrum/4e6aef8500d9a5a378432495ce9b2de09883112d/storybook/index.html?path=/story/listview-selection--links&args=selectionMode:single&providerSwitcher-express=false
I should be able to do selection and navigation when selectionMode is single and selectionStyle is checkbox?
What I'm seeing is onSelectionChange in action panel, but no visual confirmation of selection, and I can't do the navigation.

In selectionmode single and selectionstyle highlight, if I double click on an item in that same story, then I can get stuck in selection mode. I think this is the combination that is supposed to be broken though?

Can we add a story for Tags with links and onRemove?

I'm happy for all of this to be followup, so approving, I'd like it to be in testing session.

@rspbot
Copy link

rspbot commented Sep 13, 2023

if (linkBehavior === 'selection') {
router.open(ref.current, e);
// Always set selected keys back to what they were so that select and combobox close.
manager.setSelectedKeys(manager.selectedKeys);
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Not totally sure about this because it'll also result in onSelectionChange being fired...

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is ok since it is just setting it back to what it was. If we are worried about it firing onSelectionChange even though a link action was fired instead, I don't think it is that bad to keep the dropdown open for select/combobox since the page will be rerouted with a default click. If the user uses command + click then keeping the menu open is similar to the experience if they performed a right click -> open in new tab interaction

@rspbot
Copy link

rspbot commented Sep 13, 2023

@devongovett devongovett mentioned this pull request Sep 13, 2023
Copy link
Member

@reidbarber reidbarber left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Maybe out of scope, but I noticed a slight deviation from native behavior: If you command + enter to open a link in a new tab, it navigates to that new tab, where native links stay on the current tab.

Copy link
Member

@LFDanLu LFDanLu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Still looking through the PR, here are my thoughts on the open questions.

In what package should we put RouterProvider?

Perhaps we should put it in the link package to follow the same precedent we have for I18nProvider (aka it is in @react-aria/i18n)? That being said, the RouterProvider isn't related to the useLink hook at all so I guess it is fine to keep it in @react-aria/utils. Maybe a new package just for general Providers/contexts (not sure if we have/will have any other good candidate to live in a new package though)

Should we integrate RouterProvider into the Spectrum Provider component?

I think we should, otherwise people could forget to add it to their apps and then things would break when they upgrade right?

Copy link
Member

@LFDanLu LFDanLu left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

LGTM, did a brief sweep of the behavior and the logic looks good to me. I see we sometimes are calling the preexisitng event callbacks (onAction, onSelectionChange, etc) upon link navigation (aka highlight selection tables/combobox) but other times we don't (listbox), should we be worried about the inconsistency? Alternatively should we provide a standardized callback when link navigation is triggered via press? Not sure how hard it would be to make things consistent/provide this callback though

Provisional approval pending feedback on the open questions in the PR

if (linkBehavior === 'selection') {
router.open(ref.current, e);
// Always set selected keys back to what they were so that select and combobox close.
manager.setSelectedKeys(manager.selectedKeys);
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think this is ok since it is just setting it back to what it was. If we are worried about it firing onSelectionChange even though a link action was fired instead, I don't think it is that bad to keep the dropdown open for select/combobox since the page will be rerouted with a default click. If the user uses command + click then keeping the menu open is similar to the experience if they performed a right click -> open in new tab interaction

@snowystinger
Copy link
Member

snowystinger commented Sep 14, 2023

Still looking through the PR, here are my thoughts on the open questions.

In what package should we put RouterProvider?

Perhaps we should put it in the link package to follow the same precedent we have for I18nProvider (aka it is in @react-aria/i18n)? That being said, the RouterProvider isn't related to the useLink hook at all so I guess it is fine to keep it in @react-aria/utils. Maybe a new package just for general Providers/contexts (not sure if we have/will have any other good candidate to live in a new package though)

I think @react-aria/utils is a good spot for it as that is our lowest common package, so even if the interactions package one day depends on the RouterProvider context, we'd have access to it. If it's in Link, then we wouldn't.
Otherwise i'd vote for a new package, @react-aria/(navigation|router|...) but not something to combine all the contexts in general, that doesn't seem quite right.

Should we integrate RouterProvider into the Spectrum Provider component?

I think we should, otherwise people could forget to add it to their apps and then things would break when they upgrade right?

That'd increase the number of props people send to our Provider, which is currently about visuals/langauge/etc, it's not about navigation at the moment. I'd prefer a separate provider which is specific to this. It'd also make it easier for people to keep our Provider at the top, but they could handle routing lower down.

// I think this existing (and I'm guessing common) setup would be hard to adapt to the new RouterProvider if it's inside our Provider, they'd need to flip the order if I understood the PR correctly
<Provider>
  <Router>

Copy link
Member

@snowystinger snowystinger left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Can still have approval while we discuss where things will live before release

@devongovett
Copy link
Member Author

I see we sometimes are calling the preexisitng event callbacks (onAction, onSelectionChange, etc) upon link navigation (aka highlight selection tables/combobox) but other times we don't (listbox), should we be worried about the inconsistency?

I think because links override selection in listbox (i.e. link items are not selectable)?

@rspbot
Copy link

rspbot commented Sep 14, 2023

@rspbot
Copy link

rspbot commented Sep 14, 2023

## API Changes

unknown top level export { type: 'identifier', name: 'Column' }
unknown top level export { type: 'identifier', name: 'Column' }
unknown type { type: 'link' }
unknown type { type: 'link' }
unknown type { type: 'link' }
unknown type { type: 'link' }
unknown type { type: 'link' }
unknown type { type: 'link' }

@react-aria/gridlist

AriaGridListOptions

 AriaGridListOptions<T> {
   disabledBehavior?: DisabledBehavior
   isVirtualized?: boolean
   keyboardDelegate?: KeyboardDelegate
+  linkBehavior?: 'action' | 'selection' | 'override' = 'action'
   onAction?: (Key) => void
   shouldFocusWrap?: boolean = false
 }

it changed:

  • useGridList

@react-aria/listbox

AriaListBoxProps

 AriaListBoxOptions<T> {
   isVirtualized?: boolean
   keyboardDelegate?: KeyboardDelegate
+  linkBehavior?: 'action' | 'selection' | 'override' = 'override'
   shouldFocusOnHover?: boolean
   shouldSelectOnPressUp?: boolean
   shouldUseVirtualFocus?: boolean
 }

@react-aria/selection

AriaSelectableCollectionOptions

 AriaSelectableCollectionOptions {
   allowsTabNavigation?: boolean
   autoFocus?: boolean | FocusStrategy = false
   disallowEmptySelection?: boolean = false
   disallowSelectAll?: boolean = false
   disallowTypeAhead?: boolean = false
   isVirtualized?: boolean
   keyboardDelegate: KeyboardDelegate
+  linkBehavior?: 'action' | 'selection' | 'override' = 'action'
   ref: RefObject<HTMLElement>
   scrollRef?: RefObject<HTMLElement>
   selectOnFocus?: boolean = false
   selectionManager: MultipleSelectionManager
   shouldUseVirtualFocus?: boolean
 }
 

it changed:

  • useSelectableCollection

AriaSelectableListOptions

 AriaSelectableListOptions {
-  allowsTabNavigation?: boolean
-  autoFocus?: boolean | FocusStrategy = false
   collection: Collection<Node<unknown>>
   disabledKeys: Set<Key>
-  disallowEmptySelection?: boolean = false
-  disallowTypeAhead?: boolean = false
-  isVirtualized?: boolean
   keyboardDelegate?: KeyboardDelegate
-  ref?: RefObject<HTMLElement>
-  selectOnFocus?: boolean = false
-  selectionManager: MultipleSelectionManager
-  shouldFocusWrap?: boolean = false
-  shouldUseVirtualFocus?: boolean
 }

it changed:

  • useSelectableList

SelectableItemOptions

 SelectableItemOptions {
   allowsDifferentPressOrigin?: boolean
   focus?: () => void
   isDisabled?: boolean
   isVirtualized?: boolean
   key: Key
+  linkBehavior?: 'action' | 'selection' | 'override' | 'none' = 'action'
   onAction?: () => void
   ref: RefObject<FocusableElement>
   selectionManager: MultipleSelectionManager
   shouldSelectOnPressUp?: boolean
 }
 

it changed:

  • useSelectableItem

@react-aria/utils

filterDOMProps

 filterDOMProps {
-  props: (DOMProps & AriaLabelingProps)
+  props: (DOMProps & AriaLabelingProps & LinkDOMProps)
   opts: Options
   returnVal: undefined
 }

useFormReset

-
+openLink {
+  target: HTMLAnchorElement
+  modifiers: Modifiers
+  setOpening: any
+  returnVal: undefined
+}

openLink

-
+getSyntheticLinkProps {
+  props: LinkDOMProps
+  returnVal: undefined
+}

getSyntheticLinkProps

-
+RouterProvider {
+  children: ReactNode
+  navigate: (string) => void
+}

RouterProvider

-
+useRouter {
+  returnVal: undefined
+}

@react-stately/selection

MultipleSelectionManager

 MultipleSelectionManager {
   canSelectItem: (Key) => boolean
   childFocusStrategy: FocusStrategy
   clearSelection: () => void
   disabledBehavior: DisabledBehavior
   disabledKeys: Set<Key>
   disallowEmptySelection?: boolean
   extendSelection: (Key) => void
   firstSelectedKey: Key | null
   focusedKey: Key
   isDisabled: (Key) => boolean
   isEmpty: boolean
   isFocused: boolean
+  isLink: (Key) => boolean
   isSelectAll: boolean
   isSelected: (Key) => boolean
   isSelectionEqual: (Set<Key>) => boolean
   lastSelectedKey: Key | null
   select: (Key, PressEvent | LongPressEvent | PointerEvent) => void
   selectAll: () => void
   selectedKeys: Set<Key>
   selectionBehavior: SelectionBehavior
   selectionMode: SelectionMode
   setFocused: (boolean) => void
   setFocusedKey: (Key, FocusStrategy) => void
   setSelectedKeys: (Iterable<Key>) => void
   setSelectionBehavior: (SelectionBehavior) => void
   toggleSelectAll: () => void
   toggleSelection: (Key) => void
 }
 

SelectionManager

 SelectionManager {
   canSelectItem: (Key) => void
   childFocusStrategy: FocusStrategy
   clearSelection: () => void
   constructor: (Collection<Node<unknown>>, MultipleSelectionState, SelectionManagerOptions) => void
   disabledBehavior: DisabledBehavior
   disabledKeys: Set<Key>
   disallowEmptySelection: boolean
   extendSelection: (Key) => void
   firstSelectedKey: Key | null
   focusedKey: Key
   isDisabled: (Key) => void
   isEmpty: boolean
   isFocused: boolean
+  isLink: (Key) => void
   isSelectAll: boolean
   isSelected: (Key) => void
   isSelectionEqual: (Set<Key>) => void
   lastSelectedKey: Key | null
   replaceSelection: (Key) => void
   select: (Key, PressEvent | LongPressEvent | PointerEvent) => void
   selectAll: () => void
   selectedKeys: Set<Key>
   selectionBehavior: SelectionBehavior
   selectionMode: SelectionMode
   setFocused: (boolean) => void
   setFocusedKey: (Key | null, FocusStrategy) => void
   setSelectedKeys: (Iterable<Key>) => void
   setSelectionBehavior: (SelectionBehavior) => void
   toggleSelectAll: () => void
   toggleSelection: (Key) => void
 }
 

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
5 participants