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

[release/1.x] Close dateinput popover #2093

Merged
merged 10 commits into from
Feb 15, 2018
Merged

[release/1.x] Close dateinput popover #2093

merged 10 commits into from
Feb 15, 2018

Conversation

mud
Copy link
Contributor

@mud mud commented Feb 6, 2018

Changes proposed in this pull request:

This PR enables the dateinput popover to be closed:

  • after tabbing out of the last tabbable element inside the date popover
  • clicking ESC key

Currently, shift tabbing out of the dateinput will close the date popover, however, tabbing through the popover elements do not close the popover after tabbing out of the last tabbable element. ESC also does not have any effect.

Reviewers should focus on:

Is there a better way to implement this feature?

Screenshot

feb-05-2018 21-41-13

Copy link
Contributor

@giladgray giladgray left a comment

Choose a reason for hiding this comment

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

thanks @mud! looks pretty good, reasonable approach.

docs-app preview from 031cfdf

public render() {
const { value, valueString } = this.state;
const dateString = this.state.isInputFocused ? valueString : this.getDateString(value);
const date = this.state.isInputFocused ? this.createMoment(valueString) : value;
const dateValue = this.isMomentValidAndInRange(value) ? fromMomentToDate(value) : null;
const dayPickerProps = {
...this.props.dayPickerProps,
onMonthChange: () => setTimeout(this.registerPopoverBlurHandler, 0),
Copy link
Contributor

Choose a reason for hiding this comment

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

why the timeout here? leave a code comment explaining why it's necessary

Copy link
Contributor

Choose a reason for hiding this comment

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

also, should safeInvoke(this.props.dayPickerProps.onMonthChange) in here, or suddenly that one prop is not supported.

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • add why timeout is required as comment
  • safeInvoke(this.props.dayPickerProps.onMonthChange)

@@ -198,15 +200,29 @@ export class DateInput extends AbstractComponent<IDateInputProps, IDateInputStat
};
}

public componentWillUnmount() {
this.unregisterPopoverBlurHandler();
super.componentWillUnmount();
Copy link
Contributor

Choose a reason for hiding this comment

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

call super first

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • call super first

@@ -63,6 +63,31 @@ describe("<DateInput>", () => {
assert.isFalse(wrapper.find(Popover).prop("isOpen"));
});

it("Popover closes when ESC key pressed", () => {
Copy link
Contributor

Choose a reason for hiding this comment

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

Popover provides this feature, and has corresponding tests. can we leverage it instead of reimplementing/testing here?
https://github.com/palantir/blueprint/blob/develop/packages/core/test/popover/popoverTests.tsx#L565

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • use popover's close implementation, remove this test

Copy link
Contributor Author

Choose a reason for hiding this comment

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

As mentioned up top, we still need the functionality for hitting ESC while the input is in focus. So I still think we need this.

const node = ReactDOM.findDOMNode(this);
const tabbableElements = node.querySelectorAll("[tabindex]:not([tabindex='-1'])");
const numOfElements = tabbableElements.length;
if (numOfElements) {
Copy link
Contributor

Choose a reason for hiding this comment

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

avoid boolean coercion. tabbableElements.length > 0

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • tabbableElements.length > 0

}
};

private handlePopoverBlur = () => {
Copy link
Contributor

Choose a reason for hiding this comment

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

no need for this method with optional arg in handleClosePopover, just use handleClosePopover directly

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • remove and call handleClosePopover

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I had to do something like: (this.handleClosePopover as any) as EventListener which doesn't look great. Is removing the extra method still more preferable?

@@ -447,10 +464,37 @@ export class DateInput extends AbstractComponent<IDateInputProps, IDateInputStat
// the page. tabbing forward should *not* close the popover, because
// focus will be moving into the popover itself.
this.setState({ isOpen: false });
} else if (e.which === Keys.ESCAPE) {
Copy link
Contributor

Choose a reason for hiding this comment

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

popovers have a feature to closeOnEscapeKey... can we leverage that instead?

try adding an onInteraction handler to the Popover.

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • use closeOnEscapeKey and try onInteraction on the popover

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Looks like without additional work, pressing ESC while focus is within the popover, the popover will close.

However, when the focus is on the input we still need to handle ESC and set the popover to close.

}
this.safeInvokeInputProp("onKeyDown", e);
};

private registerPopoverBlurHandler = () => {
const node = ReactDOM.findDOMNode(this);
Copy link
Contributor

Choose a reason for hiding this comment

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

i'm not a huge fan of using findDOMNode, tend to prefer refs. but this is alright.

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • maybe use refs?

Copy link
Contributor Author

Choose a reason for hiding this comment

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

I just wrapped the popover content since ReactDayPicker.Props doesn't have a prop to expose ref.

}
this.safeInvokeInputProp("onKeyDown", e);
};

