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

Add close through a child function to dialogs #234

Merged
merged 27 commits into from
Mar 10, 2020

Conversation

snowystinger
Copy link
Member

@snowystinger snowystinger commented Feb 28, 2020

Closes https://jira.corp.adobe.com/browse/RSP-1533 and https://jira.corp.adobe.com/browse/RSP-1548. Additionally changes the logic for isDismissable modal dialogs so that they can be closed by clicking outside the dialog. Non-dismissable modal dialogs cannot be closed by clicking outside the dialog. Popovers and tray should still always be closable via outside click.

✅ Pull Request Checklist:

  • Included link to corresponding Issue.
  • Added/updated unit tests and storybook for this change (for new code or code which already has tests).
  • Filled out test instructions.
  • Updated documentation (if it already exists for this component).
  • Looked at the Accessibility Practices for this feature - Aria Practices

📝 Test Instructions:

Test that the dialog/dialogTrigger stories are all closable via escape, cancel/confirm buttons. Confirm that isDismissable modal dialogs, popover, and tray dialogs can be closed by clicking outside of the dialog. Confirm that normal modal dialogs cannot.

🧢 Your Team:

@codecov
Copy link

codecov bot commented Feb 28, 2020

Codecov Report

❗ No coverage uploaded for pull request base (master@9c47417). Click here to learn what that means.
The diff coverage is 50%.

Impacted file tree graph

@@           Coverage Diff            @@
##             master    #234   +/-   ##
========================================
  Coverage          ?   71.2%           
========================================
  Files             ?     206           
  Lines             ?    3966           
  Branches          ?     844           
========================================
  Hits              ?    2824           
  Misses            ?     570           
  Partials          ?     572
Impacted Files Coverage Δ
packages/@react-spectrum/dialog/src/Dialog.tsx 86.66% <ø> (ø)
packages/@react-spectrum/dialog/src/context.ts 100% <ø> (ø)
...ackages/@react-spectrum/dialog/src/AlertDialog.tsx 66.66% <0%> (ø)
...kages/@react-spectrum/dialog/src/DialogTrigger.tsx 82.75% <66.66%> (ø)

@adobe-bot
Copy link

Build successful! 🎉

@adobe-bot
Copy link

Build successful! 🎉

@adobe-bot
Copy link

Build successful! 🎉

@@ -66,13 +71,13 @@ export function DialogTrigger(props: SpectrumDialogTriggerProps) {
case 'modal':
return (
<Modal isOpen={isOpen} onClose={onClose}>
{content}
{typeof content === 'function' ? content(onClose) : content}
Copy link
Member

Choose a reason for hiding this comment

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

I'm not a fan of asking the people who create modals to write (onClose) => {<content><Button onPress={close}>Close</Button></content>}. It feels like we're asking them to write jsx that isn't just elements.

The first idea I thought of was a modal context to pass the onClose to button children which is also problematic, but it doesn't make the end user jump through any weird hoops.

I'm concerned that this will be a support issue of us having to train everyone to write things this way.

Copy link
Member Author

Choose a reason for hiding this comment

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

This is a pretty common React pattern called 'Function as children' so it shouldn't be hard to understand.
This should also be documentable via our prop tables.
There isn't really a great way to do this via context without creating <DialogButton or something equivalent that can pick up a different context than the slots and knows which of the three (four) possible buttons it is.

We went through many options before arriving at this one. I think Devon might have it written down.

expect(onCancelSpy).toHaveBeenCalledWith();
});

it('renders alert dialog with onConfirm / onCancel', function () {
Copy link
Member

Choose a reason for hiding this comment

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

this has the same test description as the previous test, renders alert dialog with onConfirm / onCancel.

Also, why aren't we checking what onConfirmSpy was called with in the previous test and onCancelSpy in this test so we do see that things are different?

@adobe-bot
Copy link

Build successful! 🎉

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.

Just a single question, otherwise just waiting for the removal of the dupe tests that Kyle noted

@adobe-bot
Copy link

Build successful! 🎉

@adobe-bot
Copy link

Build successful! 🎉

LFDanLu added 2 commits March 5, 2020 16:01
Popover, tray, tooltip, etc should all still be dismissible by outside interaction click
@adobe-bot
Copy link

Build successful! 🎉

@adobe-bot
Copy link

Build successful! 🎉

function Test({defaultOpen, onOpenChange}) {
return (
<Provider theme={theme}>
<DialogTrigger type="popover" defaultOpen={defaultOpen} onOpenChange={onOpenChange}>
Copy link
Member

Choose a reason for hiding this comment

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

Shouldn't this example have isDismissable on it?

Copy link
Member

Choose a reason for hiding this comment

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

Oh ya lemme add isDismissable=false here

@adobe-bot
Copy link

Build successful! 🎉

</Modal>
);
case 'tray':
return (
<Tray isOpen={isOpen} onClose={onClose}>
{content}
{typeof content === 'function' ? content(onClose) : content}
Copy link
Member Author

Choose a reason for hiding this comment

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

hmm... thought tray's couldn't have the buttons? I'm fine to just have it all done the same though, seems useful

@adobe-bot
Copy link

Build successful! 🎉

ktabors
ktabors previously approved these changes Mar 9, 2020
Copy link
Member

@ktabors ktabors left a comment

Choose a reason for hiding this comment

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

lgtm

MidnightCoder06
MidnightCoder06 previously approved these changes Mar 9, 2020
Copy link
Member

@devongovett devongovett left a comment

Choose a reason for hiding this comment

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

Can we handle this automatically for AlertDialog via context so you don't need to wrap it in a close function yourself?

@@ -27,6 +29,18 @@ export interface SpectrumDialogTriggerProps extends PositionProps {
isDismissable?: boolean
}

export interface SpectrumDialogTriggerBase {
Copy link
Member

Choose a reason for hiding this comment

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

Move this into react-spectrum. It isn't a public component.

isOpen?: boolean
isOpen?: boolean,
// Whether to close overlay if underlay is clicked
closeOnInteractOutside?: boolean
Copy link
Member

Choose a reason for hiding this comment

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

Can we call this isDismissible and default it to false for consistency?

@LFDanLu LFDanLu dismissed stale reviews from MidnightCoder06 and ktabors via 528bb83 March 9, 2020 23:11
@adobe-bot
Copy link

Build successful! 🎉

@adobe-bot
Copy link

Build successful! 🎉

@devongovett devongovett merged commit b42a6d5 into master Mar 10, 2020
@devongovett devongovett deleted the function-as-a-child-dialog-triggers branch March 10, 2020 02:28
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants