Skip to content

Commit

Permalink
✨ Add Bento Date Picker Component (ampproject#37385)
Browse files Browse the repository at this point in the history
* Generate initial amp-date-picker 1.0

* Add initial test

* Set value from single input in state

* Add date parsing and formatting

* Add hidden input field for single date

* Add hidden start and end date elements

* Refactor to fix component length

* Add additional range test

* Add imports and copy/pasted CSS

* Add DayPicker controller imports

* Add date-fns and react-day-picker

* Test overlay mode error cases

* Add DayPicker component (this currently breaks the tests)

* Temporarily stub react to fix external module test

* Re-add react-dates and add change date tests

* Fix createLTR error by importing react-dates/initialize

* Add selected dates and initial month

* Swap out react-dates for react-day-picker

* Fix test babel config

* Refactor single input to avoid saving input field in state

* Refactor start and end dates

* Update calendar state logic

* Add basic aria label to date butons

* Add basic blocked dates functionality

* Add blocked date range rules

* Add RRULE blocked date parsing

* Add highlighted dates

* Refactor date picker into single and range components

* Extract test helpers to separate file

* Replace children logic

* Refactor conditional logic for top-level date picker

* Fix rrule tests to include start date

* Add min date

* Add default min value and test stub

* Add max date

* Add month formatting and base date picker abstraction

* Fix storybook build and test week day formatting

* Add initial state hook and refactor single input state

* Add basic opening logic

* Move state machine to single date picker

* Add basic click and focus behavior

* Update build-system/compile/bundles.config.extensions.json

Add npm flag

Co-authored-by: William Johnson <[email protected]>

* Change conditional date picker syntax

* Remove onErrorRef

* Add range input state machine

* Extract date picker state hook

* Refactor state hook arguments

* Remove unused imports

* Add imperative clear function

* Add single date picker API and types

* Update type definitions

* Add API methods for date range picker

* Add locale

* Add locale to base date picker

* Fix storybook

* Add first disabled date based on maximumNights prop

* Add minimum nights prop

* Refactor to use DOM input for existing input field

* Add event handler for input change

* Replace date range children logic with DOM functions

* Add provider for accessing selected dates

* Add overlay closing logic

* Add range event listeners

* Refactor state machine to remove unnecessary values

* Add controlled month for setting date from input

* Add initial typescript configuration

* Convert base component to typescript

* Convert single date picker to typescript

* Convert constants and helper files

* Refactor to use shared input functions

* Replace locale string prop with object

* Update API args

* Add generic attribute parser to parse-props.js

* Add parsers and base element props

* Add openAfterClear attribute (not implemented)

* Update extensions config

* Move components and helpers into /component directory

* Add initial amp test setup

* Add packages and config to render AMP storybook

* Add numberOfMonths prop

* Add initial date picker bento test fixture

* Update src/preact exports to export preact types

* Add initial e2e test (not working right now)

* Impement openAfterClear prop

* Remove unnecessary babel package

* Remove duplicate extension config

* Move component.type

* Fix .tsx extension

Co-authored-by: Justin Ridgewell <[email protected]>

* Undo preact type exports

* Fix date picker e2e test (still no calendar)

* Add React import resolvers for all external imports

* Revert "Add React import resolvers for all external imports"

This reverts commit 0e826f0.

* Add basic CSS styles to mimic original date picker

* Add aliases to unminified babel config

* Use local preact file as react alias

* Add optional today prop for simulating today's date

* Use today as default for initialVisibleMonth

* Add shadow dom support to range date picker

* Refactor date pickers and context

* Add preact types

* Move date parsers and formatters into context

* Updates function descriptions

* WIP Add e2e tests

* Tempoariliy fix forwardRef is not defined error

* Fix static date range logic

* Add non-working overlay range picker test

* Fix date picker in overlay mode

* Fix single date picker click event

* Fix typo

* Fix select date range and storybook

* Update comments and date picker component return type

* Move helpers to reduce base date picker props

* Add TODO

* Re-add 0.1 component to bundles.config.extensions.json

* Move CSS variables

* Use optional chaining for invocation.args

* Add parseNumber function with radix

* Fix type definitions

* Get rid of default props object and type complexity

* Update base date picker props

* Remove pointless input hidden check

* Move default props to provider

In an attempt to share defaults and make it less confusing which props
are coming from the provider and which props are getting passed into the
component, this moves all default props to the provider and creates more
type safety for the date picker components that get data from this
provider.

A potential downside of this change is that it makes it more difficult
to use the SingleDatePicker component and RangeDatePicker in isolation
if that is something we want to do later.

* Extract iterateDateRange to helpers

* Rename change handlers

* Inline initializeStateMachine function

* Add JSS stylesheet to for calendar

* Hide calendar using display:none instead of unmounting

* Add basic overlay styles (no fixed height)

* Downgrade chrome driver

* Remove extraAliases from config and rename preact alias to react

* Remove AMP test files

* Fix z-index file

* Fix test helper (to trigger CI)

Co-authored-by: William Johnson <[email protected]>
Co-authored-by: Justin Ridgewell <[email protected]>
  • Loading branch information
3 people authored Mar 17, 2022
1 parent 9acbd30 commit 881b218
Show file tree
Hide file tree
Showing 37 changed files with 4,275 additions and 46 deletions.
10 changes: 10 additions & 0 deletions build-system/compile/bundles.config.extensions.json
Original file line number Diff line number Diff line change
Expand Up @@ -351,6 +351,16 @@
"hasCss": true
}
},
{
"name": "amp-date-picker",
"version": "1.0",
"latestVersion": "0.1",
"options": {
"npm": true,
"hasCss": true,
"bento": true
}
},
{
"name": "amp-delight-player",
"version": "0.1",
Expand Down
24 changes: 12 additions & 12 deletions build-system/tasks/e2e/package-lock.json

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

2 changes: 2 additions & 0 deletions css/Z_INDEX.md
Original file line number Diff line number Diff line change
Expand Up @@ -73,6 +73,7 @@
| `.amp-carousel-button` | 10 | [extensions/amp-carousel/0.1/amp-carousel.css](/extensions/amp-carousel/0.1/amp-carousel.css) |
| `.amp-carousel-button` | 10 | [extensions/amp-carousel/0.2/amp-carousel.css](/extensions/amp-carousel/0.2/amp-carousel.css) |
| `amp-date-picker[mode="overlay"] .i-amphtml-date-picker-container` | 10 | [extensions/amp-date-picker/0.1/amp-date-picker.css](/extensions/amp-date-picker/0.1/amp-date-picker.css) |
| `overlay['& .rdp']` | 10 | [extensions/amp-date-picker/1.0/component.jss.js](/extensions/amp-date-picker/1.0/component.jss.js) |
| `.i-amphtml-story-spinner` | 10 | [extensions/amp-story/1.0/amp-story.css](/extensions/amp-story/1.0/amp-story.css) |
| `.i-amphtml-byside-content-loading-container .i-amphtml-byside-content-loading-animation:before` | 9 | [extensions/amp-byside-content/0.1/amp-byside-content.css](/extensions/amp-byside-content/0.1/amp-byside-content.css) |
| `100%` | 9 | [extensions/amp-byside-content/0.1/amp-byside-content.css](/extensions/amp-byside-content/0.1/amp-byside-content.css) |
Expand Down Expand Up @@ -110,6 +111,7 @@
| `.i-amphtml-base-carousel-arrow-prev-slot` | 1 | [extensions/amp-base-carousel/0.1/amp-base-carousel.css](/extensions/amp-base-carousel/0.1/amp-base-carousel.css) |
| `.i-amphtml-base-carousel-arrows` | 1 | [extensions/amp-base-carousel/0.1/amp-base-carousel.css](/extensions/amp-base-carousel/0.1/amp-base-carousel.css) |
| `.i-amphtml-carousel-arrows` | 1 | [extensions/amp-carousel/0.2/amp-carousel.css](/extensions/amp-carousel/0.2/amp-carousel.css) |
| `dayPicker['& .rdp-caption_label']` | 1 | [extensions/amp-date-picker/1.0/component.jss.js](/extensions/amp-date-picker/1.0/component.jss.js) |
| `.i-amphtml-fit-text-measurer` | 1 | [extensions/amp-fit-text/0.1/amp-fit-text.css](/extensions/amp-fit-text/0.1/amp-fit-text.css) |
| `.i-amphtml-image-lightbox-viewer` | 1 | [extensions/amp-image-lightbox/0.1/amp-image-lightbox.css](/extensions/amp-image-lightbox/0.1/amp-image-lightbox.css) |
| `.i-amphtml-image-lightbox-viewer-image` | 1 | [extensions/amp-image-lightbox/0.1/amp-image-lightbox.css](/extensions/amp-image-lightbox/0.1/amp-image-lightbox.css) |
Expand Down
Empty file.
51 changes: 51 additions & 0 deletions extensions/amp-date-picker/1.0/amp-date-picker.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,51 @@
import {isExperimentOn} from '#experiments';

import {AmpPreactBaseElement, setSuperClass} from '#preact/amp-base-element';

import {userAssert} from '#utils/log';

import {BaseElement} from './base-element';

import {CSS} from '../../../build/amp-date-picker-1.0.css';

/** @const {string} */
const TAG = 'amp-date-picker';

/** @extends {PreactBaseElement<BentoDatePickerDef.BentoDatePickerApi} */
class AmpDatePicker extends setSuperClass(BaseElement, AmpPreactBaseElement) {
/** @override */
init() {
this.registerApiAction('clear', (api) => api./*OK*/ clear());
this.registerApiAction('today', (api, invocation) =>
api./*OK*/ today(invocation.args?.['offset'])
);
this.registerApiAction('startToday', (api, invocation) =>
api./*OK*/ startToday(invocation.args?.['offset'])
);
this.registerApiAction('endToday', (api, invocation) =>
api./*OK*/ endToday(invocation.args?.['offset'])
);
this.registerApiAction('setDate', (api, invocation) =>
api./*OK*/ setDate(invocation.args?.['date'])
);
this.registerApiAction('setDates', (api, invocation) =>
api./*OK*/ setDates(invocation.args?.['start'], invocation.args?.['end'])
);

return super.init();
}

/** @override */
isLayoutSupported() {
userAssert(
isExperimentOn(this.win, 'bento') ||
isExperimentOn(this.win, 'bento-date-picker'),
'expected global "bento" or specific "bento-date-picker" experiment to be enabled'
);
return true;
}
}

AMP.extension(TAG, '1.0', (AMP) => {
AMP.registerElement(TAG, AmpDatePicker, CSS);
});
71 changes: 71 additions & 0 deletions extensions/amp-date-picker/1.0/base-element.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,71 @@
import {parseBooleanAttribute} from '#core/dom';

import {PreactBaseElement} from '#preact/base-element';

import {BentoDatePicker} from './component';
import {CSS as COMPONENT_CSS} from './component.jss';
import {parseDate, parseDateList, parseLocale, parseNumber} from './parsers';

export class BaseElement extends PreactBaseElement {}

/** @override */
BaseElement['Component'] = BentoDatePicker;

/** @override */
BaseElement['props'] = {
'allowBlockedEndDate': {
attr: 'allow-blocked-end-date',
parseAttr: parseBooleanAttribute,
},
'allowBlockedRanges': {
attr: 'allow-blocked-ranges',
parseAttr: parseBooleanAttribute,
},
'blocked': {attr: 'blocked', parseAttr: parseDateList},
'children': {passthrough: true},
// Not implemented
'daySize': {attr: 'day-size', parseAttr: parseNumber},
'endInputSelector': {attr: 'end-input-selector'},
// Not implemented
'firstDayOfWeek': {attr: 'first-day-of-week', parseAttr: parseNumber},
'format': {attr: 'format'},
// Not implemented
'fullscreen': {attr: 'fullscreen', parseAttr: parseBooleanAttribute},
// Not implemented
'hideKeyboardShortcutsPanel': {
attr: 'hide-keyboard-shortcuts-panel',
parseAttr: parseBooleanAttribute,
},
'highlighted': {attr: 'highlighted', parseAttr: parseDateList},
'initialVisibleMonth': {attr: 'initial-visible-month', parseAttr: parseDate},
'inputSelector': {attr: 'input-selector'},
'locale': {attr: 'locale', parseAttr: parseLocale},
'max': {attr: 'max', parseAttr: parseDate},
'maximumNights': {attr: 'maximum-nights', parseAttr: parseNumber},
'min': {attr: 'min', parseAttr: parseDate},
'minimumNights': {attr: 'minimum-nights', parseAttr: parseNumber},
'mode': {attr: 'mode'},
'monthFormat': {attr: 'month-format'},
'numberOfMonths': {attr: 'number-of-months', parseAttr: parseNumber},
'openAfterClear': {
attr: 'open-after-clear',
parseAttr: parseBooleanAttribute,
},
'openAfterSelect': {
attr: 'open-after-select',
parseAttr: parseBooleanAttribute,
},
'startInputSelector': {attr: 'start-input-selector'},
'today': {attr: 'today', parseAttr: parseDate},
'type': {attr: 'type'},
'weekDayFormat': {attr: 'week-day-format'},
};

/** @override */
BaseElement['layoutSizeDefined'] = true;

/** @override */
BaseElement['usesShadowDom'] = true;

/** @override */
BaseElement['shadowCss'] = COMPONENT_CSS;
1 change: 1 addition & 0 deletions extensions/amp-date-picker/1.0/component.js
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
export * from './component/component';
Loading

0 comments on commit 881b218

Please sign in to comment.