123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721 |
- /*!
- * (C) Ionic http://ionicframework.com - MIT License
- */
- 'use strict';
- Object.defineProperty(exports, '__esModule', { value: true });
- const index = require('./index-2e236a04.js');
- const index$2 = require('./index-31b07b9c.js');
- const helpers = require('./helpers-8a48fdea.js');
- const index$1 = require('./index-cc858e97.js');
- const dir = require('./dir-94c21456.js');
- const theme = require('./theme-d1c573d2.js');
- const ionicGlobal = require('./ionic-global-6dea5a96.js');
- function getDecimalPlaces(n) {
- if (!helpers.isSafeNumber(n))
- return 0;
- if (n % 1 === 0)
- return 0;
- return n.toString().split('.')[1].length;
- }
- /**
- * Fixes floating point rounding errors in a result by rounding
- * to the same specificity, or number of decimal places (*not*
- * significant figures) as provided reference numbers. If multiple
- * references are provided, the highest number of decimal places
- * between them will be used.
- *
- * The main use case is when numbers x and y are added to produce n,
- * but x and y are floats, so n may have rounding errors (such as
- * 3.1000000004 instead of 3.1). As long as only addition/subtraction
- * occurs between x and y, the specificity of the result will never
- * increase, so x and y should be passed in as the references.
- *
- * If multiplication, division, or other operations were used to
- * calculate n, the rounded result may have less specificity than
- * desired. For example, 1 / 3 = 0.33333(...), but
- * roundToMaxDecimalPlaces((1 / 3), 1, 3) will return 0, since both
- * 1 and 3 are whole numbers.
- *
- * Note that extremely precise reference numbers may lead to rounding
- * errors not being trimmed, due to the error result having the same or
- * fewer decimal places as the reference(s). This is acceptable as we
- * would not be able to tell the difference between a rounding error
- * and correct value in this case, but it does mean there is an implicit
- * precision limit. If precision that high is needed, it is recommended
- * to use a third party data type designed to handle floating point
- * errors instead.
- *
- * @param n The number to round.
- * @param references Number(s) used to calculate n, or that should otherwise
- * be used as a reference for the desired specificity.
- */
- function roundToMaxDecimalPlaces(n, ...references) {
- if (!helpers.isSafeNumber(n))
- return 0;
- const maxPlaces = Math.max(...references.map((r) => getDecimalPlaces(r)));
- return Number(n.toFixed(maxPlaces));
- }
- const rangeIosCss = ":host{--knob-handle-size:calc(var(--knob-size) * 2);display:-ms-flexbox;display:flex;position:relative;-ms-flex:3;flex:3;-ms-flex-align:center;align-items:center;font-family:var(--ion-font-family, inherit);-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;z-index:2}:host(.range-disabled){pointer-events:none}::slotted(ion-label){-ms-flex:initial;flex:initial}::slotted(ion-icon[slot]){font-size:24px}.range-slider{position:relative;-ms-flex:1;flex:1;width:100%;height:var(--height);contain:size layout style;cursor:-webkit-grab;cursor:grab;-ms-touch-action:pan-y;touch-action:pan-y}:host(.range-pressed) .range-slider{cursor:-webkit-grabbing;cursor:grabbing}.range-pin{position:absolute;background:var(--ion-color-base);color:var(--ion-color-contrast);text-align:center;-webkit-box-sizing:border-box;box-sizing:border-box}.range-knob-handle{top:calc((var(--height) - var(--knob-handle-size)) / 2);-webkit-margin-start:calc(0px - var(--knob-handle-size) / 2);margin-inline-start:calc(0px - var(--knob-handle-size) / 2);display:-ms-flexbox;display:flex;position:absolute;-ms-flex-pack:center;justify-content:center;width:var(--knob-handle-size);height:var(--knob-handle-size);text-align:center}.range-knob-handle{inset-inline-start:0}:host-context([dir=rtl]) .range-knob-handle{left:unset}[dir=rtl] .range-knob-handle{left:unset}@supports selector(:dir(rtl)){.range-knob-handle:dir(rtl){left:unset}}.range-knob-handle:active,.range-knob-handle:focus{outline:none}.range-bar-container{border-radius:var(--bar-border-radius);top:calc((var(--height) - var(--bar-height)) / 2);position:absolute;width:100%;height:var(--bar-height)}.range-bar-container{inset-inline-start:0}:host-context([dir=rtl]) .range-bar-container{left:unset}[dir=rtl] .range-bar-container{left:unset}@supports selector(:dir(rtl)){.range-bar-container:dir(rtl){left:unset}}.range-bar{border-radius:var(--bar-border-radius);position:absolute;width:100%;height:var(--bar-height);background:var(--bar-background);pointer-events:none}.range-knob{border-radius:var(--knob-border-radius);top:calc(50% - var(--knob-size) / 2);position:absolute;width:var(--knob-size);height:var(--knob-size);background:var(--knob-background);-webkit-box-shadow:var(--knob-box-shadow);box-shadow:var(--knob-box-shadow);z-index:2;pointer-events:none}.range-knob{inset-inline-start:calc(50% - var(--knob-size) / 2)}:host-context([dir=rtl]) .range-knob{left:unset}[dir=rtl] .range-knob{left:unset}@supports selector(:dir(rtl)){.range-knob:dir(rtl){left:unset}}:host(.range-pressed) .range-bar-active{will-change:left, right}:host(.in-item){width:100%}:host([slot=start]),:host([slot=end]){width:auto}:host(.in-item) ::slotted(ion-label){-ms-flex-item-align:center;align-self:center}.range-wrapper{display:-ms-flexbox;display:flex;position:relative;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center;height:inherit}::slotted([slot=label]){max-width:200px;text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.label-text-wrapper-hidden{display:none}.native-wrapper{display:-ms-flexbox;display:flex;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center}:host(.range-label-placement-start) .range-wrapper{-ms-flex-direction:row;flex-direction:row}:host(.range-label-placement-start) .label-text-wrapper{-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:16px;margin-inline-end:16px;margin-top:0;margin-bottom:0}:host(.range-label-placement-end) .range-wrapper{-ms-flex-direction:row-reverse;flex-direction:row-reverse}:host(.range-label-placement-end) .label-text-wrapper{-webkit-margin-start:16px;margin-inline-start:16px;-webkit-margin-end:0;margin-inline-end:0;margin-top:0;margin-bottom:0}:host(.range-label-placement-fixed) .label-text-wrapper{-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:16px;margin-inline-end:16px;margin-top:0;margin-bottom:0}:host(.range-label-placement-fixed) .label-text-wrapper{-ms-flex:0 0 100px;flex:0 0 100px;width:100px;min-width:100px;max-width:200px}:host(.range-label-placement-stacked) .range-wrapper{-ms-flex-direction:column;flex-direction:column;-ms-flex-align:stretch;align-items:stretch}:host(.range-label-placement-stacked) .label-text-wrapper{-webkit-transform-origin:left top;transform-origin:left top;-webkit-transform:scale(0.75);transform:scale(0.75);margin-left:0;margin-right:0;margin-bottom:16px;max-width:calc(100% / 0.75)}:host-context([dir=rtl]):host(.range-label-placement-stacked) .label-text-wrapper,:host-context([dir=rtl]).range-label-placement-stacked .label-text-wrapper{-webkit-transform-origin:right top;transform-origin:right top}@supports selector(:dir(rtl)){:host(.range-label-placement-stacked:dir(rtl)) .label-text-wrapper{-webkit-transform-origin:right top;transform-origin:right top}}:host(.in-item.range-label-placement-stacked) .label-text-wrapper{margin-top:10px;margin-bottom:16px}:host(.in-item.range-label-placement-stacked) .native-wrapper{margin-bottom:0px}:host{--knob-border-radius:50%;--knob-background:#ffffff;--knob-box-shadow:0px 0.5px 4px rgba(0, 0, 0, 0.12), 0px 6px 13px rgba(0, 0, 0, 0.12);--knob-size:26px;--bar-height:4px;--bar-background:var(--ion-color-step-900, var(--ion-background-color-step-900, #e6e6e6));--bar-background-active:var(--ion-color-primary, #0054e9);--bar-border-radius:2px;--height:42px}:host(.range-item-start-adjustment){-webkit-padding-start:24px;padding-inline-start:24px}:host(.range-item-end-adjustment){-webkit-padding-end:24px;padding-inline-end:24px}:host(.ion-color) .range-bar-active,:host(.ion-color) .range-tick-active{background:var(--ion-color-base)}::slotted([slot=start]){-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:16px;margin-inline-end:16px;margin-top:0;margin-bottom:0}::slotted([slot=end]){-webkit-margin-start:16px;margin-inline-start:16px;-webkit-margin-end:0;margin-inline-end:0;margin-top:0;margin-bottom:0}:host(.range-has-pin:not(.range-label-placement-stacked)){padding-top:calc(8px + 0.75rem)}:host(.range-has-pin.range-label-placement-stacked) .label-text-wrapper{margin-bottom:calc(8px + 0.75rem)}.range-bar-active{bottom:0;width:auto;background:var(--bar-background-active)}.range-bar-active.has-ticks{border-radius:0;-webkit-margin-start:-2px;margin-inline-start:-2px;-webkit-margin-end:-2px;margin-inline-end:-2px}.range-tick{-webkit-margin-start:-2px;margin-inline-start:-2px;border-radius:0;position:absolute;top:17px;width:4px;height:8px;background:var(--ion-color-step-900, var(--ion-background-color-step-900, #e6e6e6));pointer-events:none}.range-tick-active{background:var(--bar-background-active)}.range-pin{-webkit-transform:translate3d(0, 100%, 0) scale(0.01);transform:translate3d(0, 100%, 0) scale(0.01);-webkit-padding-start:8px;padding-inline-start:8px;-webkit-padding-end:8px;padding-inline-end:8px;padding-top:8px;padding-bottom:8px;min-width:28px;-webkit-transition:-webkit-transform 120ms ease;transition:-webkit-transform 120ms ease;transition:transform 120ms ease;transition:transform 120ms ease, -webkit-transform 120ms ease;background:transparent;color:var(--ion-text-color, #000);font-size:0.75rem;text-align:center}.range-knob-pressed .range-pin,.range-knob-handle.ion-focused .range-pin{-webkit-transform:translate3d(0, calc(-100% + 11px), 0) scale(1);transform:translate3d(0, calc(-100% + 11px), 0) scale(1)}:host(.range-disabled){opacity:0.3}";
- const IonRangeIosStyle0 = rangeIosCss;
- const rangeMdCss = ":host{--knob-handle-size:calc(var(--knob-size) * 2);display:-ms-flexbox;display:flex;position:relative;-ms-flex:3;flex:3;-ms-flex-align:center;align-items:center;font-family:var(--ion-font-family, inherit);-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;z-index:2}:host(.range-disabled){pointer-events:none}::slotted(ion-label){-ms-flex:initial;flex:initial}::slotted(ion-icon[slot]){font-size:24px}.range-slider{position:relative;-ms-flex:1;flex:1;width:100%;height:var(--height);contain:size layout style;cursor:-webkit-grab;cursor:grab;-ms-touch-action:pan-y;touch-action:pan-y}:host(.range-pressed) .range-slider{cursor:-webkit-grabbing;cursor:grabbing}.range-pin{position:absolute;background:var(--ion-color-base);color:var(--ion-color-contrast);text-align:center;-webkit-box-sizing:border-box;box-sizing:border-box}.range-knob-handle{top:calc((var(--height) - var(--knob-handle-size)) / 2);-webkit-margin-start:calc(0px - var(--knob-handle-size) / 2);margin-inline-start:calc(0px - var(--knob-handle-size) / 2);display:-ms-flexbox;display:flex;position:absolute;-ms-flex-pack:center;justify-content:center;width:var(--knob-handle-size);height:var(--knob-handle-size);text-align:center}.range-knob-handle{inset-inline-start:0}:host-context([dir=rtl]) .range-knob-handle{left:unset}[dir=rtl] .range-knob-handle{left:unset}@supports selector(:dir(rtl)){.range-knob-handle:dir(rtl){left:unset}}.range-knob-handle:active,.range-knob-handle:focus{outline:none}.range-bar-container{border-radius:var(--bar-border-radius);top:calc((var(--height) - var(--bar-height)) / 2);position:absolute;width:100%;height:var(--bar-height)}.range-bar-container{inset-inline-start:0}:host-context([dir=rtl]) .range-bar-container{left:unset}[dir=rtl] .range-bar-container{left:unset}@supports selector(:dir(rtl)){.range-bar-container:dir(rtl){left:unset}}.range-bar{border-radius:var(--bar-border-radius);position:absolute;width:100%;height:var(--bar-height);background:var(--bar-background);pointer-events:none}.range-knob{border-radius:var(--knob-border-radius);top:calc(50% - var(--knob-size) / 2);position:absolute;width:var(--knob-size);height:var(--knob-size);background:var(--knob-background);-webkit-box-shadow:var(--knob-box-shadow);box-shadow:var(--knob-box-shadow);z-index:2;pointer-events:none}.range-knob{inset-inline-start:calc(50% - var(--knob-size) / 2)}:host-context([dir=rtl]) .range-knob{left:unset}[dir=rtl] .range-knob{left:unset}@supports selector(:dir(rtl)){.range-knob:dir(rtl){left:unset}}:host(.range-pressed) .range-bar-active{will-change:left, right}:host(.in-item){width:100%}:host([slot=start]),:host([slot=end]){width:auto}:host(.in-item) ::slotted(ion-label){-ms-flex-item-align:center;align-self:center}.range-wrapper{display:-ms-flexbox;display:flex;position:relative;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center;height:inherit}::slotted([slot=label]){max-width:200px;text-overflow:ellipsis;white-space:nowrap;overflow:hidden}.label-text-wrapper-hidden{display:none}.native-wrapper{display:-ms-flexbox;display:flex;-ms-flex-positive:1;flex-grow:1;-ms-flex-align:center;align-items:center}:host(.range-label-placement-start) .range-wrapper{-ms-flex-direction:row;flex-direction:row}:host(.range-label-placement-start) .label-text-wrapper{-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:16px;margin-inline-end:16px;margin-top:0;margin-bottom:0}:host(.range-label-placement-end) .range-wrapper{-ms-flex-direction:row-reverse;flex-direction:row-reverse}:host(.range-label-placement-end) .label-text-wrapper{-webkit-margin-start:16px;margin-inline-start:16px;-webkit-margin-end:0;margin-inline-end:0;margin-top:0;margin-bottom:0}:host(.range-label-placement-fixed) .label-text-wrapper{-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:16px;margin-inline-end:16px;margin-top:0;margin-bottom:0}:host(.range-label-placement-fixed) .label-text-wrapper{-ms-flex:0 0 100px;flex:0 0 100px;width:100px;min-width:100px;max-width:200px}:host(.range-label-placement-stacked) .range-wrapper{-ms-flex-direction:column;flex-direction:column;-ms-flex-align:stretch;align-items:stretch}:host(.range-label-placement-stacked) .label-text-wrapper{-webkit-transform-origin:left top;transform-origin:left top;-webkit-transform:scale(0.75);transform:scale(0.75);margin-left:0;margin-right:0;margin-bottom:16px;max-width:calc(100% / 0.75)}:host-context([dir=rtl]):host(.range-label-placement-stacked) .label-text-wrapper,:host-context([dir=rtl]).range-label-placement-stacked .label-text-wrapper{-webkit-transform-origin:right top;transform-origin:right top}@supports selector(:dir(rtl)){:host(.range-label-placement-stacked:dir(rtl)) .label-text-wrapper{-webkit-transform-origin:right top;transform-origin:right top}}:host(.in-item.range-label-placement-stacked) .label-text-wrapper{margin-top:10px;margin-bottom:16px}:host(.in-item.range-label-placement-stacked) .native-wrapper{margin-bottom:0px}:host{--knob-border-radius:50%;--knob-background:var(--bar-background-active);--knob-box-shadow:none;--knob-size:18px;--bar-height:2px;--bar-background:rgba(var(--ion-color-primary-rgb, 0, 84, 233), 0.26);--bar-background-active:var(--ion-color-primary, #0054e9);--bar-border-radius:0;--height:42px;--pin-background:var(--ion-color-primary, #0054e9);--pin-color:var(--ion-color-primary-contrast, #fff)}::slotted(:not(ion-icon)[slot=start]),::slotted(:not(ion-icon)[slot=end]),.native-wrapper{font-size:0.75rem}:host(.range-item-start-adjustment){-webkit-padding-start:18px;padding-inline-start:18px}:host(.range-item-end-adjustment){-webkit-padding-end:18px;padding-inline-end:18px}:host(.ion-color) .range-bar{background:rgba(var(--ion-color-base-rgb), 0.26)}:host(.ion-color) .range-bar-active,:host(.ion-color) .range-knob,:host(.ion-color) .range-knob::before,:host(.ion-color) .range-pin,:host(.ion-color) .range-pin::before,:host(.ion-color) .range-tick{background:var(--ion-color-base);color:var(--ion-color-contrast)}::slotted([slot=start]){-webkit-margin-start:0;margin-inline-start:0;-webkit-margin-end:14px;margin-inline-end:14px;margin-top:0;margin-bottom:0}::slotted([slot=end]){-webkit-margin-start:14px;margin-inline-start:14px;-webkit-margin-end:0;margin-inline-end:0;margin-top:0;margin-bottom:0}:host(.range-has-pin:not(.range-label-placement-stacked)){padding-top:1.75rem}:host(.range-has-pin.range-label-placement-stacked) .label-text-wrapper{margin-bottom:1.75rem}.range-bar-active{bottom:0;width:auto;background:var(--bar-background-active)}.range-knob{-webkit-transform:scale(0.67);transform:scale(0.67);-webkit-transition-duration:120ms;transition-duration:120ms;-webkit-transition-property:background-color, border, -webkit-transform;transition-property:background-color, border, -webkit-transform;transition-property:transform, background-color, border;transition-property:transform, background-color, border, -webkit-transform;-webkit-transition-timing-function:ease;transition-timing-function:ease;z-index:2}.range-knob::before{border-radius:50%;position:absolute;width:var(--knob-size);height:var(--knob-size);-webkit-transform:scale(1);transform:scale(1);-webkit-transition:0.267s cubic-bezier(0, 0, 0.58, 1);transition:0.267s cubic-bezier(0, 0, 0.58, 1);background:var(--knob-background);content:\"\";opacity:0.13;pointer-events:none}.range-knob::before{inset-inline-start:0}.range-tick{position:absolute;top:calc((var(--height) - var(--bar-height)) / 2);width:var(--bar-height);height:var(--bar-height);background:var(--bar-background-active);z-index:1;pointer-events:none}.range-tick-active{background:transparent}.range-pin{padding-left:0;padding-right:0;padding-top:8px;padding-bottom:8px;border-radius:50%;-webkit-transform:translate3d(0, 0, 0) scale(0.01);transform:translate3d(0, 0, 0) scale(0.01);display:-ms-flexbox;display:flex;-ms-flex-align:center;align-items:center;-ms-flex-pack:center;justify-content:center;width:1.75rem;height:1.75rem;-webkit-transition:background 120ms ease, -webkit-transform 120ms ease;transition:background 120ms ease, -webkit-transform 120ms ease;transition:transform 120ms ease, background 120ms ease;transition:transform 120ms ease, background 120ms ease, -webkit-transform 120ms ease;background:var(--pin-background);color:var(--pin-color)}.range-pin::before{bottom:-1px;-webkit-margin-start:-13px;margin-inline-start:-13px;border-radius:50% 50% 50% 0;position:absolute;width:26px;height:26px;-webkit-transform:rotate(-45deg);transform:rotate(-45deg);-webkit-transition:background 120ms ease;transition:background 120ms ease;background:var(--pin-background);content:\"\";z-index:-1}.range-pin::before{inset-inline-start:50%}:host-context([dir=rtl]) .range-pin::before{left:unset}[dir=rtl] .range-pin::before{left:unset}@supports selector(:dir(rtl)){.range-pin::before:dir(rtl){left:unset}}.range-knob-pressed .range-pin,.range-knob-handle.ion-focused .range-pin{-webkit-transform:translate3d(0, calc(-100% + 4px), 0) scale(1);transform:translate3d(0, calc(-100% + 4px), 0) scale(1)}@media (any-hover: hover){.range-knob-handle:hover .range-knob:before{-webkit-transform:scale(2);transform:scale(2);opacity:0.13}}.range-knob-handle.ion-activated .range-knob:before,.range-knob-handle.ion-focused .range-knob:before,.range-knob-handle.range-knob-pressed .range-knob:before{-webkit-transform:scale(2);transform:scale(2)}.range-knob-handle.ion-focused .range-knob::before{opacity:0.13}.range-knob-handle.ion-activated .range-knob::before,.range-knob-handle.range-knob-pressed .range-knob::before{opacity:0.25}:host(:not(.range-has-pin)) .range-knob-pressed .range-knob,:host(:not(.range-has-pin)) .range-knob-handle.ion-focused .range-knob{-webkit-transform:scale(1);transform:scale(1)}:host(.range-disabled) .range-bar-active,:host(.range-disabled) .range-bar,:host(.range-disabled) .range-tick{background-color:var(--ion-color-step-250, var(--ion-background-color-step-250, #bfbfbf))}:host(.range-disabled) .range-knob{-webkit-transform:scale(0.55);transform:scale(0.55);outline:5px solid #fff;background-color:var(--ion-color-step-250, var(--ion-background-color-step-250, #bfbfbf))}:host(.range-disabled) .label-text-wrapper,:host(.range-disabled) ::slotted([slot=start]),:host(.range-disabled) ::slotted([slot=end]){opacity:0.38}";
- const IonRangeMdStyle0 = rangeMdCss;
- const Range = class {
- constructor(hostRef) {
- index.registerInstance(this, hostRef);
- this.ionChange = index.createEvent(this, "ionChange", 7);
- this.ionInput = index.createEvent(this, "ionInput", 7);
- this.ionFocus = index.createEvent(this, "ionFocus", 7);
- this.ionBlur = index.createEvent(this, "ionBlur", 7);
- this.ionKnobMoveStart = index.createEvent(this, "ionKnobMoveStart", 7);
- this.ionKnobMoveEnd = index.createEvent(this, "ionKnobMoveEnd", 7);
- this.rangeId = `ion-r-${rangeIds++}`;
- this.didLoad = false;
- this.noUpdate = false;
- this.hasFocus = false;
- this.inheritedAttributes = {};
- this.contentEl = null;
- this.initialContentScrollY = true;
- /**
- * Compares two RangeValue inputs to determine if they are different.
- *
- * @param newVal - The new value.
- * @param oldVal - The old value.
- * @returns `true` if the values are different, `false` otherwise.
- */
- this.compareValues = (newVal, oldVal) => {
- if (typeof newVal === 'object' && typeof oldVal === 'object') {
- return newVal.lower !== oldVal.lower || newVal.upper !== oldVal.upper;
- }
- return newVal !== oldVal;
- };
- this.clampBounds = (value) => {
- return helpers.clamp(this.min, value, this.max);
- };
- this.ensureValueInBounds = (value) => {
- if (this.dualKnobs) {
- return {
- lower: this.clampBounds(value.lower),
- upper: this.clampBounds(value.upper),
- };
- }
- else {
- return this.clampBounds(value);
- }
- };
- this.setupGesture = async () => {
- const rangeSlider = this.rangeSlider;
- if (rangeSlider) {
- this.gesture = (await Promise.resolve().then(function () { return require('./index-ee07ed59.js'); })).createGesture({
- el: rangeSlider,
- gestureName: 'range',
- gesturePriority: 100,
- /**
- * Provide a threshold since the drag movement
- * might be a user scrolling the view.
- * If this is true, then the range
- * should not move.
- */
- threshold: 10,
- onStart: () => this.onStart(),
- onMove: (ev) => this.onMove(ev),
- onEnd: (ev) => this.onEnd(ev),
- });
- this.gesture.enable(!this.disabled);
- }
- };
- this.handleKeyboard = (knob, isIncrease) => {
- const { ensureValueInBounds } = this;
- let step = this.step;
- step = step > 0 ? step : 1;
- step = step / (this.max - this.min);
- if (!isIncrease) {
- step *= -1;
- }
- if (knob === 'A') {
- this.ratioA = helpers.clamp(0, this.ratioA + step, 1);
- }
- else {
- this.ratioB = helpers.clamp(0, this.ratioB + step, 1);
- }
- this.ionKnobMoveStart.emit({ value: ensureValueInBounds(this.value) });
- this.updateValue();
- this.emitValueChange();
- this.ionKnobMoveEnd.emit({ value: ensureValueInBounds(this.value) });
- };
- this.onBlur = () => {
- if (this.hasFocus) {
- this.hasFocus = false;
- this.ionBlur.emit();
- }
- };
- this.onFocus = () => {
- if (!this.hasFocus) {
- this.hasFocus = true;
- this.ionFocus.emit();
- }
- };
- this.ratioA = 0;
- this.ratioB = 0;
- this.pressedKnob = undefined;
- this.color = undefined;
- this.debounce = undefined;
- this.name = this.rangeId;
- this.label = undefined;
- this.dualKnobs = false;
- this.min = 0;
- this.max = 100;
- this.pin = false;
- this.pinFormatter = (value) => Math.round(value);
- this.snaps = false;
- this.step = 1;
- this.ticks = true;
- this.activeBarStart = undefined;
- this.disabled = false;
- this.value = 0;
- this.labelPlacement = 'start';
- }
- debounceChanged() {
- const { ionInput, debounce, originalIonInput } = this;
- /**
- * If debounce is undefined, we have to manually revert the ionInput emitter in case
- * debounce used to be set to a number. Otherwise, the event would stay debounced.
- */
- this.ionInput = debounce === undefined ? originalIonInput !== null && originalIonInput !== void 0 ? originalIonInput : ionInput : helpers.debounceEvent(ionInput, debounce);
- }
- minChanged(newValue) {
- if (!helpers.isSafeNumber(newValue)) {
- this.min = 0;
- }
- if (!this.noUpdate) {
- this.updateRatio();
- }
- }
- maxChanged(newValue) {
- if (!helpers.isSafeNumber(newValue)) {
- this.max = 100;
- }
- if (!this.noUpdate) {
- this.updateRatio();
- }
- }
- stepChanged(newValue) {
- if (!helpers.isSafeNumber(newValue)) {
- this.step = 1;
- }
- }
- activeBarStartChanged() {
- const { activeBarStart } = this;
- if (activeBarStart !== undefined) {
- if (activeBarStart > this.max) {
- index$1.printIonWarning(`[ion-range] - The value of activeBarStart (${activeBarStart}) is greater than the max (${this.max}). Valid values are greater than or equal to the min value and less than or equal to the max value.`, this.el);
- this.activeBarStart = this.max;
- }
- else if (activeBarStart < this.min) {
- index$1.printIonWarning(`[ion-range] - The value of activeBarStart (${activeBarStart}) is less than the min (${this.min}). Valid values are greater than or equal to the min value and less than or equal to the max value.`, this.el);
- this.activeBarStart = this.min;
- }
- }
- }
- disabledChanged() {
- if (this.gesture) {
- this.gesture.enable(!this.disabled);
- }
- }
- valueChanged(newValue, oldValue) {
- const valuesChanged = this.compareValues(newValue, oldValue);
- if (valuesChanged) {
- this.ionInput.emit({ value: this.value });
- }
- if (!this.noUpdate) {
- this.updateRatio();
- }
- }
- componentWillLoad() {
- /**
- * If user has custom ID set then we should
- * not assign the default incrementing ID.
- */
- if (this.el.hasAttribute('id')) {
- this.rangeId = this.el.getAttribute('id');
- }
- this.inheritedAttributes = helpers.inheritAriaAttributes(this.el);
- // If min, max, or step are not safe, set them to 0, 100, and 1, respectively.
- // Each watch does this, but not before the initial load.
- this.min = helpers.isSafeNumber(this.min) ? this.min : 0;
- this.max = helpers.isSafeNumber(this.max) ? this.max : 100;
- this.step = helpers.isSafeNumber(this.step) ? this.step : 1;
- }
- componentDidLoad() {
- this.originalIonInput = this.ionInput;
- this.setupGesture();
- this.updateRatio();
- this.didLoad = true;
- }
- connectedCallback() {
- var _a;
- this.updateRatio();
- this.debounceChanged();
- this.disabledChanged();
- this.activeBarStartChanged();
- /**
- * If we have not yet rendered
- * ion-range, then rangeSlider is not defined.
- * But if we are moving ion-range via appendChild,
- * then rangeSlider will be defined.
- */
- if (this.didLoad) {
- this.setupGesture();
- }
- const ionContent = index$2.findClosestIonContent(this.el);
- this.contentEl = (_a = ionContent === null || ionContent === void 0 ? void 0 : ionContent.querySelector('.ion-content-scroll-host')) !== null && _a !== void 0 ? _a : ionContent;
- }
- disconnectedCallback() {
- if (this.gesture) {
- this.gesture.destroy();
- this.gesture = undefined;
- }
- }
- getValue() {
- var _a;
- const value = (_a = this.value) !== null && _a !== void 0 ? _a : 0;
- if (this.dualKnobs) {
- if (typeof value === 'object') {
- return value;
- }
- return {
- lower: 0,
- upper: value,
- };
- }
- else {
- if (typeof value === 'object') {
- return value.upper;
- }
- return value;
- }
- }
- /**
- * Emits an `ionChange` event.
- *
- * This API should be called for user committed changes.
- * This API should not be used for external value changes.
- */
- emitValueChange() {
- this.value = this.ensureValueInBounds(this.value);
- this.ionChange.emit({ value: this.value });
- }
- /**
- * The value should be updated on touch end or
- * when the component is being dragged.
- * This follows the native behavior of mobile devices.
- *
- * For example: When the user lifts their finger from the
- * screen after tapping the bar or dragging the bar or knob.
- */
- onStart() {
- this.ionKnobMoveStart.emit({ value: this.ensureValueInBounds(this.value) });
- }
- /**
- * The value should be updated while dragging the
- * bar or knob.
- *
- * While the user is dragging, the view
- * should not scroll. This is to prevent the user from
- * feeling disoriented while dragging.
- *
- * The user can scroll on the view if the knob or
- * bar is not being dragged.
- *
- * @param detail The details of the gesture event.
- */
- onMove(detail) {
- const { contentEl, pressedKnob } = this;
- const currentX = detail.currentX;
- /**
- * Since the user is dragging on the bar or knob, the view should not scroll.
- *
- * This only needs to be done once.
- */
- if (contentEl && this.pressedKnob === undefined) {
- this.initialContentScrollY = index$2.disableContentScrollY(contentEl);
- }
- /**
- * The `pressedKnob` can be undefined if the user just
- * started dragging the knob.
- *
- * This is necessary to determine which knob the user is dragging,
- * especially when it's a dual knob.
- * Plus, it determines when to apply certain styles.
- *
- * This only needs to be done once since the knob won't change
- * while the user is dragging.
- */
- if (pressedKnob === undefined) {
- this.setPressedKnob(currentX);
- }
- this.update(currentX);
- }
- /**
- * The value should be updated on touch end:
- * - When the user lifts their finger from the screen after
- * tapping the bar.
- *
- * @param detail The details of the gesture or mouse event.
- */
- onEnd(detail) {
- var _a;
- const { contentEl, initialContentScrollY } = this;
- const currentX = (_a = detail.currentX) !== null && _a !== void 0 ? _a : detail.clientX;
- /**
- * The `pressedKnob` can be undefined if the user never
- * dragged the knob. They just tapped on the bar.
- *
- * This is necessary to determine which knob the user is changing,
- * especially when it's a dual knob.
- * Plus, it determines when to apply certain styles.
- */
- if (this.pressedKnob === undefined) {
- this.setPressedKnob(currentX);
- }
- /**
- * The user is no longer dragging the bar or
- * knob (if they were dragging it).
- *
- * The user can now scroll on the view in the next gesture event.
- */
- if (contentEl && this.pressedKnob !== undefined) {
- index$2.resetContentScrollY(contentEl, initialContentScrollY);
- }
- // update the active knob's position
- this.update(currentX);
- /**
- * Reset the pressed knob to undefined since the user
- * may start dragging a different knob in the next gesture event.
- */
- this.pressedKnob = undefined;
- this.emitValueChange();
- this.ionKnobMoveEnd.emit({ value: this.ensureValueInBounds(this.value) });
- }
- update(currentX) {
- // figure out where the pointer is currently at
- // update the knob being interacted with
- const rect = this.rect;
- let ratio = helpers.clamp(0, (currentX - rect.left) / rect.width, 1);
- if (dir.isRTL(this.el)) {
- ratio = 1 - ratio;
- }
- if (this.snaps) {
- // snaps the ratio to the current value
- ratio = valueToRatio(ratioToValue(ratio, this.min, this.max, this.step), this.min, this.max);
- }
- // update which knob is pressed
- if (this.pressedKnob === 'A') {
- this.ratioA = ratio;
- }
- else {
- this.ratioB = ratio;
- }
- // Update input value
- this.updateValue();
- }
- setPressedKnob(currentX) {
- const rect = (this.rect = this.rangeSlider.getBoundingClientRect());
- // figure out which knob they started closer to
- let ratio = helpers.clamp(0, (currentX - rect.left) / rect.width, 1);
- if (dir.isRTL(this.el)) {
- ratio = 1 - ratio;
- }
- this.pressedKnob = !this.dualKnobs || Math.abs(this.ratioA - ratio) < Math.abs(this.ratioB - ratio) ? 'A' : 'B';
- this.setFocus(this.pressedKnob);
- }
- get valA() {
- return ratioToValue(this.ratioA, this.min, this.max, this.step);
- }
- get valB() {
- return ratioToValue(this.ratioB, this.min, this.max, this.step);
- }
- get ratioLower() {
- if (this.dualKnobs) {
- return Math.min(this.ratioA, this.ratioB);
- }
- const { activeBarStart } = this;
- if (activeBarStart == null) {
- return 0;
- }
- return valueToRatio(activeBarStart, this.min, this.max);
- }
- get ratioUpper() {
- if (this.dualKnobs) {
- return Math.max(this.ratioA, this.ratioB);
- }
- return this.ratioA;
- }
- updateRatio() {
- const value = this.getValue();
- const { min, max } = this;
- if (this.dualKnobs) {
- this.ratioA = valueToRatio(value.lower, min, max);
- this.ratioB = valueToRatio(value.upper, min, max);
- }
- else {
- this.ratioA = valueToRatio(value, min, max);
- }
- }
- updateValue() {
- this.noUpdate = true;
- const { valA, valB } = this;
- this.value = !this.dualKnobs
- ? valA
- : {
- lower: Math.min(valA, valB),
- upper: Math.max(valA, valB),
- };
- this.noUpdate = false;
- }
- setFocus(knob) {
- if (this.el.shadowRoot) {
- const knobEl = this.el.shadowRoot.querySelector(knob === 'A' ? '.range-knob-a' : '.range-knob-b');
- if (knobEl) {
- knobEl.focus();
- }
- }
- }
- /**
- * Returns true if content was passed to the "start" slot
- */
- get hasStartSlotContent() {
- return this.el.querySelector('[slot="start"]') !== null;
- }
- /**
- * Returns true if content was passed to the "end" slot
- */
- get hasEndSlotContent() {
- return this.el.querySelector('[slot="end"]') !== null;
- }
- get hasLabel() {
- return this.label !== undefined || this.el.querySelector('[slot="label"]') !== null;
- }
- renderRangeSlider() {
- var _a;
- const { min, max, step, handleKeyboard, pressedKnob, disabled, pin, ratioLower, ratioUpper, pinFormatter, inheritedAttributes, } = this;
- let barStart = `${ratioLower * 100}%`;
- let barEnd = `${100 - ratioUpper * 100}%`;
- const rtl = dir.isRTL(this.el);
- const start = rtl ? 'right' : 'left';
- const end = rtl ? 'left' : 'right';
- const tickStyle = (tick) => {
- return {
- [start]: tick[start],
- };
- };
- if (this.dualKnobs === false) {
- /**
- * When the value is less than the activeBarStart or the min value,
- * the knob will display at the start of the active bar.
- */
- if (this.valA < ((_a = this.activeBarStart) !== null && _a !== void 0 ? _a : this.min)) {
- /**
- * Sets the bar positions relative to the upper and lower limits.
- * Converts the ratio values into percentages, used as offsets for left/right styles.
- *
- * The ratioUpper refers to the knob position on the bar.
- * The ratioLower refers to the end position of the active bar (the value).
- */
- barStart = `${ratioUpper * 100}%`;
- barEnd = `${100 - ratioLower * 100}%`;
- }
- else {
- /**
- * Otherwise, the knob will display at the end of the active bar.
- *
- * The ratioLower refers to the start position of the active bar (the value).
- * The ratioUpper refers to the knob position on the bar.
- */
- barStart = `${ratioLower * 100}%`;
- barEnd = `${100 - ratioUpper * 100}%`;
- }
- }
- const barStyle = {
- [start]: barStart,
- [end]: barEnd,
- };
- const ticks = [];
- if (this.snaps && this.ticks) {
- for (let value = min; value <= max; value += step) {
- const ratio = valueToRatio(value, min, max);
- const ratioMin = Math.min(ratioLower, ratioUpper);
- const ratioMax = Math.max(ratioLower, ratioUpper);
- const tick = {
- ratio,
- /**
- * Sets the tick mark as active when the tick is between the min bounds and the knob.
- * When using activeBarStart, the tick mark will be active between the knob and activeBarStart.
- */
- active: ratio >= ratioMin && ratio <= ratioMax,
- };
- tick[start] = `${ratio * 100}%`;
- ticks.push(tick);
- }
- }
- return (index.h("div", { class: "range-slider", ref: (rangeEl) => (this.rangeSlider = rangeEl),
- /**
- * Since the gesture has a threshold, the value
- * won't change until the user has dragged past
- * the threshold. This is to prevent the range
- * from moving when the user is scrolling.
- *
- * This results in the value not being updated
- * and the event emitters not being triggered
- * if the user taps on the range. This is why
- * we need to listen for the "pointerUp" event.
- */
- onPointerUp: (ev) => {
- /**
- * If the user drags the knob on the web
- * version (does not occur on mobile),
- * the "pointerUp" event will be triggered
- * along with the gesture's events.
- * This leads to duplicate events.
- *
- * By checking if the pressedKnob is undefined,
- * we can determine if the "pointerUp" event was
- * triggered by a tap or a drag. If it was
- * dragged, the pressedKnob will be defined.
- */
- if (this.pressedKnob === undefined) {
- this.onStart();
- this.onEnd(ev);
- }
- } }, ticks.map((tick) => (index.h("div", { style: tickStyle(tick), role: "presentation", class: {
- 'range-tick': true,
- 'range-tick-active': tick.active,
- }, part: tick.active ? 'tick-active' : 'tick' }))), index.h("div", { class: "range-bar-container" }, index.h("div", { class: "range-bar", role: "presentation", part: "bar" }), index.h("div", { class: {
- 'range-bar': true,
- 'range-bar-active': true,
- 'has-ticks': ticks.length > 0,
- }, role: "presentation", style: barStyle, part: "bar-active" })), renderKnob(rtl, {
- knob: 'A',
- pressed: pressedKnob === 'A',
- value: this.valA,
- ratio: this.ratioA,
- pin,
- pinFormatter,
- disabled,
- handleKeyboard,
- min,
- max,
- inheritedAttributes,
- }), this.dualKnobs &&
- renderKnob(rtl, {
- knob: 'B',
- pressed: pressedKnob === 'B',
- value: this.valB,
- ratio: this.ratioB,
- pin,
- pinFormatter,
- disabled,
- handleKeyboard,
- min,
- max,
- inheritedAttributes,
- })));
- }
- render() {
- const { disabled, el, hasLabel, rangeId, pin, pressedKnob, labelPlacement, label } = this;
- const inItem = theme.hostContext('ion-item', el);
- /**
- * If there is no start content then the knob at
- * the min value will be cut off by the item margin.
- */
- const hasStartContent = (hasLabel && (labelPlacement === 'start' || labelPlacement === 'fixed')) || this.hasStartSlotContent;
- const needsStartAdjustment = inItem && !hasStartContent;
- /**
- * If there is no end content then the knob at
- * the max value will be cut off by the item margin.
- */
- const hasEndContent = (hasLabel && labelPlacement === 'end') || this.hasEndSlotContent;
- const needsEndAdjustment = inItem && !hasEndContent;
- const mode = ionicGlobal.getIonMode(this);
- helpers.renderHiddenInput(true, el, this.name, JSON.stringify(this.getValue()), disabled);
- return (index.h(index.Host, { key: '3e065039ee048f1f70d97dba5dae98fa1315d867', onFocusin: this.onFocus, onFocusout: this.onBlur, id: rangeId, class: theme.createColorClasses(this.color, {
- [mode]: true,
- 'in-item': inItem,
- 'range-disabled': disabled,
- 'range-pressed': pressedKnob !== undefined,
- 'range-has-pin': pin,
- [`range-label-placement-${labelPlacement}`]: true,
- 'range-item-start-adjustment': needsStartAdjustment,
- 'range-item-end-adjustment': needsEndAdjustment,
- }) }, index.h("label", { key: '27ff22842c9ea79a1b9495302b926f70c9080a95', class: "range-wrapper", id: "range-label" }, index.h("div", { key: 'da1f9784be02dfe87d2fef34931d8b7f2148189e', class: {
- 'label-text-wrapper': true,
- 'label-text-wrapper-hidden': !hasLabel,
- }, part: "label" }, label !== undefined ? index.h("div", { class: "label-text" }, label) : index.h("slot", { name: "label" })), index.h("div", { key: '4389bf30b08214f5b5917fc30976b38f7bcdd29b', class: "native-wrapper" }, index.h("slot", { key: 'ad1b2745f8b061ee189617bb5c567e4f1d02250c', name: "start" }), this.renderRangeSlider(), index.h("slot", { key: 'c6dec9e843e232af2a5f16a0f8ee56439c545d7a', name: "end" })))));
- }
- get el() { return index.getElement(this); }
- static get watchers() { return {
- "debounce": ["debounceChanged"],
- "min": ["minChanged"],
- "max": ["maxChanged"],
- "step": ["stepChanged"],
- "activeBarStart": ["activeBarStartChanged"],
- "disabled": ["disabledChanged"],
- "value": ["valueChanged"]
- }; }
- };
- const renderKnob = (rtl, { knob, value, ratio, min, max, disabled, pressed, pin, handleKeyboard, pinFormatter, inheritedAttributes }) => {
- const start = rtl ? 'right' : 'left';
- const knobStyle = () => {
- const style = {};
- style[start] = `${ratio * 100}%`;
- return style;
- };
- // The aria label should be preferred over visible text if both are specified
- const ariaLabel = inheritedAttributes['aria-label'];
- return (index.h("div", { onKeyDown: (ev) => {
- const key = ev.key;
- if (key === 'ArrowLeft' || key === 'ArrowDown') {
- handleKeyboard(knob, false);
- ev.preventDefault();
- ev.stopPropagation();
- }
- else if (key === 'ArrowRight' || key === 'ArrowUp') {
- handleKeyboard(knob, true);
- ev.preventDefault();
- ev.stopPropagation();
- }
- }, class: {
- 'range-knob-handle': true,
- 'range-knob-a': knob === 'A',
- 'range-knob-b': knob === 'B',
- 'range-knob-pressed': pressed,
- 'range-knob-min': value === min,
- 'range-knob-max': value === max,
- 'ion-activatable': true,
- 'ion-focusable': true,
- }, style: knobStyle(), role: "slider", tabindex: disabled ? -1 : 0, "aria-label": ariaLabel !== undefined ? ariaLabel : null, "aria-labelledby": ariaLabel === undefined ? 'range-label' : null, "aria-valuemin": min, "aria-valuemax": max, "aria-disabled": disabled ? 'true' : null, "aria-valuenow": value }, pin && (index.h("div", { class: "range-pin", role: "presentation", part: "pin" }, pinFormatter(value))), index.h("div", { class: "range-knob", role: "presentation", part: "knob" })));
- };
- const ratioToValue = (ratio, min, max, step) => {
- let value = (max - min) * ratio;
- if (step > 0) {
- // round to nearest multiple of step, then add min
- value = Math.round(value / step) * step + min;
- }
- const clampedValue = helpers.clamp(min, value, max);
- return roundToMaxDecimalPlaces(clampedValue, min, max, step);
- };
- const valueToRatio = (value, min, max) => {
- return helpers.clamp(0, (value - min) / (max - min), 1);
- };
- let rangeIds = 0;
- Range.style = {
- ios: IonRangeIosStyle0,
- md: IonRangeMdStyle0
- };
- exports.ion_range = Range;
|