private registerPopoverBlurHandler = () => {
const node = ReactDOM.findDOMNode(this);
const tabbableElements = node.querySelectorAll("[tabindex]:not([tabindex='-1'])");
Copy link
Contributor

Choose a reason for hiding this comment

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

any sense in supporting other focusable things like <button> or are we safe cuz the calendar contents are well-defined and only use tabIndex? (hoping we're safe... this is a rabbit hole)

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • check to see if we can reasonably support other focusable components

Copy link
Contributor Author

Choose a reason for hiding this comment

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

Actually, we will need to if we also want to support datetime.

@mud
Copy link
Contributor Author

mud commented Feb 6, 2018

Sweet, thanks for the CR @giladgray. Will work on your comments.

// for the updated month to be rendered
onMonthChange: (month: Date) =>
setTimeout(() => {
Utils.safeInvoke(this.props.dayPickerProps.onMonthChange, month);
Copy link
Contributor

Choose a reason for hiding this comment

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

but this shouldn't be in the timeout, it should be called synchronously.

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • move safeInvoke out of setTimeout

onMonthChange: () => setTimeout(this.registerPopoverBlurHandler, 0),
// dom elements for the updated month is not available when
// onMonthChange is called. setTimeout is necessary to wait
// for the updated month to be rendered
Copy link
Contributor

Choose a reason for hiding this comment

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

👌

// onMonthChange is called. setTimeout is necessary to wait
// for the updated month to be rendered
onMonthChange: (month: Date) =>
setTimeout(() => {
Copy link
Contributor

Choose a reason for hiding this comment

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

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • use this.setTimeout remove 0

@@ -232,6 +239,7 @@ export class DateInput extends AbstractComponent<IDateInputProps, IDateInputStat
timePickerProps={{ ...this.props.timePickerProps, precision: this.props.timePrecision }}
/>
);
const wrappedPopoverContent = <div ref={this.setContentRef}>{popoverContent}</div>;
Copy link
Contributor

Choose a reason for hiding this comment

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

another wrapper element 😢
guess that's how it is.

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • acknowledge sadness 😿

Copy link
Contributor

Choose a reason for hiding this comment

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

😂 feeling better already

this.unregisterPopoverBlurHandler();
this.lastPopoverElement = lastPopoverElement;
this.lastPopoverElement.addEventListener("blur", this.handlePopoverBlur);
if (this.contentRef) {
Copy link
Contributor

Choose a reason for hiding this comment

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

avoid boolean coercion. != null

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • this.contentRef != null

}
}
};

private unregisterPopoverBlurHandler = () => {
if (this.lastPopoverElement) {
this.lastPopoverElement.removeEventListener("blur", this.handlePopoverBlur);
this.lastPopoverElement.removeEventListener("blur", (this.handleClosePopover as any) as EventListener);
Copy link
Contributor

Choose a reason for hiding this comment

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

oh whoa well if the alternative is an as any then just do the handlePopoverBlur, but put it all on one line with a comment.

// blur DOM event listener (not React event)
private handlePopoverBlur = () => this.handleClosePopover();

Copy link
Contributor Author

@mud mud Feb 6, 2018

Choose a reason for hiding this comment

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

  • use snippet above

}, 0),
onMonthChange: (month: Date) => {
Utils.safeInvoke(this.props.dayPickerProps.onMonthChange, month);
this.setTimeout(this.registerPopoverBlurHandler);
Copy link
Contributor

Choose a reason for hiding this comment

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

🔥

@@ -232,6 +239,7 @@ export class DateInput extends AbstractComponent<IDateInputProps, IDateInputStat
timePickerProps={{ ...this.props.timePickerProps, precision: this.props.timePrecision }}
/>
);
const wrappedPopoverContent = <div ref={this.setContentRef}>{popoverContent}</div>;
Copy link
Contributor

Choose a reason for hiding this comment

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

😂 feeling better already

@adidahiya
Copy link
Contributor

one min -- taking a look now

if (this.contentRef != null) {
// Popover contents are well structured, but the selector will need
// to be updated if more focusable components are added in the future
const tabbableElements = this.contentRef.querySelectorAll("input, [tabindex]:not([tabindex='-1'])");
Copy link
Contributor

Choose a reason for hiding this comment

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

note to self / @giladgray: we'll need to update this for dom4 v2

Copy link
Contributor

Choose a reason for hiding this comment

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

this is already updated. queryAll was removed.

const tabbableElements = this.contentRef.querySelectorAll("input, [tabindex]:not([tabindex='-1'])");
const numOfElements = tabbableElements.length;
if (numOfElements > 0) {
const lastPopoverElement = tabbableElements[numOfElements - 1] as HTMLElement;
Copy link
Contributor

Choose a reason for hiding this comment

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

why do we need to do all this (keep track of the last popover element and juggle the event handlers)? please leave comments

Copy link
Contributor

Choose a reason for hiding this comment

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

at first I thought this was the last popover element on the page, but it's actually more like lastElementInPopover

Copy link
Contributor Author

@mud mud Feb 7, 2018

Choose a reason for hiding this comment

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

  • Add comment on why we keep track of lastPopoverElement
  • rename lastPopoverElement -> lastElementInPopover

};

private unregisterPopoverBlurHandler = () => {
if (this.lastPopoverElement) {
Copy link
Contributor

Choose a reason for hiding this comment

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

if (this.lastPopoverElement != null)

Copy link
Contributor Author

@mud mud Feb 7, 2018

Choose a reason for hiding this comment

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

  • fix

if (this.lastPopoverElement !== lastPopoverElement) {
this.unregisterPopoverBlurHandler();
this.lastPopoverElement = lastPopoverElement;
this.lastPopoverElement.addEventListener("blur", this.handlePopoverBlur);
Copy link
Contributor

Choose a reason for hiding this comment

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

is this right? what if you TAB to the last element and then SHIFT+TAB to go backwards?

Copy link
Contributor Author

@mud mud Feb 7, 2018

Choose a reason for hiding this comment

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

Actually, if you SHIFT+TAB on the last element, it will close. Will fix.

  • Handle SHIFT+TAB and make sure it focuses previous element.

@adidahiya
Copy link
Contributor

sorry, I meant to hint at the more general case -- blurring the last tabbable element in the popover can happen in a number of ways, and only one of them should close the popover through this new code you're adding (TAB forward). another case we should make sure to test is focussing that lastElementInPopover and then clicking somewhere inside the popover like the month dropdown menu -- this blurs the last element, but shouldn't close the popover.

// keyboard DOM event listener (not React event)
private handlePopoverBlur = (e: KeyboardEvent) => {
if (e.which === Keys.TAB && !e.shiftKey) {
e.target.dispatchEvent(new FocusEvent("blur"));
Copy link
Contributor

Choose a reason for hiding this comment

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

why? this is suspicious. we're not in the habit of dispatching DOM events from blueprint code (except where absolutely necessary in tests).

// Keep track of the last focusable element in popover and add
// a keydown handler, so that:
// * popover closes when the user tabs to the next element
// * or focus moves to previous element if shift+tab
Copy link
Contributor

Choose a reason for hiding this comment

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

something is wrong with the language in this comment. it doesn't parse correctly.

i think "popover closes:" belongs after "so that", but then the second bullet sounds incorrect.

mud added 2 commits February 6, 2018 20:34
…event.

Using relatedTarget in focus event to see if focus moved out of popover or not.
@mud
Copy link
Contributor Author

mud commented Feb 15, 2018

@giladgray @adidahiya can you take a look and see if we can get this merged?

Copy link
Contributor

@giladgray giladgray left a comment

Choose a reason for hiding this comment

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

👍 looks good! sorry @mud, we're all focused on final 2.0 work

@mud
Copy link
Contributor Author

mud commented Feb 15, 2018

np. good luck with 2.0!

@adidahiya adidahiya merged commit a687739 into palantir:release/1.x Feb 15, 2018
@adidahiya adidahiya changed the title Close dateinput popover [release/1.x] Close dateinput popover Feb 19, 2018
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.

3 participants