Skip to content

Commit

Permalink
fix(input-date-picker): reset active date picker date after closing (#…
Browse files Browse the repository at this point in the history
…7219)

**Related Issue:** #6495 

## Summary

This ensures the active date is reset properly after a
`calcite-input-date-picker`'s date picker is closed (either by blurring
or selecting a date).

## Notes

* This adds an internal `reset` method that can be made public in the
future if needed.
* There was an existing `reset` method prior to these changes that seems
to clear the active date when the date picker is blurred or when
`Escape` is pressed. We should revisit this behavior since it was [added
when the input and date picker were a single
component](https://github.com/Esri/calcite-components/blob/v1.0.0-beta.22/src/components/calcite-date/calcite-date.tsx#L263-L270)
and it does seem odd to have the date-picker reset in these scenarios.
cc @macandcheese @SkyeSeitz @ashetland
* Utility test methods to interact with internal components were added
and tests were updated to leverage them
* Slightly increases the duration factor in the tests to work around
#6604

---------

Co-authored-by: Ben Elan <no-reply@benelan.dev>
  • Loading branch information
jcfranco and benelan committed Jun 28, 2023
1 parent 3baf1d5 commit 005836d
Show file tree
Hide file tree
Showing 6 changed files with 144 additions and 111 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -197,6 +197,16 @@ export class DatePicker implements LocalizedComponent, LoadableComponent, T9nCom
this.el.focus();
}

/**
* Resets active date state.
* @internal
*/
@Method()
async reset(): Promise<void> {
this.resetActiveDates();
this.mostRecentRangeValue = undefined;
}

// --------------------------------------------------------------------------
//
// Lifecycle
Expand Down Expand Up @@ -275,7 +285,7 @@ export class DatePicker implements LocalizedComponent, LoadableComponent, T9nCom
: this.maxAsDate
: this.maxAsDate;
return (
<Host onBlur={this.reset} onKeyDown={this.keyDownHandler}>
<Host onBlur={this.resetActiveDates} onKeyDown={this.keyDownHandler}>
{this.renderCalendar(activeDate, maxDate, minDate, date, endDate)}
</Host>
);
Expand Down Expand Up @@ -319,7 +329,7 @@ export class DatePicker implements LocalizedComponent, LoadableComponent, T9nCom

@State() private localeData: DateLocaleData;

private mostRecentRangeValue?: Date;
@State() private mostRecentRangeValue?: Date;

@State() startAsDate: Date;

Expand All @@ -331,7 +341,7 @@ export class DatePicker implements LocalizedComponent, LoadableComponent, T9nCom

keyDownHandler = (event: KeyboardEvent): void => {
if (event.key === "Escape") {
this.reset();
this.resetActiveDates();
}
};

Expand Down Expand Up @@ -509,31 +519,18 @@ export class DatePicker implements LocalizedComponent, LoadableComponent, T9nCom
);
}

