/*! * (C) Ionic http://ionicframework.com - MIT License */ import { proxyCustomElement, HTMLElement, createEvent, h, Host } from '@stencil/core/internal/client'; import { g as getElementRoot } from './helpers.js'; const pickerIosCss = ":host{display:-ms-flexbox;display:flex;position:relative;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:100%;height:200px;direction:ltr;z-index:0}:host .picker-before,:host .picker-after{position:absolute;width:100%;-webkit-transform:translateZ(0);transform:translateZ(0);z-index:1;pointer-events:none}:host .picker-before{top:0;height:83px}:host .picker-before{inset-inline-start:0}:host .picker-after{top:116px;height:84px}:host .picker-after{inset-inline-start:0}:host .picker-highlight{border-radius:var(--highlight-border-radius, 8px);left:0;right:0;top:50%;bottom:0;-webkit-margin-start:auto;margin-inline-start:auto;-webkit-margin-end:auto;margin-inline-end:auto;margin-top:0;margin-bottom:0;position:absolute;width:calc(100% - 16px);height:34px;-webkit-transform:translateY(-50%);transform:translateY(-50%);background:var(--highlight-background);z-index:-1}:host input{position:absolute;top:0;left:0;right:0;bottom:0;width:100%;height:100%;margin:0;padding:0;border:0;outline:0;clip:rect(0 0 0 0);opacity:0;overflow:hidden;-webkit-appearance:none;-moz-appearance:none}:host ::slotted(ion-picker-column:first-of-type){text-align:start}:host ::slotted(ion-picker-column:last-of-type){text-align:end}:host ::slotted(ion-picker-column:only-child){text-align:center}:host .picker-before{background:-webkit-gradient(linear, left top, left bottom, color-stop(20%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1)), to(rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0.8)));background:linear-gradient(to bottom, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1) 20%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0.8) 100%)}:host .picker-after{background:-webkit-gradient(linear, left bottom, left top, color-stop(20%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1)), to(rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0.8)));background:linear-gradient(to top, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1) 20%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0.8) 100%)}:host .picker-highlight{background:var(--highlight-background, var(--ion-color-step-150, var(--ion-background-color-step-150, #eeeeef)))}"; const IonPickerIosStyle0 = pickerIosCss; const pickerMdCss = ":host{display:-ms-flexbox;display:flex;position:relative;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:100%;height:200px;direction:ltr;z-index:0}:host .picker-before,:host .picker-after{position:absolute;width:100%;-webkit-transform:translateZ(0);transform:translateZ(0);z-index:1;pointer-events:none}:host .picker-before{top:0;height:83px}:host .picker-before{inset-inline-start:0}:host .picker-after{top:116px;height:84px}:host .picker-after{inset-inline-start:0}:host .picker-highlight{border-radius:var(--highlight-border-radius, 8px);left:0;right:0;top:50%;bottom:0;-webkit-margin-start:auto;margin-inline-start:auto;-webkit-margin-end:auto;margin-inline-end:auto;margin-top:0;margin-bottom:0;position:absolute;width:calc(100% - 16px);height:34px;-webkit-transform:translateY(-50%);transform:translateY(-50%);background:var(--highlight-background);z-index:-1}:host input{position:absolute;top:0;left:0;right:0;bottom:0;width:100%;height:100%;margin:0;padding:0;border:0;outline:0;clip:rect(0 0 0 0);opacity:0;overflow:hidden;-webkit-appearance:none;-moz-appearance:none}:host ::slotted(ion-picker-column:first-of-type){text-align:start}:host ::slotted(ion-picker-column:last-of-type){text-align:end}:host ::slotted(ion-picker-column:only-child){text-align:center}:host .picker-before{background:-webkit-gradient(linear, left top, left bottom, color-stop(20%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1)), color-stop(90%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0)));background:linear-gradient(to bottom, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1) 20%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0) 90%)}:host .picker-after{background:-webkit-gradient(linear, left bottom, left top, color-stop(30%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1)), color-stop(90%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0)));background:linear-gradient(to top, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 1) 30%, rgba(var(--fade-background-rgb, var(--background-rgb, var(--ion-background-color-rgb, 255, 255, 255))), 0) 90%)}"; const IonPickerMdStyle0 = pickerMdCss; const Picker = /*@__PURE__*/ proxyCustomElement(class Picker extends HTMLElement { constructor() { super(); this.__registerHost(); this.__attachShadow(); this.ionInputModeChange = createEvent(this, "ionInputModeChange", 7); this.useInputMode = false; this.isInHighlightBounds = (ev) => { const { highlightEl } = this; if (!highlightEl) { return false; } const bbox = highlightEl.getBoundingClientRect(); /** * Check to see if the user clicked * outside the bounds of the highlight. */ const outsideX = ev.clientX < bbox.left || ev.clientX > bbox.right; const outsideY = ev.clientY < bbox.top || ev.clientY > bbox.bottom; if (outsideX || outsideY) { return false; } return true; }; /** * If we are no longer focused * on a picker column, then we should * exit input mode. An exception is made * for the input in the picker since having * that focused means we are still in input mode. */ this.onFocusOut = (ev) => { // TODO(FW-2832): type const { relatedTarget } = ev; if (!relatedTarget || (relatedTarget.tagName !== 'ION-PICKER-COLUMN' && relatedTarget !== this.inputEl)) { this.exitInputMode(); } }; /** * When picker columns receive focus * the parent picker needs to determine * whether to enter/exit input mode. */ this.onFocusIn = (ev) => { // TODO(FW-2832): type const { target } = ev; /** * Due to browser differences in how/when focus * is dispatched on certain elements, we need to * make sure that this function only ever runs when * focusing a picker column. */ if (target.tagName !== 'ION-PICKER-COLUMN') { return; } /** * If we have actionOnClick * then this means the user focused * a picker column via mouse or * touch (i.e. a PointerEvent). As a result, * we should not enter/exit input mode * until the click event has fired, which happens * after the `focusin` event. * * Otherwise, the user likely focused * the column using their keyboard and * we should enter/exit input mode automatically. */ if (!this.actionOnClick) { const columnEl = target; const allowInput = columnEl.numericInput; if (allowInput) { this.enterInputMode(columnEl, false); } else { this.exitInputMode(); } } }; /** * On click we need to run an actionOnClick * function that has been set in onPointerDown * so that we enter/exit input mode correctly. */ this.onClick = () => { const { actionOnClick } = this; if (actionOnClick) { actionOnClick(); this.actionOnClick = undefined; } }; /** * Clicking a column also focuses the column on * certain browsers, so we use onPointerDown * to tell the onFocusIn function that users * are trying to click the column rather than * focus the column using the keyboard. When the * user completes the click, the onClick function * runs and runs the actionOnClick callback. */ this.onPointerDown = (ev) => { const { useInputMode, inputModeColumn, el } = this; if (this.isInHighlightBounds(ev)) { /** * If we were already in * input mode, then we should determine * if we tapped a particular column and * should switch to input mode for * that specific column. */ if (useInputMode) { /** * If we tapped a picker column * then we should either switch to input * mode for that column or all columns. * Otherwise we should exit input mode * since we just tapped the highlight and * not a column. */ if (ev.target.tagName === 'ION-PICKER-COLUMN') { /** * If user taps 2 different columns * then we should just switch to input mode * for the new column rather than switching to * input mode for all columns. */ if (inputModeColumn && inputModeColumn === ev.target) { this.actionOnClick = () => { this.enterInputMode(); }; } else { this.actionOnClick = () => { this.enterInputMode(ev.target); }; } } else { this.actionOnClick = () => { this.exitInputMode(); }; } /** * If we were not already in * input mode, then we should * enter input mode for all columns. */ } else { /** * If there is only 1 numeric input column * then we should skip multi column input. */ const columns = el.querySelectorAll('ion-picker-column.picker-column-numeric-input'); const columnEl = columns.length === 1 ? ev.target : undefined; this.actionOnClick = () => { this.enterInputMode(columnEl); }; } return; } this.actionOnClick = () => { this.exitInputMode(); }; }; /** * Enters input mode to allow * for text entry of numeric values. * If on mobile, we focus a hidden input * field so that the on screen keyboard * is brought up. When tabbing using a * keyboard, picker columns receive an outline * to indicate they are focused. As a result, * we should not focus the hidden input as it * would cause the outline to go away, preventing * users from having any visual indication of which * column is focused. */ this.enterInputMode = (columnEl, focusInput = true) => { const { inputEl, el } = this; if (!inputEl) { return; } /** * Only active input mode if there is at * least one column that accepts numeric input. */ const hasInputColumn = el.querySelector('ion-picker-column.picker-column-numeric-input'); if (!hasInputColumn) { return; } /** * If columnEl is undefined then * it is assumed that all numeric pickers * are eligible for text entry. * (i.e. hour and minute columns) */ this.useInputMode = true; this.inputModeColumn = columnEl; /** * Users with a keyboard and mouse can * activate input mode where the input is * focused as well as when it is not focused, * so we need to make sure we clean up any * old listeners. */ if (focusInput) { if (this.destroyKeypressListener) { this.destroyKeypressListener(); this.destroyKeypressListener = undefined; } inputEl.focus(); } else { // TODO FW-5900 Use keydown instead el.addEventListener('keypress', this.onKeyPress); this.destroyKeypressListener = () => { el.removeEventListener('keypress', this.onKeyPress); }; } this.emitInputModeChange(); }; this.onKeyPress = (ev) => { const { inputEl } = this; if (!inputEl) { return; } const parsedValue = parseInt(ev.key, 10); /** * Only numbers should be allowed */ if (!Number.isNaN(parsedValue)) { inputEl.value += ev.key; this.onInputChange(); } }; this.selectSingleColumn = () => { const { inputEl, inputModeColumn, singleColumnSearchTimeout } = this; if (!inputEl || !inputModeColumn) { return; } const options = Array.from(inputModeColumn.querySelectorAll('ion-picker-column-option')).filter((el) => el.disabled !== true); /** * If users pause for a bit, the search * value should be reset similar to how a *