calcite-input-date-picker.cjs.entry.js 26 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516
  1. /*!
  2. * All material copyright ESRI, All Rights Reserved, unless otherwise specified.
  3. * See https://github.com/Esri/calcite-components/blob/master/LICENSE.md for details.
  4. * v1.0.0-beta.82
  5. */
  6. 'use strict';
  7. Object.defineProperty(exports, '__esModule', { value: true });
  8. const index = require('./index-5c65e149.js');
  9. const resources = require('./resources-247d05d5.js');
  10. const label = require('./label-165fc611.js');
  11. const form = require('./form-11926121.js');
  12. const popper = require('./popper-3f97eb01.js');
  13. const interactive = require('./interactive-e294111f.js');
  14. const dom = require('./dom-9ac0341c.js');
  15. require('./locale-f2ebe4fa.js');
  16. require('./number-364babbc.js');
  17. require('./guid-8b6d6cb4.js');
  18. const inputDatePickerCss = "@-webkit-keyframes in{0%{opacity:0}100%{opacity:1}}@keyframes in{0%{opacity:0}100%{opacity:1}}@-webkit-keyframes in-down{0%{opacity:0;-webkit-transform:translate3D(0, -5px, 0);transform:translate3D(0, -5px, 0)}100%{opacity:1;-webkit-transform:translate3D(0, 0, 0);transform:translate3D(0, 0, 0)}}@keyframes in-down{0%{opacity:0;-webkit-transform:translate3D(0, -5px, 0);transform:translate3D(0, -5px, 0)}100%{opacity:1;-webkit-transform:translate3D(0, 0, 0);transform:translate3D(0, 0, 0)}}@-webkit-keyframes in-up{0%{opacity:0;-webkit-transform:translate3D(0, 5px, 0);transform:translate3D(0, 5px, 0)}100%{opacity:1;-webkit-transform:translate3D(0, 0, 0);transform:translate3D(0, 0, 0)}}@keyframes in-up{0%{opacity:0;-webkit-transform:translate3D(0, 5px, 0);transform:translate3D(0, 5px, 0)}100%{opacity:1;-webkit-transform:translate3D(0, 0, 0);transform:translate3D(0, 0, 0)}}@-webkit-keyframes in-scale{0%{opacity:0;-webkit-transform:scale3D(0.95, 0.95, 1);transform:scale3D(0.95, 0.95, 1)}100%{opacity:1;-webkit-transform:scale3D(1, 1, 1);transform:scale3D(1, 1, 1)}}@keyframes in-scale{0%{opacity:0;-webkit-transform:scale3D(0.95, 0.95, 1);transform:scale3D(0.95, 0.95, 1)}100%{opacity:1;-webkit-transform:scale3D(1, 1, 1);transform:scale3D(1, 1, 1)}}:root{--calcite-animation-timing:calc(150ms * var(--calcite-internal-duration-factor));--calcite-internal-duration-factor:var(--calcite-duration-factor, 1);--calcite-internal-animation-timing-fast:calc(100ms * var(--calcite-internal-duration-factor));--calcite-internal-animation-timing-medium:calc(200ms * var(--calcite-internal-duration-factor));--calcite-internal-animation-timing-slow:calc(300ms * var(--calcite-internal-duration-factor))}.calcite-animate{opacity:0;-webkit-animation-fill-mode:both;animation-fill-mode:both;-webkit-animation-duration:var(--calcite-animation-timing);animation-duration:var(--calcite-animation-timing)}.calcite-animate__in{-webkit-animation-name:in;animation-name:in}.calcite-animate__in-down{-webkit-animation-name:in-down;animation-name:in-down}.calcite-animate__in-up{-webkit-animation-name:in-up;animation-name:in-up}.calcite-animate__in-scale{-webkit-animation-name:in-scale;animation-name:in-scale}:host{--calcite-icon-size:1rem;--calcite-spacing-eighth:0.125rem;--calcite-spacing-quarter:0.25rem;--calcite-spacing-half:0.5rem;--calcite-spacing-three-quarters:0.75rem;--calcite-spacing:1rem;--calcite-spacing-plus-quarter:1.25rem;--calcite-spacing-plus-half:1.5rem;--calcite-spacing-double:2rem;--calcite-menu-min-width:10rem;--calcite-header-min-height:3rem;--calcite-footer-min-height:3rem}:root{--calcite-popper-transition:var(--calcite-animation-timing)}:host([hidden]){display:none}:host{position:relative;display:inline-block;width:100%;overflow:visible;vertical-align:top;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;-webkit-box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow)}:host([disabled]){pointer-events:none;cursor:default;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;opacity:var(--calcite-ui-opacity-disabled)}:host([disabled]) ::slotted([calcite-hydrated][disabled]),:host([disabled]) [calcite-hydrated][disabled]{opacity:1}.calendar-picker-wrapper{position:static;width:100%;--tw-shadow:0 0 #0000;--tw-shadow-colored:0 0 #0000;-webkit-box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);box-shadow:var(--tw-ring-offset-shadow, 0 0 #0000), var(--tw-ring-shadow, 0 0 #0000), var(--tw-shadow);-webkit-transform:translate3d(0, 0, 0);transform:translate3d(0, 0, 0)}.input-wrapper{position:relative}:host([range]) .input-container{display:-ms-flexbox;display:flex}:host([range]) .input-wrapper{-ms-flex:1 1 auto;flex:1 1 auto}:host([range]) .horizontal-arrow-container{display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;border-width:1px;border-left-width:0px;border-right-width:0px;border-style:solid;border-color:var(--calcite-ui-border-input);background-color:var(--calcite-ui-background);padding-top:0px;padding-bottom:0px;padding-left:0.25rem;padding-right:0.25rem}:host([range][layout=vertical]) .input-wrapper{width:100%}:host([range][layout=vertical]) .input-container{-ms-flex-direction:column;flex-direction:column;-ms-flex-align:start;align-items:flex-start}:host([range][layout=vertical]) .calendar-picker-wrapper--end{-webkit-transform:translate3d(0, 0, 0);transform:translate3d(0, 0, 0)}:host([range][layout=vertical]) .vertical-arrow-container{top:1.5rem;position:absolute;left:0px;z-index:10;margin-left:1px;margin-right:1px;background-color:var(--calcite-ui-foreground-1);padding-left:0.625rem;padding-right:0.625rem}:host([scale=s][range]:not([layout=vertical])) .calendar-picker-wrapper{width:216px}:host([scale=m][range]:not([layout=vertical])) .calendar-picker-wrapper{width:286px}:host([scale=l][range]:not([layout=vertical])) .calendar-picker-wrapper{width:398px}.menu-container{display:block;position:absolute;z-index:900;-webkit-transform:scale(0);transform:scale(0);visibility:hidden;pointer-events:none}.menu-container .calcite-popper-anim{position:relative;z-index:1;-webkit-transition:var(--calcite-popper-transition);transition:var(--calcite-popper-transition);visibility:hidden;-webkit-transition-property:visibility, opacity, -webkit-transform;transition-property:visibility, opacity, -webkit-transform;transition-property:transform, visibility, opacity;transition-property:transform, visibility, opacity, -webkit-transform;opacity:0;-webkit-box-shadow:0 0 16px 0 rgba(0, 0, 0, 0.16);box-shadow:0 0 16px 0 rgba(0, 0, 0, 0.16);border-radius:0.25rem}.menu-container[data-popper-placement^=bottom] .calcite-popper-anim{-webkit-transform:translateY(-5px);transform:translateY(-5px)}.menu-container[data-popper-placement^=top] .calcite-popper-anim{-webkit-transform:translateY(5px);transform:translateY(5px)}.menu-container[data-popper-placement^=left] .calcite-popper-anim{-webkit-transform:translateX(5px);transform:translateX(5px)}.menu-container[data-popper-placement^=right] .calcite-popper-anim{-webkit-transform:translateX(-5px);transform:translateX(-5px)}.menu-container[data-popper-placement] .calcite-popper-anim--active{opacity:1;visibility:visible;-webkit-transform:translate(0);transform:translate(0)}:host([active]) .menu-container{pointer-events:initial;visibility:visible}.menu-container--active .menu-container .calcite-popper-anim{position:relative;z-index:1;-webkit-transition:var(--calcite-popper-transition);transition:var(--calcite-popper-transition);visibility:hidden;-webkit-transition-property:visibility, opacity, -webkit-transform;transition-property:visibility, opacity, -webkit-transform;transition-property:transform, visibility, opacity;transition-property:transform, visibility, opacity, -webkit-transform;opacity:0;-webkit-box-shadow:0 0 16px 0 rgba(0, 0, 0, 0.16);box-shadow:0 0 16px 0 rgba(0, 0, 0, 0.16);border-radius:0.25rem}.menu-container--active .menu-container[data-popper-placement^=bottom] .calcite-popper-anim{-webkit-transform:translateY(-5px);transform:translateY(-5px)}.menu-container--active .menu-container[data-popper-placement^=top] .calcite-popper-anim{-webkit-transform:translateY(5px);transform:translateY(5px)}.menu-container--active .menu-container[data-popper-placement^=left] .calcite-popper-anim{-webkit-transform:translateX(5px);transform:translateX(5px)}.menu-container--active .menu-container[data-popper-placement^=right] .calcite-popper-anim{-webkit-transform:translateX(-5px);transform:translateX(-5px)}.menu-container--active .menu-container[data-popper-placement] .calcite-popper-anim--active{opacity:1;visibility:visible;-webkit-transform:translate(0);transform:translate(0)}.input .calcite-input__wrapper{margin-top:0px}:host([range][layout=vertical][scale=m]) .vertical-arrow-container{top:1.5rem;padding-left:0.75rem}:host([range][layout=vertical][scale=m]) .vertical-arrow-container calcite-icon{height:0.75rem;width:0.75rem;min-width:0px}:host([range][layout=vertical][scale=l]) .vertical-arrow-container{top:2.25rem;padding-left:0.875rem;padding-right:0.875rem}:host([range][layout=vertical][active]) .vertical-arrow-container{display:none}::slotted(input[slot=hidden-form-input]){bottom:0 !important;left:0 !important;margin:0 !important;opacity:0 !important;outline:none !important;padding:0 !important;position:absolute !important;right:0 !important;top:0 !important;-webkit-transform:none !important;transform:none !important;-webkit-appearance:none !important;z-index:-1 !important}";
  19. const InputDatePicker = class {
  20. constructor(hostRef) {
  21. index.registerInstance(this, hostRef);
  22. this.calciteDatePickerChange = index.createEvent(this, "calciteDatePickerChange", 7);
  23. this.calciteDatePickerRangeChange = index.createEvent(this, "calciteDatePickerRangeChange", 7);
  24. this.calciteInputDatePickerChange = index.createEvent(this, "calciteInputDatePickerChange", 7);
  25. this.calciteInputDatePickerOpen = index.createEvent(this, "calciteInputDatePickerOpen", 7);
  26. this.calciteInputDatePickerClose = index.createEvent(this, "calciteInputDatePickerClose", 7);
  27. //--------------------------------------------------------------------------
  28. //
  29. // Public Properties
  30. //
  31. //--------------------------------------------------------------------------
  32. /**
  33. * When false, the component won't be interactive.
  34. */
  35. this.disabled = false;
  36. /** Expand or collapse when calendar does not have input */
  37. this.active = false;
  38. /** Localized string for "previous month" (used for aria label)
  39. * @default "Previous month"
  40. */
  41. this.intlPrevMonth = resources.TEXT.prevMonth;
  42. /** Localized string for "next month" (used for aria label)
  43. * @default "Next month"
  44. */
  45. this.intlNextMonth = resources.TEXT.nextMonth;
  46. /** Localized string for "year" (used for aria label)
  47. * @default "Year"
  48. */
  49. this.intlYear = resources.TEXT.year;
  50. /** BCP 47 language tag for desired language and country format */
  51. this.locale = document.documentElement.lang || "en";
  52. /** specify the scale of the date picker */
  53. this.scale = "m";
  54. /**
  55. * Determines where the date-picker component will be positioned relative to the input.
  56. * @default "bottom-leading"
  57. */
  58. this.placement = popper.defaultMenuPlacement;
  59. /** Range mode activation */
  60. this.range = false;
  61. /**
  62. * When true, makes the component required for form-submission.
  63. *
  64. * @internal
  65. */
  66. this.required = false;
  67. /** Describes the type of positioning to use for the overlaid content. If your element is in a fixed container, use the 'fixed' value. */
  68. this.overlayPositioning = "absolute";
  69. /** Disables the default behaviour on the third click of narrowing or extending the range and instead starts a new range. */
  70. this.proximitySelectionDisabled = false;
  71. /** Layout */
  72. this.layout = "horizontal";
  73. this.focusedInput = "start";
  74. this.activeTransitionProp = "opacity";
  75. //--------------------------------------------------------------------------
  76. //
  77. // Private Methods
  78. //
  79. //--------------------------------------------------------------------------
  80. this.setFilteredPlacements = () => {
  81. const { el, flipPlacements } = this;
  82. this.filteredFlipPlacements = flipPlacements
  83. ? popper.filterComputedPlacements(flipPlacements, el)
  84. : null;
  85. };
  86. this.transitionEnd = (event) => {
  87. if (event.propertyName === this.activeTransitionProp) {
  88. this.active
  89. ? this.calciteInputDatePickerOpen.emit()
  90. : this.calciteInputDatePickerClose.emit();
  91. }
  92. };
  93. this.setStartInput = (el) => {
  94. this.startInput = el;
  95. };
  96. this.setEndInput = (el) => {
  97. this.endInput = el;
  98. };
  99. this.deactivate = () => {
  100. this.active = false;
  101. };
  102. this.keyDownHandler = (event) => {
  103. if (event.key === "Enter" && !event.defaultPrevented) {
  104. form.submitForm(this);
  105. }
  106. };
  107. this.keyUpHandler = (e) => {
  108. if (e.key === "Escape") {
  109. this.active = false;
  110. }
  111. };
  112. this.inputBlur = (e) => {
  113. this.blur(e.detail);
  114. };
  115. this.startInputFocus = () => {
  116. this.active = true;
  117. this.focusedInput = "start";
  118. };
  119. this.endInputFocus = () => {
  120. this.active = true;
  121. this.focusedInput = "end";
  122. };
  123. this.inputInput = (e) => {
  124. this.input(e.detail.value);
  125. };
  126. this.setMenuEl = (el) => {
  127. if (el) {
  128. this.menuEl = el;
  129. this.createPopper();
  130. }
  131. };
  132. this.setStartWrapper = (el) => {
  133. this.startWrapper = el;
  134. this.setReferenceEl();
  135. };
  136. this.setEndWrapper = (el) => {
  137. this.endWrapper = el;
  138. this.setReferenceEl();
  139. };
  140. /**
  141. * Event handler for when the selected date changes
  142. */
  143. this.handleDateChange = (event) => {
  144. if (this.range) {
  145. return;
  146. }
  147. this.value = resources.dateToISO(event.detail);
  148. };
  149. this.handleDateRangeChange = (event) => {
  150. var _a, _b;
  151. if (!this.range || !event.detail) {
  152. return;
  153. }
  154. const { startDate, endDate } = event.detail;
  155. this.start = resources.dateToISO(startDate);
  156. this.end = resources.dateToISO(endDate);
  157. this.value = [this.start, this.end];
  158. if (this.shouldFocusRangeEnd()) {
  159. (_a = this.endInput) === null || _a === void 0 ? void 0 : _a.setFocus();
  160. }
  161. else if (this.shouldFocusRangeStart()) {
  162. (_b = this.startInput) === null || _b === void 0 ? void 0 : _b.setFocus();
  163. }
  164. };
  165. }
  166. handleDisabledChange(value) {
  167. if (!value) {
  168. this.active = false;
  169. }
  170. }
  171. valueHandler(value) {
  172. if (Array.isArray(value)) {
  173. this.valueAsDate = value.map((v) => resources.dateFromISO(v));
  174. this.start = value[0];
  175. this.end = value[1];
  176. }
  177. else if (value) {
  178. this.valueAsDate = resources.dateFromISO(value);
  179. this.start = "";
  180. this.end = "";
  181. }
  182. else {
  183. this.valueAsDate = undefined;
  184. this.start = undefined;
  185. this.end = undefined;
  186. }
  187. }
  188. flipPlacementsHandler() {
  189. this.setFilteredPlacements();
  190. }
  191. onMinChanged(min) {
  192. if (min) {
  193. this.minAsDate = resources.dateFromISO(min);
  194. }
  195. }
  196. onMaxChanged(max) {
  197. if (max) {
  198. this.maxAsDate = resources.dateFromISO(max);
  199. }
  200. }
  201. activeHandler() {
  202. if (!this.disabled) {
  203. this.reposition();
  204. return;
  205. }
  206. this.active = false;
  207. }
  208. //--------------------------------------------------------------------------
  209. //
  210. // Event Listeners
  211. //
  212. //--------------------------------------------------------------------------
  213. handleDateOrRangeChange() {
  214. this.calciteInputDatePickerChange.emit();
  215. }
  216. calciteDaySelectHandler() {
  217. if (this.shouldFocusRangeStart() || this.shouldFocusRangeEnd()) {
  218. return;
  219. }
  220. this.active = false;
  221. }
  222. // --------------------------------------------------------------------------
  223. //
  224. // Public Methods
  225. //
  226. // --------------------------------------------------------------------------
  227. /** Updates the position of the component. */
  228. async setFocus() {
  229. var _a;
  230. (_a = this.startInput) === null || _a === void 0 ? void 0 : _a.setFocus();
  231. }
  232. /** Updates the position of the component. */
  233. async reposition() {
  234. const { placement, popper: popper$1, menuEl } = this;
  235. const modifiers = this.getModifiers();
  236. popper$1
  237. ? await popper.updatePopper({
  238. el: menuEl,
  239. modifiers,
  240. placement,
  241. popper: popper$1
  242. })
  243. : this.createPopper();
  244. }
  245. // --------------------------------------------------------------------------
  246. //
  247. // Lifecycle
  248. //
  249. // --------------------------------------------------------------------------
  250. connectedCallback() {
  251. if (Array.isArray(this.value)) {
  252. this.valueAsDate = this.value.map((v) => resources.dateFromISO(v));
  253. this.start = this.value[0];
  254. this.end = this.value[1];
  255. }
  256. else if (this.value) {
  257. this.valueAsDate = resources.dateFromISO(this.value);
  258. this.start = "";
  259. this.end = "";
  260. }
  261. if (this.start) {
  262. this.startAsDate = resources.dateFromISO(this.start);
  263. }
  264. if (this.end) {
  265. this.endAsDate = resources.dateFromISO(this.end);
  266. }
  267. if (this.min) {
  268. this.minAsDate = resources.dateFromISO(this.min);
  269. }
  270. if (this.max) {
  271. this.maxAsDate = resources.dateFromISO(this.max);
  272. }
  273. this.createPopper();
  274. label.connectLabel(this);
  275. form.connectForm(this);
  276. this.setFilteredPlacements();
  277. }
  278. async componentWillLoad() {
  279. await this.loadLocaleData();
  280. this.onMinChanged(this.min);
  281. this.onMaxChanged(this.max);
  282. }
  283. disconnectedCallback() {
  284. this.destroyPopper();
  285. label.disconnectLabel(this);
  286. form.disconnectForm(this);
  287. }
  288. componentDidRender() {
  289. interactive.updateHostInteraction(this);
  290. }
  291. render() {
  292. var _a, _b;
  293. const { disabled } = this;
  294. const date = resources.dateFromRange(this.range ? this.startAsDate : this.valueAsDate, this.minAsDate, this.maxAsDate);
  295. const endDate = this.range
  296. ? resources.dateFromRange(this.endAsDate, this.minAsDate, this.maxAsDate)
  297. : null;
  298. const formattedEndDate = endDate ? endDate.toLocaleDateString(this.locale) : "";
  299. const formattedDate = date ? date.toLocaleDateString(this.locale) : "";
  300. return (index.h(index.Host, { onBlur: this.deactivate, onKeyDown: this.keyDownHandler, onKeyUp: this.keyUpHandler, role: "application" }, this.localeData && (index.h("div", { "aria-expanded": dom.toAriaBoolean(this.active), class: "input-container", role: "application" }, index.h("div", { class: "input-wrapper", ref: this.setStartWrapper }, index.h("calcite-input", { class: `input ${this.layout === "vertical" && this.range ? `no-bottom-border` : ``}`, disabled: disabled, icon: "calendar", label: label.getLabelText(this), "number-button-type": "none", onCalciteInputBlur: this.inputBlur, onCalciteInputFocus: this.startInputFocus, onCalciteInputInput: this.inputInput, placeholder: (_a = this.localeData) === null || _a === void 0 ? void 0 : _a.placeholder, ref: this.setStartInput, scale: this.scale, type: "text", value: formattedDate })), index.h("div", { "aria-hidden": dom.toAriaBoolean(!this.active), class: {
  301. "menu-container": true,
  302. "menu-container--active": this.active
  303. }, ref: this.setMenuEl }, index.h("div", { class: {
  304. ["calendar-picker-wrapper"]: true,
  305. ["calendar-picker-wrapper--end"]: this.focusedInput === "end",
  306. [popper.CSS.animation]: true,
  307. [popper.CSS.animationActive]: this.active
  308. }, onTransitionEnd: this.transitionEnd }, index.h("calcite-date-picker", { activeRange: this.focusedInput, endAsDate: this.endAsDate, headingLevel: this.headingLevel, intlNextMonth: this.intlNextMonth, intlPrevMonth: this.intlPrevMonth, intlYear: this.intlYear, locale: this.locale, max: this.max, maxAsDate: this.maxAsDate, min: this.min, minAsDate: this.minAsDate, onCalciteDatePickerChange: this.handleDateChange, onCalciteDatePickerRangeChange: this.handleDateRangeChange, proximitySelectionDisabled: this.proximitySelectionDisabled, range: this.range, scale: this.scale, startAsDate: this.startAsDate, tabIndex: 0, valueAsDate: this.valueAsDate }))), this.range && this.layout === "horizontal" && (index.h("div", { class: "horizontal-arrow-container" }, index.h("calcite-icon", { flipRtl: true, icon: "arrow-right", scale: "s" }))), this.range && this.layout === "vertical" && this.scale !== "s" && (index.h("div", { class: "vertical-arrow-container" }, index.h("calcite-icon", { icon: "arrow-down", scale: "s" }))), this.range && (index.h("div", { class: "input-wrapper", ref: this.setEndWrapper }, index.h("calcite-input", { class: {
  309. input: true,
  310. "border-top-color-one": this.layout === "vertical" && this.range
  311. }, disabled: disabled, icon: "calendar", "number-button-type": "none", onCalciteInputBlur: this.inputBlur, onCalciteInputFocus: this.endInputFocus, onCalciteInputInput: this.inputInput, placeholder: (_b = this.localeData) === null || _b === void 0 ? void 0 : _b.placeholder, ref: this.setEndInput, scale: this.scale, type: "text", value: formattedEndDate }))))), index.h(form.HiddenFormInputSlot, { component: this })));
  312. }
  313. setReferenceEl() {
  314. const { focusedInput, layout, endWrapper, startWrapper } = this;
  315. this.referenceEl =
  316. focusedInput === "end" || layout === "vertical"
  317. ? endWrapper || startWrapper
  318. : startWrapper || endWrapper;
  319. this.createPopper();
  320. }
  321. onLabelClick() {
  322. this.setFocus();
  323. }
  324. getModifiers() {
  325. const flipModifier = {
  326. name: "flip",
  327. enabled: true
  328. };
  329. flipModifier.options = {
  330. fallbackPlacements: this.filteredFlipPlacements || popper.popperMenuComputedPlacements
  331. };
  332. const eventListenerModifier = {
  333. name: "eventListeners",
  334. enabled: this.active
  335. };
  336. return [flipModifier, eventListenerModifier];
  337. }
  338. createPopper() {
  339. this.destroyPopper();
  340. const { menuEl, placement, referenceEl, overlayPositioning } = this;
  341. if (!menuEl || !referenceEl) {
  342. return;
  343. }
  344. const modifiers = this.getModifiers();
  345. this.popper = popper.createPopper({
  346. el: menuEl,
  347. modifiers,
  348. overlayPositioning,
  349. placement,
  350. referenceEl
  351. });
  352. }
  353. destroyPopper() {
  354. const { popper } = this;
  355. if (popper) {
  356. popper.destroy();
  357. }
  358. this.popper = null;
  359. }
  360. startWatcher(start) {
  361. this.startAsDate = resources.dateFromISO(start);
  362. }
  363. endWatcher(end) {
  364. this.endAsDate = resources.dateFromISO(end);
  365. }
  366. async loadLocaleData() {
  367. const { locale } = this;
  368. this.localeData = await resources.getLocaleData(locale);
  369. }
  370. clearCurrentValue() {
  371. if (!this.range) {
  372. if (typeof this.value === "string" && this.value) {
  373. this.calciteDatePickerChange.emit();
  374. }
  375. this.value = "";
  376. return;
  377. }
  378. const { focusedInput } = this;
  379. if (focusedInput === "start") {
  380. if (this.start) {
  381. this.calciteDatePickerRangeChange.emit();
  382. }
  383. this.value = Array.isArray(this.value) ? ["", this.value[1] || ""] : [""];
  384. this.start = undefined;
  385. }
  386. else if (focusedInput === "end") {
  387. if (this.end) {
  388. this.calciteDatePickerRangeChange.emit();
  389. }
  390. this.value = Array.isArray(this.value) ? [this.value[0] || "", ""] : ["", ""];
  391. this.end = undefined;
  392. }
  393. }
  394. /**
  395. * If inputted string is a valid date, update value/active
  396. */
  397. input(value) {
  398. const date = this.getDateFromInput(value);
  399. if (!date) {
  400. this.clearCurrentValue();
  401. return;
  402. }
  403. if (!this.range) {
  404. this.value = resources.dateToISO(date);
  405. this.calciteDatePickerChange.emit(date);
  406. return;
  407. }
  408. const { focusedInput } = this;
  409. if (focusedInput === "start") {
  410. if (!this.startAsDate || !resources.sameDate(date, this.startAsDate)) {
  411. const startDateISO = resources.dateToISO(date);
  412. this.value = Array.isArray(this.value)
  413. ? [startDateISO, this.value[1] || ""]
  414. : [startDateISO];
  415. this.start = startDateISO;
  416. this.calciteDatePickerRangeChange.emit({
  417. startDate: date,
  418. endDate: this.endAsDate
  419. });
  420. }
  421. }
  422. else if (focusedInput === "end") {
  423. if (!this.endAsDate || !resources.sameDate(date, this.endAsDate)) {
  424. const endDateISO = resources.dateToISO(date);
  425. this.value = Array.isArray(this.value)
  426. ? [this.value[0] || "", endDateISO]
  427. : ["", endDateISO];
  428. this.end = endDateISO;
  429. this.calciteDatePickerRangeChange.emit({
  430. startDate: this.startAsDate,
  431. endDate: date
  432. });
  433. }
  434. }
  435. }
  436. /**
  437. * Clean up invalid date from input on blur
  438. */
  439. blur(target) {
  440. const { locale, focusedInput, endAsDate, range, startAsDate, valueAsDate } = this;
  441. const date = this.getDateFromInput(target.value);
  442. if (!date) {
  443. if (!range && valueAsDate) {
  444. target.value = Array.isArray(valueAsDate)
  445. ? valueAsDate[focusedInput === "end" ? 1 : 0].toLocaleDateString(locale)
  446. : valueAsDate.toLocaleDateString(locale);
  447. }
  448. else if (focusedInput === "start" && startAsDate) {
  449. target.value = startAsDate.toLocaleDateString(locale);
  450. }
  451. else if (focusedInput === "end" && endAsDate) {
  452. target.value = endAsDate.toLocaleDateString(locale);
  453. }
  454. }
  455. }
  456. shouldFocusRangeStart() {
  457. return !!(this.endAsDate &&
  458. !this.startAsDate &&
  459. this.focusedInput === "end" &&
  460. this.startInput);
  461. }
  462. shouldFocusRangeEnd() {
  463. return !!(this.startAsDate &&
  464. !this.endAsDate &&
  465. this.focusedInput === "start" &&
  466. this.endInput);
  467. }
  468. /**
  469. * Find a date from input string
  470. * return false if date is invalid, or out of range
  471. */
  472. getDateFromInput(value) {
  473. if (!this.localeData) {
  474. return false;
  475. }
  476. const { separator } = this.localeData;
  477. const { day, month, year } = resources.parseDateString(value, this.localeData);
  478. const validDay = day > 0;
  479. const validMonth = month > -1;
  480. const date = new Date(year, month, day);
  481. date.setFullYear(year);
  482. const validDate = !isNaN(date.getTime());
  483. const validLength = value.split(separator).filter((c) => c).length > 2;
  484. const validYear = year.toString().length > 0;
  485. if (validDay &&
  486. validMonth &&
  487. validDate &&
  488. validLength &&
  489. validYear &&
  490. resources.inRange(date, this.min, this.max)) {
  491. return date;
  492. }
  493. return false;
  494. }
  495. get el() { return index.getElement(this); }
  496. static get watchers() { return {
  497. "disabled": ["handleDisabledChange"],
  498. "value": ["valueHandler"],
  499. "flipPlacements": ["flipPlacementsHandler"],
  500. "min": ["onMinChanged"],
  501. "max": ["onMaxChanged"],
  502. "active": ["activeHandler"],
  503. "layout": ["setReferenceEl"],
  504. "focusedInput": ["setReferenceEl"],
  505. "start": ["startWatcher"],
  506. "end": ["endWatcher"],
  507. "locale": ["loadLocaleData"]
  508. }; }
  509. };
  510. InputDatePicker.style = inputDatePickerCss;
  511. exports.calcite_input_date_picker = InputDatePicker;