/**
* Reset active date and close
*/
reset = (): void => {
private resetActiveDates = (): void => {
const { valueAsDate } = this;
if (
!Array.isArray(valueAsDate) &&
valueAsDate &&
valueAsDate?.getTime() !== this.activeDate?.getTime()
) {

if (!Array.isArray(valueAsDate) && valueAsDate && valueAsDate !== this.activeDate) {
this.activeDate = new Date(valueAsDate);
}

if (Array.isArray(valueAsDate)) {
if (
valueAsDate[0] &&
valueAsDate[0]?.getTime() !==
(this.activeStartDate instanceof Date && this.activeStartDate?.getTime())
) {
if (valueAsDate[0] && valueAsDate[0] !== this.activeStartDate) {
this.activeStartDate = new Date(valueAsDate[0]);
}
if (
valueAsDate[1] &&
valueAsDate[1]?.getTime() !==
(this.activeStartDate instanceof Date && this.activeEndDate?.getTime())
) {
if (valueAsDate[1] && valueAsDate[1] !== this.activeEndDate) {
this.activeEndDate = new Date(valueAsDate[1]);
}
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,12 @@ describe("calcite-dropdown", () => {
});

describe("disabled", () => {
disabled(simpleDropdownHTML, { focusTarget: "child" });
disabled(simpleDropdownHTML, {
focusTarget: {
tab: "calcite-button",
click: "calcite-dropdown-item"
}
});
});

interface SelectedItemsAssertionOptions {
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -57,6 +57,64 @@ describe("calcite-input-date-picker", () => {

it.skip("supports t9n", () => t9n("calcite-input-date-picker"));

async function navigateMonth(page: E2EPage, direction: "previous" | "next"): Promise<void> {
const linkIndex = direction === "previous" ? 0 : 1;

await page.evaluate(
async (MONTH_HEADER_CSS, linkIndex: number): Promise<void> =>
document
.querySelector("calcite-input-date-picker")
.shadowRoot.querySelector("calcite-date-picker")
.shadowRoot.querySelector("calcite-date-picker-month-header")
.shadowRoot.querySelectorAll<HTMLAnchorElement>(`.${MONTH_HEADER_CSS.chevron}`)
[linkIndex].click(),
MONTH_HEADER_CSS,
linkIndex
);
await page.waitForChanges();
}

async function selectDayInMonth(page: E2EPage, day: number): Promise<void> {
const dayIndex = day - 1;

await page.evaluate(
async (dayIndex: number) =>
document
.querySelector<HTMLCalciteInputDatePickerElement>("calcite-input-date-picker")
.shadowRoot.querySelector<HTMLCalciteDatePickerElement>("calcite-date-picker")
.shadowRoot.querySelector<HTMLCalciteDatePickerMonthElement>("calcite-date-picker-month")
.shadowRoot.querySelectorAll<HTMLCalciteDatePickerDayElement>("calcite-date-picker-day[current-month]")
[dayIndex].click(),
dayIndex
);
await page.waitForChanges();
}

async function getActiveMonth(page: E2EPage): Promise<string> {
return page.evaluate(
async (MONTH_HEADER_CSS) =>
document
.querySelector("calcite-input-date-picker")
.shadowRoot.querySelector("calcite-date-picker")
.shadowRoot.querySelector("calcite-date-picker-month-header")
.shadowRoot.querySelector(`.${MONTH_HEADER_CSS.month}`).textContent,
MONTH_HEADER_CSS
);
}

async function getDateInputValue(page: E2EPage, type: "start" | "end" = "start"): Promise<string> {
const inputIndex = type === "start" ? 0 : 1;

return page.evaluate(
async (inputIndex: number): Promise<string> =>
document
.querySelector("calcite-input-date-picker")
.shadowRoot.querySelectorAll("calcite-input")
[inputIndex].shadowRoot.querySelector("input").value,
inputIndex
);
}

describe("event emitting when the value changes", () => {
it("emits change event when value is committed for single date", async () => {
const page = await newE2EPage();
Expand Down Expand Up @@ -201,14 +259,7 @@ describe("calcite-input-date-picker", () => {
await page.waitForChanges();

expect(changeEvent).toHaveReceivedEventTimes(0);

const inputValue = await page.evaluate(() => {
const inputDatePicker = document.querySelector("calcite-input-date-picker");
const calciteInput = inputDatePicker.shadowRoot.querySelector("calcite-input");
const input = calciteInput.shadowRoot.querySelector("input");
return input.value;
});
expect(inputValue).toBe("3/7/");
expect(await getDateInputValue(page)).toBe("3/7/");
});
});

Expand Down Expand Up @@ -294,38 +345,30 @@ describe("calcite-input-date-picker", () => {
await page.setContent(
`<calcite-input-date-picker lang="ar" numbering-system="arab"></calcite-input-date-picker>`
);
const getInputValue = async () =>
await page.evaluate(
() =>
document
.querySelector("calcite-input-date-picker")
.shadowRoot.querySelector("calcite-input")
.shadowRoot.querySelector("input").value
);

await page.keyboard.press("Tab");
await page.keyboard.type("1/");
await page.waitForChanges();

expect(await getInputValue()).toBe("١‏/");
expect(await getDateInputValue(page)).toBe("١‏/");

await page.keyboard.type("2");
await page.waitForChanges();

// NOTE: This asserted value was copied from the received value in a test failure caused by
// typing these same values into the test file using an Arabic input source on macOS.
// Make sure to preserve this value when refactoring instead of typing these characters from scratch.
expect(await getInputValue()).toBe("١‏‏/٢");
expect(await getDateInputValue(page)).toBe("١‏‏/٢");

await page.keyboard.type("/");
await page.waitForChanges();

expect(await getInputValue()).toBe("١‏‏‏/٢‏/");
expect(await getDateInputValue(page)).toBe("١‏‏‏/٢‏/");

await page.keyboard.type("1234");
await page.waitForChanges();

expect(await getInputValue()).toBe("١‏‏‏‏‏‏‏/٢‏‏‏‏‏/١٢٣٤");
expect(await getDateInputValue(page)).toBe("١‏‏‏‏‏‏‏/٢‏‏‏‏‏/١٢٣٤");
});

it("syncs lang changes to internal date-picker and input", async () => {
Expand All @@ -348,36 +391,16 @@ describe("calcite-input-date-picker", () => {
);
const inputDatePicker = await page.find("calcite-input-date-picker");

const getLocalizedMonth = async () =>
await page.evaluate(
async (MONTH_HEADER_CSS) =>
document
.querySelector("calcite-input-date-picker")
.shadowRoot.querySelector("calcite-date-picker")
.shadowRoot.querySelector("calcite-date-picker-month-header")
.shadowRoot.querySelector(`.${MONTH_HEADER_CSS.month}`).textContent,
MONTH_HEADER_CSS
);

const getLocalizedInputValue = async () =>
await page.evaluate(
async () =>
document
.querySelector("calcite-input-date-picker")
.shadowRoot.querySelector("calcite-input")
.shadowRoot.querySelector("input").value
);

expect(await getLocalizedMonth()).toEqual(langTranslations.months.wide[Number(month) - 1]);
expect(await getLocalizedInputValue()).toBe(
expect(await getActiveMonth(page)).toEqual(langTranslations.months.wide[Number(month) - 1]);
expect(await getDateInputValue(page)).toBe(
langTranslations.placeholder.replace("DD", day).replace("MM", month).replace("YYYY", year)
);

inputDatePicker.setProperty("lang", newLang);
await page.waitForChanges();

expect(await getLocalizedMonth()).toEqual(newLangTranslations.months.wide[Number(month) - 1]);
expect(await getLocalizedInputValue()).toBe(
expect(await getActiveMonth(page)).toEqual(newLangTranslations.months.wide[Number(month) - 1]);
expect(await getDateInputValue(page)).toBe(
newLangTranslations.placeholder.replace("DD", day).replace("MM", month).replace("YYYY", year)
);
});
Expand All @@ -391,16 +414,7 @@ describe("calcite-input-date-picker", () => {
await inputDatePicker.click();
await calciteInputDatePickerOpenEvent;

await page.evaluate(async () =>
// select first day of month
document
.querySelector<HTMLCalciteInputDatePickerElement>("calcite-input-date-picker")
.shadowRoot.querySelector<HTMLCalciteDatePickerElement>("calcite-date-picker")
.shadowRoot.querySelector<HTMLCalciteDatePickerMonthElement>("calcite-date-picker-month")
.shadowRoot.querySelector<HTMLCalciteDatePickerDayElement>("calcite-date-picker-day[current-month]")
.click()
);
await page.waitForChanges();
await selectDayInMonth(page, 1);
await inputDatePicker.callMethod("blur");

expect(await inputDatePicker.getProperty("value")).toBe("2023-05-01");
Expand All @@ -415,14 +429,7 @@ describe("calcite-input-date-picker", () => {
await inputDatePicker.click();
await page.waitForChanges();

await page.evaluate(() => {
const inputDatePicker = document.querySelector("calcite-input-date-picker");
const datePicker = inputDatePicker.shadowRoot.querySelector("calcite-date-picker");
const datePickerMonth = datePicker.shadowRoot.querySelector("calcite-date-picker-month");
const datePickerDay = datePickerMonth.shadowRoot.querySelector("calcite-date-picker-day");

datePickerDay.click();
});
await selectDayInMonth(page, 1);

expect(await inputDatePicker.getProperty("value")).toBe("2023-01-01");
});
Expand Down Expand Up @@ -567,15 +574,7 @@ describe("calcite-input-date-picker", () => {
const expectedInputValue = "10/1/2022";

expect(await inputDatePickerEl.getProperty("value")).toEqual(expectedValue);

const inputValue = await page.evaluate(() => {
const inputDatePicker = document.querySelector("calcite-input-date-picker");
const calciteInput = inputDatePicker.shadowRoot.querySelector("calcite-input");
const input = calciteInput.shadowRoot.querySelector("input");
return input.value;
});

expect(inputValue).toEqual(expectedInputValue);
expect(await getDateInputValue(page)).toEqual(expectedInputValue);
});

it("should update this.value and both input values when valueAsDate is set for range", async () => {
Expand All @@ -597,22 +596,8 @@ describe("calcite-input-date-picker", () => {
const expectedStartDateInputValue = "10/1/2022";
const expectedEndDateInputValue = "10/2/2022";

const startDateInputValue = await page.evaluate(() => {
const inputDatePicker = document.querySelector("calcite-input-date-picker");
const calciteInput = inputDatePicker.shadowRoot.querySelector("calcite-input");
const input = calciteInput.shadowRoot.querySelector("input");
return input.value;
});

const endDateInputValue = await page.evaluate(() => {
const inputDatePicker = document.querySelector("calcite-input-date-picker");
const calciteInputs = inputDatePicker.shadowRoot.querySelectorAll("calcite-input");
const input = calciteInputs[1].shadowRoot.querySelector("input");
return input.value;
});

expect(startDateInputValue).toEqual(expectedStartDateInputValue);
expect(endDateInputValue).toEqual(expectedEndDateInputValue);
expect(await getDateInputValue(page, "start")).toEqual(expectedStartDateInputValue);
expect(await getDateInputValue(page, "end")).toEqual(expectedEndDateInputValue);
});

it("should return endDate time as 23:59:999 when valueAsDate property is parsed", async () => {
Expand Down Expand Up @@ -827,4 +812,41 @@ describe("calcite-input-date-picker", () => {
expect(await inputDatePickerEl.getProperty("value")).toEqual("");
expect(await input.getProperty("value")).toEqual("");
});

it("should sync its date-pickers when updated programmatically after a user modifies the range", async () => {
const page = await newE2EPage();
await page.setContent(html`<calcite-input-date-picker range></calcite-input-date-picker>`);
await skipAnimations(page);

const inputDatePicker = await page.find("calcite-input-date-picker");
inputDatePicker.setProperty("value", ["2023-02-01", "2023-02-28"]);
await page.waitForChanges();

const [startDatePicker, endDatePicker] = await page.findAll("calcite-input-date-picker >>> calcite-input");

await startDatePicker.click();
await page.waitForChanges();

await navigateMonth(page, "previous");
await selectDayInMonth(page, 1);

await endDatePicker.click();
await page.waitForChanges();

await navigateMonth(page, "previous");
await selectDayInMonth(page, 31);

inputDatePicker.setProperty("value", ["2022-10-01", "2022-10-31"]);
await page.waitForChanges();

await startDatePicker.click();
await page.waitForChanges();

expect(await getActiveMonth(page)).toBe("October");

await endDatePicker.click();
await page.waitForChanges();

expect(await getActiveMonth(page)).toBe("October");
});
});
Original file line number Diff line number Diff line change
Expand Up @@ -787,6 +787,7 @@ export class InputDatePicker
deactivateFocusTrap(this);
this.restoreInputFocus();
this.focusOnOpen = false;
this.datePickerEl.reset();
}

setStartInput = (el: HTMLCalciteInputElement): void => {
Expand Down
Loading

0 comments on commit 005836d

Please sign in to comment.