@descope/web-components-ui 1.44.0 → 1.46.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/dist/cjs/index.cjs.js +58 -24
  2. package/dist/cjs/index.cjs.js.map +1 -1
  3. package/dist/index.esm.js +469 -232
  4. package/dist/index.esm.js.map +1 -1
  5. package/dist/umd/3620.js +1 -1
  6. package/dist/umd/3620.js.map +1 -1
  7. package/dist/umd/5348.js +2 -0
  8. package/dist/umd/5348.js.map +1 -0
  9. package/dist/umd/6477.js +149 -0
  10. package/dist/umd/6477.js.map +1 -0
  11. package/dist/umd/9365.js +1 -1
  12. package/dist/umd/9365.js.map +1 -1
  13. package/dist/umd/DescopeDev.js +1 -1
  14. package/dist/umd/DescopeDev.js.map +1 -1
  15. package/dist/umd/descope-hybrid-field-index-js.js +3 -3
  16. package/dist/umd/descope-hybrid-field-index-js.js.map +1 -1
  17. package/dist/umd/descope-passcode-index-js.js +1 -1
  18. package/dist/umd/descope-passcode-index-js.js.map +1 -1
  19. package/dist/umd/index.js +1 -1
  20. package/dist/umd/index.js.map +1 -1
  21. package/dist/umd/phone-fields-descope-phone-field-descope-phone-field-internal-index-js.js +1 -1
  22. package/dist/umd/phone-fields-descope-phone-field-descope-phone-field-internal-index-js.js.map +1 -1
  23. package/dist/umd/phone-fields-descope-phone-field-index-js.js +1 -1
  24. package/dist/umd/phone-fields-descope-phone-field-index-js.js.map +1 -1
  25. package/dist/umd/phone-fields-descope-phone-input-box-field-descope-phone-input-box-internal-index-js.js +2 -2
  26. package/dist/umd/phone-fields-descope-phone-input-box-field-descope-phone-input-box-internal-index-js.js.map +1 -1
  27. package/dist/umd/phone-fields-descope-phone-input-box-field-index-js.js +2 -113
  28. package/dist/umd/phone-fields-descope-phone-input-box-field-index-js.js.LICENSE.txt +0 -6
  29. package/dist/umd/phone-fields-descope-phone-input-box-field-index-js.js.map +1 -1
  30. package/package.json +7 -6
  31. package/src/components/descope-hybrid-field/HybridFieldClass.js +6 -0
  32. package/src/components/descope-passcode/PasscodeClass.js +2 -0
  33. package/src/components/phone-fields/descope-phone-field/PhoneFieldClass.js +10 -2
  34. package/src/components/phone-fields/descope-phone-field/descope-phone-field-internal/PhoneFieldInternal.js +229 -126
  35. package/src/components/phone-fields/descope-phone-input-box-field/PhoneFieldInputBoxClass.js +42 -24
  36. package/src/components/phone-fields/descope-phone-input-box-field/descope-phone-input-box-internal/PhoneFieldInternalInputBox.js +179 -83
  37. package/src/components/phone-fields/descope-phone-input-box-field/index.js +0 -1
  38. package/src/components/phone-fields/helpers.js +7 -0
  39. package/src/mixins/index.js +1 -0
  40. package/src/mixins/inputOverrideValidConstraints.js +12 -0
  41. package/dist/umd/6424.js +0 -149
  42. package/dist/umd/6424.js.map +0 -1
  43. /package/dist/umd/{6424.js.LICENSE.txt → 6477.js.LICENSE.txt} +0 -0
package/dist/index.esm.js CHANGED
@@ -15,7 +15,7 @@ import '@vaadin/number-field';
15
15
  import '@vaadin/password-field';
16
16
  import MarkdownIt from 'markdown-it';
17
17
  import '@vaadin/text-area';
18
- import { parsePhoneNumberFromString } from 'libphonenumber-js/min';
18
+ import parsePhoneNumberFromString$1, { parsePhoneNumberFromString, AsYouType } from 'libphonenumber-js/min';
19
19
  import '@vaadin/grid';
20
20
  import { GridSortColumn } from '@vaadin/grid/vaadin-grid-sort-column';
21
21
  import { GridSelectionColumn } from '@vaadin/grid/vaadin-grid-selection-column';
@@ -2244,6 +2244,19 @@ const externalInputMixin =
2244
2244
  }
2245
2245
  };
2246
2246
 
2247
+ const inputOverrideValidConstraintsMixin = (superclass) =>
2248
+ class InputOverrideValidConstraintsMixinClass extends superclass {
2249
+ init() {
2250
+ super.init?.();
2251
+
2252
+ // vaadin uses `validConstraints` (required, pattern, minlength, maxlength) to determine if it should validate
2253
+ // the input or not. We want to override this behavior, so we can enforce validation even if these attributes are not present.
2254
+ if (this.baseElement._hasValidConstraints) {
2255
+ this.baseElement._hasValidConstraints = () => true;
2256
+ }
2257
+ }
2258
+ };
2259
+
2247
2260
  const createBaseInputClass$1 = (...args) =>
2248
2261
  compose$1(
2249
2262
  inputValidationMixin$1,
@@ -8163,6 +8176,7 @@ const PasscodeClass = compose$1(
8163
8176
  },
8164
8177
  }),
8165
8178
  draggableMixin$1,
8179
+ inputOverrideValidConstraintsMixin,
8166
8180
  composedProxyInputMixin$1({ proxyProps: ['value', 'selectionStart'] }),
8167
8181
  componentNameValidationMixin$1,
8168
8182
  customMixin$c
@@ -10210,7 +10224,7 @@ const parsePhoneNumber = (val) => {
10210
10224
 
10211
10225
  const componentName$P = getComponentName$1('phone-field-internal');
10212
10226
 
10213
- const commonAttrs$1 = ['disabled', 'size', 'bordered', 'readonly', 'allow-alphanumeric-input'];
10227
+ const commonAttrs$1 = ['disabled', 'size', 'bordered', 'readonly'];
10214
10228
  const countryAttrs = ['country-input-placeholder', 'default-code', 'restrict-countries'];
10215
10229
  const phoneAttrs = ['phone-input-placeholder', 'maxlength', 'autocomplete', 'name'];
10216
10230
  const labelTypeAttrs = ['label-type', 'country-input-label', 'label'];
@@ -10229,6 +10243,8 @@ let PhoneFieldInternal$1 = class PhoneFieldInternal extends BaseInputClass$8 {
10229
10243
  return [].concat(BaseInputClass$8.observedAttributes || [], inputRelatedAttrs$1);
10230
10244
  }
10231
10245
 
10246
+ #ayt;
10247
+
10232
10248
  constructor() {
10233
10249
  super();
10234
10250
 
@@ -10245,200 +10261,328 @@ let PhoneFieldInternal$1 = class PhoneFieldInternal extends BaseInputClass$8 {
10245
10261
  </div>
10246
10262
  `;
10247
10263
 
10248
- this.countryCodeInput = this.querySelector('descope-combo-box');
10249
- this.phoneNumberInput = this.querySelector('descope-text-field');
10250
- this.inputs = [this.countryCodeInput, this.phoneNumberInput];
10264
+ this.comboBox = this.querySelector('descope-combo-box');
10265
+ this.textField = this.querySelector('descope-text-field');
10266
+
10267
+ this.inputs = [this.comboBox, this.textField];
10251
10268
 
10252
- forwardAttrs$1(this, this.countryCodeInput, { includeAttrs: ['label-type'] });
10253
- forwardAttrs$1(this, this.phoneNumberInput, { includeAttrs: ['label-type', 'required'] });
10269
+ forwardAttrs$1(this, this.comboBox, { includeAttrs: ['label-type'] });
10270
+ forwardAttrs$1(this, this.textField, { includeAttrs: ['label-type', 'required'] });
10254
10271
 
10255
10272
  // override combo box setter to display dialCode value in input
10256
- this.countryCodeInput.customValueTransformFn = (val) => {
10273
+ this.comboBox.customValueTransformFn = (val) => {
10257
10274
  const [, dialCode] = val?.split?.(' ') || [];
10258
10275
  return dialCode;
10259
10276
  };
10260
10277
  }
10261
10278
 
10279
+ // exposed from main component
10262
10280
  get countryCodeInputData() {
10263
- return this.countryCodeInput.items;
10281
+ return this.comboBox.items;
10282
+ }
10283
+
10284
+ // exposed from main component
10285
+ get countryCodeValue() {
10286
+ return this.comboBox.shadowRoot.querySelector('input').value;
10287
+ }
10288
+
10289
+ // exposed from main component
10290
+ get phoneNumberInputEle() {
10291
+ return this.textField.shadowRoot.querySelector('input');
10264
10292
  }
10265
10293
 
10266
10294
  get allowAlphanumericInput() {
10267
10295
  return this.getAttribute('allow-alphanumeric-input') === 'true';
10268
10296
  }
10269
10297
 
10298
+ get defaultCode() {
10299
+ return this.getAttribute('default-code');
10300
+ }
10301
+
10302
+ get selectionStart() {
10303
+ return this.textField.selectionStart;
10304
+ }
10305
+
10306
+ get minLength() {
10307
+ return parseInt(this.getAttribute('minlength'), 10) || 0;
10308
+ }
10309
+
10310
+ get selectedCountryCode() {
10311
+ return this.comboBox?.selectedItem?.getAttribute('data-country-code');
10312
+ }
10313
+
10314
+ get restrictCountries() {
10315
+ const attr = this.getAttribute('restrict-countries');
10316
+ return attr?.split(',').filter(Boolean) || [];
10317
+ }
10318
+
10319
+ get isFormatValue() {
10320
+ return this.getAttribute('format-value') === 'true';
10321
+ }
10322
+
10323
+ // `strict validation` enforces value parsing with libphonenumber-js
10324
+ get isStrictValidation() {
10325
+ return this.getAttribute('strict-validation') === 'true';
10326
+ }
10327
+
10270
10328
  get value() {
10271
- if (!this.phoneNumberValue) {
10329
+ if (!this.comboBox.value || !this.textField.value) {
10272
10330
  return '';
10273
10331
  }
10274
- return `${this.countryCodeInput.value}-${this.phoneNumberInput.value}`;
10332
+
10333
+ const [countryCode, phoneNumber] = parsePhoneNumber(
10334
+ `${this.comboBox.value}-${this.textField.value}`
10335
+ );
10336
+
10337
+ return `${countryCode || this.comboBox.value}-${phoneNumber || this.textField.value}`;
10275
10338
  }
10276
10339
 
10277
10340
  set value(val) {
10278
- const [countryCode, phoneNumber] = parsePhoneNumber(val);
10341
+ const [countryCode, nationalNumber] = parsePhoneNumber(val);
10279
10342
 
10280
10343
  this.#setCountryCode(countryCode);
10281
- this.#setPhoneNumber(phoneNumber);
10344
+ this.#setNationalNumber(nationalNumber);
10282
10345
  }
10283
10346
 
10284
- setSelectionRange(...args) {
10285
- this.phoneNumberInput.setSelectionRange(...args);
10286
- }
10347
+ init() {
10348
+ this.addEventListener('focus', (e) => {
10349
+ // we want to ignore focus events we are dispatching
10350
+ if (e.isTrusted) this.inputs[1].focus();
10351
+ });
10287
10352
 
10288
- get selectionStart() {
10289
- return this.phoneNumberInput.selectionStart;
10353
+ super.init?.();
10354
+
10355
+ this.#initInputs();
10290
10356
  }
10291
10357
 
10292
- #setCountryCode(val) {
10293
- if (val) {
10294
- const countryCodeItem = this.getCountryByDialCode(val);
10295
- if (countryCodeItem) {
10296
- this.countryCodeInput.selectedItem = countryCodeItem;
10297
- }
10298
- } else {
10299
- this.countryCodeInput.selectedItem = undefined;
10358
+ getValidity() {
10359
+ const countryCode = this.comboBox.value;
10360
+ const nationalNumer = this.textField.value;
10361
+
10362
+ const isEmpty = !countryCode || !nationalNumer;
10363
+ const isValidLength = nationalNumer && nationalNumer.length >= this.minLength;
10364
+
10365
+ if (this.isRequired && isEmpty) {
10366
+ return { valueMissing: true };
10300
10367
  }
10301
- }
10302
10368
 
10303
- #setPhoneNumber(val) {
10304
- if (this.phoneNumberInput.value === val) {
10305
- return;
10369
+ if (this.value) {
10370
+ if (!isValidLength) {
10371
+ return { tooShort: true };
10372
+ }
10373
+
10374
+ if (this.isStrictValidation && !this.#isValidParsedValue()) {
10375
+ return { patternMismatch: true };
10376
+ }
10306
10377
  }
10307
10378
 
10308
- this.phoneNumberInput.value = val;
10379
+ return {};
10309
10380
  }
10310
10381
 
10311
- get phoneNumberValue() {
10312
- return this.phoneNumberInput.value;
10382
+ setSelectionRange(...args) {
10383
+ this.textField.setSelectionRange(...args);
10313
10384
  }
10314
10385
 
10315
- get countryCodeValue() {
10316
- return this.countryCodeInput.shadowRoot.querySelector('input').value;
10317
- }
10386
+ attributeChangedCallback(attrName, oldValue, newValue) {
10387
+ super.attributeChangedCallback(attrName, oldValue, newValue);
10318
10388
 
10319
- get phoneNumberInputEle() {
10320
- return this.phoneNumberInput.shadowRoot.querySelector('input');
10389
+ if (oldValue !== newValue) {
10390
+ if (attrName === 'default-code' && newValue) {
10391
+ this.#handleDefaultCountryCode(newValue);
10392
+ } else if (inputRelatedAttrs$1.includes(attrName)) {
10393
+ const attr = mapAttrs$1[attrName] || attrName;
10394
+
10395
+ if (commonAttrs$1.includes(attrName)) {
10396
+ this.inputs.forEach((input) => input.setAttribute(attr, newValue));
10397
+ } else if (countryAttrs.includes(attrName)) {
10398
+ this.comboBox.setAttribute(attr, newValue);
10399
+ } else if (phoneAttrs.includes(attrName)) {
10400
+ this.textField.setAttribute(attr, newValue);
10401
+ }
10402
+ }
10403
+
10404
+ if (labelTypeAttrs.includes(attrName)) {
10405
+ this.#handleLabelTypeAttrs(attrName, newValue);
10406
+ }
10407
+
10408
+ if (attrName === 'restrict-countries') {
10409
+ this.#updateComboBoxItems(this.restrictCountries);
10410
+ }
10411
+ }
10321
10412
  }
10322
10413
 
10323
- get minLength() {
10324
- return parseInt(this.getAttribute('minlength'), 10) || 0;
10414
+ #initInputs() {
10415
+ // Sanitize phone input value to filter everything but digits
10416
+ this.textField.addEventListener('input', (e) => {
10417
+ if (!this.allowAlphanumericInput) {
10418
+ const telDigitsRegExp = /^\d$/;
10419
+ const sanitizedInput = e.target.value
10420
+ .split('')
10421
+ .filter((char) => telDigitsRegExp.test(char))
10422
+ .join('');
10423
+ e.target.value = sanitizedInput;
10424
+ }
10425
+ });
10426
+
10427
+ this.handleFocusEventsDispatching(this.inputs);
10428
+ this.handleInputEventDispatching();
10429
+
10430
+ // verify country code item against phone number pattern and replace if needed and country is allowed
10431
+ // (e.g. +1 can be US or CA, depending on the pattern)
10432
+ this.addEventListener('input', this.#handleSameCountryCodes.bind(this));
10325
10433
  }
10326
10434
 
10327
- getValidity() {
10328
- const hasCode = this.countryCodeInput.value;
10329
- const hasPhone = this.phoneNumberInput.value;
10330
- const emptyValue = !hasCode || !hasPhone;
10331
- const hasMinPhoneLength =
10332
- this.phoneNumberInput.value?.length && this.phoneNumberInput.value.length < this.minLength;
10435
+ #setCountryCode(val) {
10436
+ if (!val || val === this.selectedCountryCode) return;
10333
10437
 
10334
- if (this.isRequired && emptyValue) {
10335
- return { valueMissing: true };
10438
+ let countryCodeItem = undefined;
10439
+
10440
+ if (this.value) {
10441
+ // try to parse the phone number, and set country code item according to actual dial code (e.g. `+1` can be `US` or `CA`)
10442
+ const code = this.#getCountryCodeByPhoneNumber(`${val}-${this.textField.value}`);
10443
+ countryCodeItem = this.#getCountryByCodeId(code);
10336
10444
  }
10337
- if (hasMinPhoneLength) {
10338
- return { tooShort: true };
10445
+
10446
+ // in case country code item does not exist (for example: Parsed code is CA for +1 - but Canada is not allowed)
10447
+ // then use the first option with same dial code (e.g. in that case - `US` for +1)
10448
+ if (!countryCodeItem) {
10449
+ countryCodeItem = this.#getCountryByDialCode(val);
10339
10450
  }
10340
- if (hasPhone && !hasCode) {
10341
- return { valueMissing: true };
10451
+
10452
+ // set country code item; in it doesn't exist in list - set `undefined`
10453
+ this.comboBox.selectedItem = countryCodeItem;
10454
+ }
10455
+
10456
+ #setNationalNumber(val) {
10457
+ if (this.isFormatValue) {
10458
+ val = this.#formatNationalNumber(val);
10459
+ }
10460
+
10461
+ if (this.textField.value !== val) {
10462
+ this.textField.value = val;
10342
10463
  }
10343
- return {};
10344
10464
  }
10345
10465
 
10346
- init() {
10347
- this.addEventListener('focus', (e) => {
10348
- // we want to ignore focus events we are dispatching
10349
- if (e.isTrusted) this.inputs[1].focus();
10350
- });
10466
+ #formatNationalNumber(nationalNumber = '') {
10467
+ // re-initialize AsYouType if country code is outdated
10468
+ if (!this.#ayt || this.#ayt.country !== this.selectedCountryCode) {
10469
+ this.#ayt = new AsYouType(this.selectedCountryCode);
10470
+ }
10351
10471
 
10352
- super.init?.();
10353
- this.initInputs();
10472
+ // reset previous AsYouType input
10473
+ this.#ayt.reset();
10474
+
10475
+ const formattedVal = this.#ayt.input(nationalNumber);
10476
+
10477
+ return formattedVal || nationalNumber;
10354
10478
  }
10355
10479
 
10356
- getRestrictedCountries() {
10357
- const attr = this.getAttribute('restrict-countries');
10358
- return attr ? attr.split(',') : [];
10480
+ #isValidParsedValue() {
10481
+ const parsed = parsePhoneNumberFromString(this.value);
10482
+
10483
+ return (
10484
+ parsed && // Parsed successfully (not undefined)
10485
+ parsed.isValid?.() && // Parsed object is valid
10486
+ parsed.country && // Parsed object with a country code
10487
+ this.#isAllowedCountry(parsed.country) // Parsed with allowed country code
10488
+ );
10489
+ }
10490
+
10491
+ #isAllowedCountry(countryCode) {
10492
+ if (!this.restrictCountries) {
10493
+ return true;
10494
+ }
10495
+
10496
+ return this.restrictCountries.includes(countryCode);
10359
10497
  }
10360
10498
 
10361
- getCountryByDialCode(countryDialCode) {
10362
- return this.countryCodeInput.items?.find((c) => c.getAttribute('data-id') === countryDialCode);
10499
+ // return country item by dial code `data-id` (e.g. `+1`)
10500
+ #getCountryByDialCode(dialCode) {
10501
+ return this.comboBox.items?.find((c) => c.getAttribute('data-id') === dialCode) || undefined;
10363
10502
  }
10364
10503
 
10365
- getCountryByCodeId(countryCode) {
10366
- return this.countryCodeInput.items?.find(
10367
- (c) => c.getAttribute('data-country-code') === countryCode
10368
- );
10504
+ // return country item by country code `data-country-code` (e.g. `US`)
10505
+ #getCountryByCodeId(countryCode) {
10506
+ return this.comboBox.items?.find((c) => c.getAttribute('data-country-code') === countryCode);
10507
+ }
10508
+
10509
+ #getCountryCodeByPhoneNumber(val) {
10510
+ if (!val) return undefined;
10511
+ const parsed = parsePhoneNumberFromString(val);
10512
+ if (!parsed?.country) return undefined;
10513
+ const foundCountryItem = this.#getCountryByCodeId(parsed.country);
10514
+ return foundCountryItem?.getAttribute('data-country-code');
10369
10515
  }
10370
10516
 
10371
- updateCountryCodeItems(restrictCountries) {
10517
+ #updateComboBoxItems(restrictCountries) {
10372
10518
  const items = restrictCountries.length
10373
10519
  ? CountryCodes.filter((c) => restrictCountries.includes(c.code))
10374
10520
  : CountryCodes;
10521
+
10375
10522
  this.querySelector('descope-combo-box').innerHTML = items
10376
10523
  .map((item) => comboBoxItem(item))
10377
10524
  .join('');
10378
10525
  }
10379
10526
 
10380
- handleDefaultCountryCode(countryCode) {
10381
- if (!this.countryCodeInput.value) {
10382
- const countryCodeItem = this.getCountryByCodeId(countryCode);
10527
+ #handleDefaultCountryCode(countryCode) {
10528
+ if (!this.comboBox.value) {
10529
+ const countryCodeItem = this.#getCountryByCodeId(countryCode);
10383
10530
  // When replacing the input component (inserting internal component into text-field) -
10384
10531
  // Vaadin resets the input's value. We use setTimeout in order to make sure this happens
10385
10532
  // after the reset.
10386
10533
  if (countryCodeItem) {
10387
10534
  setTimeout(() => {
10388
- this.countryCodeInput.selectedItem = countryCodeItem;
10535
+ this.comboBox.selectedItem = countryCodeItem;
10389
10536
  });
10390
10537
  }
10391
10538
  }
10392
10539
  }
10393
10540
 
10394
- initInputs() {
10395
- // Sanitize phone input value to filter everything but digits
10396
- this.phoneNumberInput.addEventListener('input', (e) => {
10397
- // we want to update only phoneNumberInput, and avoid triggering `set value`
10398
- e.stopPropagation();
10541
+ #handleSameCountryCodes() {
10542
+ if (!this.value) return;
10399
10543
 
10400
- if (!this.allowAlphanumericInput) {
10401
- const telDigitsRegExp = /^\d$/;
10402
- const sanitizedInput = e.target.value
10403
- .split('')
10404
- .filter((char) => telDigitsRegExp.test(char))
10405
- .join('');
10406
- e.target.value = sanitizedInput;
10407
- }
10408
- });
10544
+ const country = this.#getCountryCodeByPhoneNumber(this.value);
10409
10545
 
10410
- this.handleFocusEventsDispatching(this.inputs);
10411
- this.handleInputEventDispatching();
10546
+ if (!country) return;
10547
+
10548
+ // re-set country code if needed (same coutnry code for different countries, e.g. +1 for US or CA)
10549
+ if (this.selectedCountryCode !== country) {
10550
+ const foundCountryItem = this.#getCountryByCodeId(country);
10551
+ // if found country is defined in country list then set it, otherwise - clear phone number
10552
+ if (foundCountryItem) {
10553
+ this.comboBox.selectedItem = foundCountryItem;
10554
+ }
10555
+ }
10412
10556
  }
10413
10557
 
10414
- handleLabelTypeAttrs(attrName, newValue) {
10558
+ #handleLabelTypeAttrs(attrName, newValue) {
10415
10559
  // set or remove label attributes from inner text/combo components on `label-type` change
10416
10560
  const attr = mapAttrs$1[attrName] || attrName;
10417
10561
 
10418
10562
  if (attrName === 'label-type') {
10419
- this.onLabelTypeChange(newValue);
10563
+ this.#handleLabelTypeChange(newValue);
10420
10564
  }
10421
10565
  // on inner components label attr change - set label attributes for text/combo components
10422
10566
  // only if label-type is `floating`
10423
10567
  else if (this.getAttribute('label-type') === 'floating') {
10424
10568
  if (attrName === 'country-input-label') {
10425
- this.countryCodeInput.setAttribute(attr, newValue);
10569
+ this.comboBox.setAttribute(attr, newValue);
10426
10570
  } else if (attrName === 'label') {
10427
- this.phoneNumberInput.setAttribute(attr, newValue);
10571
+ this.textField.setAttribute(attr, newValue);
10428
10572
  }
10429
10573
  }
10430
10574
  }
10431
10575
 
10432
- onLabelTypeChange(newValue) {
10576
+ #handleLabelTypeChange(newValue) {
10433
10577
  if (newValue === 'floating') {
10434
10578
  // on change to `floating` label - set inner components `label` and `placeholder`
10435
- this.countryCodeInput.setAttribute('label', this.getAttribute('country-input-label') || '');
10436
- this.countryCodeInput.setAttribute(
10579
+ this.comboBox.setAttribute('label', this.getAttribute('country-input-label') || '');
10580
+ this.comboBox.setAttribute(
10437
10581
  'placeholder',
10438
10582
  this.getAttribute('country-input-placeholder') || ''
10439
10583
  );
10440
- this.phoneNumberInput.setAttribute('label', this.getAttribute('label') || '');
10441
- this.phoneNumberInput.setAttribute(
10584
+ this.textField.setAttribute('label', this.getAttribute('label') || '');
10585
+ this.textField.setAttribute(
10442
10586
  'placeholder',
10443
10587
  this.getAttribute('phone-input-placeholder') || ''
10444
10588
  );
@@ -10447,34 +10591,6 @@ let PhoneFieldInternal$1 = class PhoneFieldInternal extends BaseInputClass$8 {
10447
10591
  this.inputs.forEach((input) => input.removeAttribute('label'));
10448
10592
  }
10449
10593
  }
10450
-
10451
- attributeChangedCallback(attrName, oldValue, newValue) {
10452
- super.attributeChangedCallback(attrName, oldValue, newValue);
10453
-
10454
- if (oldValue !== newValue) {
10455
- if (attrName === 'default-code' && newValue) {
10456
- this.handleDefaultCountryCode(newValue);
10457
- } else if (inputRelatedAttrs$1.includes(attrName)) {
10458
- const attr = mapAttrs$1[attrName] || attrName;
10459
-
10460
- if (commonAttrs$1.includes(attrName)) {
10461
- this.inputs.forEach((input) => input.setAttribute(attr, newValue));
10462
- } else if (countryAttrs.includes(attrName)) {
10463
- this.countryCodeInput.setAttribute(attr, newValue);
10464
- } else if (phoneAttrs.includes(attrName)) {
10465
- this.phoneNumberInput.setAttribute(attr, newValue);
10466
- }
10467
- }
10468
-
10469
- if (labelTypeAttrs.includes(attrName)) {
10470
- this.handleLabelTypeAttrs(attrName, newValue);
10471
- }
10472
-
10473
- if (attrName === 'restrict-countries') {
10474
- this.updateCountryCodeItems(this.getRestrictedCountries());
10475
- }
10476
- }
10477
- }
10478
10594
  };
10479
10595
 
10480
10596
  customElements.define(componentName$P, PhoneFieldInternal$1);
@@ -10510,7 +10626,6 @@ const customMixin$a = (superclass) =>
10510
10626
  includeAttrs: [
10511
10627
  'size',
10512
10628
  'bordered',
10513
- 'invalid',
10514
10629
  'minlength',
10515
10630
  'maxlength',
10516
10631
  'default-code',
@@ -10523,6 +10638,8 @@ const customMixin$a = (superclass) =>
10523
10638
  'label',
10524
10639
  'label-type',
10525
10640
  'allow-alphanumeric-input',
10641
+ 'format-value',
10642
+ 'strict-validation',
10526
10643
  ],
10527
10644
  });
10528
10645
  }
@@ -10671,6 +10788,7 @@ const PhoneFieldClass = compose$1(
10671
10788
  },
10672
10789
  }),
10673
10790
  draggableMixin$1,
10791
+ inputOverrideValidConstraintsMixin,
10674
10792
  composedProxyInputMixin$1({ proxyProps: ['value', 'selectionStart'] }),
10675
10793
  customMixin$a
10676
10794
  )(
@@ -10783,19 +10901,23 @@ const getCountryByCodeId = (countryCode) => {
10783
10901
  return CountryCodes.find((c) => c.code === countryCode)?.dialCode;
10784
10902
  };
10785
10903
 
10904
+ const matchingParenthesis = (val) => {
10905
+ const openParenMatches = val.match(/\(/g);
10906
+ const closeParenMatches = val.match(/\)/g);
10907
+ return openParenMatches?.length === closeParenMatches?.length;
10908
+ };
10909
+
10786
10910
  const componentName$N = getComponentName$1('phone-field-internal-input-box');
10787
10911
 
10788
10912
  const observedAttributes$3 = [
10789
10913
  'disabled',
10790
10914
  'size',
10791
- 'bordered',
10792
- 'invalid',
10793
10915
  'readonly',
10794
10916
  'phone-input-placeholder',
10795
10917
  'name',
10918
+ 'maxlength',
10796
10919
  'autocomplete',
10797
10920
  'label-type',
10798
- 'allow-alphanumeric-input',
10799
10921
  ];
10800
10922
  const mapAttrs = {
10801
10923
  'phone-input-placeholder': 'placeholder',
@@ -10808,145 +10930,242 @@ class PhoneFieldInternal extends BaseInputClass$7 {
10808
10930
  return [].concat(BaseInputClass$7.observedAttributes || [], observedAttributes$3);
10809
10931
  }
10810
10932
 
10933
+ #ayt;
10934
+
10811
10935
  constructor() {
10812
10936
  super();
10813
10937
 
10814
10938
  this.innerHTML = `
10815
10939
  <div>
10816
- <descope-text-field tabindex="1"></descope-text-field>
10940
+ <descope-text-field tabindex="1" type="tel" bordered="false"></descope-text-field>
10817
10941
  </div>
10818
10942
  `;
10819
10943
 
10820
- this.phoneNumberInput = this.querySelector('descope-text-field');
10944
+ this.textField = this.querySelector('descope-text-field');
10821
10945
  }
10822
10946
 
10823
- get defaultCountryCode() {
10947
+ // notice: this function is exposed in parent component
10948
+ get phoneNumberInputEle() {
10949
+ return this.textField.shadowRoot.querySelector('input');
10950
+ }
10951
+
10952
+ get defaultDialCode() {
10824
10953
  return getCountryByCodeId(this.getAttribute('default-code'));
10825
10954
  }
10826
10955
 
10827
- get hasDefaultCode() {
10828
- return !!this.getAttribute('default-code');
10956
+ get defaultCode() {
10957
+ return this.getAttribute('default-code');
10829
10958
  }
10830
10959
 
10831
10960
  get allowAlphanumericInput() {
10832
10961
  return this.getAttribute('allow-alphanumeric-input') === 'true';
10833
10962
  }
10834
10963
 
10835
- get value() {
10836
- if (!this.phoneNumberValue) {
10837
- return '';
10838
- }
10964
+ get minLength() {
10965
+ return parseInt(this.getAttribute('minlength'), 10) || 0;
10966
+ }
10839
10967
 
10840
- if (this.hasDefaultCode) {
10841
- // we want to transform phone numbers to a valid {dialCode}-{phoneNumber} format
10842
- // e.g.:
10843
- // +972-12345 => +972-12345
10844
- // 972-12345 => +972-12345
10845
- // 12345 => +972-12345
10846
- //
10847
- // we also want to handle any extra dash if added in the start of the phone number
10848
- // e.g.:
10849
- // +972--12345 => +972-12345
10850
- const pattern = new RegExp(`\\+?${parseInt(this.defaultCountryCode, 10)}--?`);
10851
- return `${this.defaultCountryCode}-${this.phoneNumberInput.value.replace(pattern, '')}`;
10852
- }
10968
+ get maxLength() {
10969
+ return parseInt(this.getAttribute('maxlength'), 10) || 50;
10970
+ }
10853
10971
 
10854
- return this.phoneNumberInput.value;
10972
+ get restrictCountries() {
10973
+ return this.getAttribute('restrict-countries')?.split(',').filter(Boolean) || [];
10855
10974
  }
10856
10975
 
10857
- set value(val) {
10858
- this.phoneNumberInput.value = val;
10976
+ get isFormatValue() {
10977
+ return this.getAttribute('format-value') === 'true';
10859
10978
  }
10860
10979
 
10861
- get phoneNumberValue() {
10862
- return this.phoneNumberInput.value;
10980
+ get isStrictValidation() {
10981
+ return this.getAttribute('strict-validation') === 'true';
10863
10982
  }
10864
10983
 
10865
- get phoneNumberInputEle() {
10866
- return this.phoneNumberInput.shadowRoot.querySelector('input');
10984
+ get value() {
10985
+ if (!this.textField.value) return '';
10986
+
10987
+ if (!this.isStrictValidation) {
10988
+ return this.#nonParsedValue();
10989
+ }
10990
+
10991
+ const parsedVal = this.#parseWithCountryCode();
10992
+
10993
+ if (parsedVal?.country && parsedVal?.countryCallingCode && parsedVal?.nationalNumber) {
10994
+ return `+${[parsedVal?.countryCallingCode, parsedVal?.nationalNumber].join('-')}`;
10995
+ }
10996
+
10997
+ // if failed to parse or to find country code return text field value
10998
+ return this.textField.value;
10867
10999
  }
10868
11000
 
10869
- get minLength() {
10870
- return parseInt(this.getAttribute('minlength'), 10) || 0;
11001
+ set value(val) {
11002
+ this.textField.value = val;
10871
11003
  }
10872
11004
 
10873
- get maxLength() {
10874
- return parseInt(this.getAttribute('maxlength'), 10) || 50;
11005
+ init() {
11006
+ this.addEventListener('focus', (e) => {
11007
+ // We want to ignore focus events we are dispatching
11008
+ if (e.isTrusted) this.textField.focus();
11009
+ });
11010
+
11011
+ super.init?.();
11012
+
11013
+ this.textField.addEventListener('input', this.#onInput.bind(this));
11014
+ this.handleFocusEventsDispatching([this.textField]);
10875
11015
  }
10876
11016
 
10877
11017
  getValidity() {
10878
11018
  const validPhonePattern = /^\+?\d{1,4}-?(?:\d-?){1,15}$/;
10879
- const stripValue = this.value.replace(/\D/g, '');
11019
+ const stripValue = this.#sanitizeVal(this.textField.value);
10880
11020
 
10881
- if (this.isRequired && !this.value) {
11021
+ if (this.isRequired && !this.textField.value) {
10882
11022
  return { valueMissing: true };
10883
11023
  }
10884
11024
 
10885
- if (stripValue.length < this.minLength) {
10886
- return { tooShort: true };
11025
+ if (this.textField.value) {
11026
+ if (stripValue.length < this.minLength) {
11027
+ return { tooShort: true };
11028
+ }
11029
+
11030
+ if (
11031
+ // has `strict-validation` and not properly parsed
11032
+ (this.isStrictValidation && this.textField.value && !this.#isValidParsedValue()) ||
11033
+ // if no `strict-validation` then conform with naive pattern
11034
+ (!this.isStrictValidation && this.textField.value && !validPhonePattern.test(this.value))
11035
+ ) {
11036
+ return { patternMismatch: true };
11037
+ }
10887
11038
  }
10888
11039
 
10889
- if (stripValue.length > this.maxLength) {
10890
- return { tooLong: true };
11040
+ return {};
11041
+ }
11042
+
11043
+ setSelectionRange(...args) {
11044
+ this.textField.setSelectionRange(...args);
11045
+ }
11046
+
11047
+ attributeChangedCallback(attrName, oldValue, newValue) {
11048
+ super.attributeChangedCallback(attrName, oldValue, newValue);
11049
+
11050
+ if (oldValue !== newValue && observedAttributes$3.includes(attrName)) {
11051
+ const attr = mapAttrs[attrName] || attrName;
11052
+ this.textField.setAttribute(attr, newValue);
10891
11053
  }
11054
+ }
10892
11055
 
10893
- if (this.value && !validPhonePattern.test(this.value)) {
10894
- return { patternMismatch: true };
11056
+ #onInput(e) {
11057
+ let sanitizedInput = this.#sanitizeInput(e.target.value);
11058
+
11059
+ if (this.isFormatValue && this.#canFormat(sanitizedInput)) {
11060
+ sanitizedInput = this.#formatPhoneNumber(sanitizedInput);
10895
11061
  }
10896
11062
 
10897
- return {};
11063
+ e.target.value = sanitizedInput;
10898
11064
  }
10899
11065
 
10900
- init() {
10901
- this.addEventListener('focus', (e) => {
10902
- // we want to ignore focus events we are dispatching
10903
- if (e.isTrusted) this.phoneNumberInput.focus();
10904
- });
11066
+ #nonParsedValue() {
11067
+ if (!this.defaultDialCode) {
11068
+ return this.textField.value;
11069
+ }
10905
11070
 
10906
- super.init?.();
10907
- this.initInputs();
11071
+ const nationalNumber = this.#trimDuplicateCountryCode(this.textField.value);
11072
+ const sanitizedVal = this.#sanitizeVal(nationalNumber);
11073
+
11074
+ return [this.defaultDialCode, sanitizedVal].join('-');
11075
+ }
11076
+
11077
+ #parseWithCountryCode() {
11078
+ if (this.defaultDialCode) {
11079
+ return parsePhoneNumberFromString$1(
11080
+ [this.defaultDialCode, this.#sanitizeVal(this.textField.value)].filter(Boolean).join('')
11081
+ );
11082
+ }
11083
+
11084
+ // if default-code or not parsed - try to extract country code from value
11085
+ return parsePhoneNumberFromString$1(this.textField.value);
11086
+ }
11087
+
11088
+ #sanitizeVal(val) {
11089
+ return val.replace(/\D/g, '');
10908
11090
  }
10909
11091
 
10910
- getCountryByDialCode(countryDialCode) {
10911
- return this.countryCodeInput.items?.find(
10912
- (c) => c.getAttribute('data-country-code') === countryDialCode
11092
+ #trimDuplicateCountryCode(val) {
11093
+ if (this.textField.value?.[0] === '+') {
11094
+ const dialCodePrefixPattern = new RegExp(`^\\${this.defaultDialCode}`);
11095
+ const trimmed = val.replace(dialCodePrefixPattern, '');
11096
+ return trimmed;
11097
+ }
11098
+ return val;
11099
+ }
11100
+
11101
+ #isValidParsedValue() {
11102
+ const parsed = parsePhoneNumberFromString$1(this.value);
11103
+ return (
11104
+ parsed && // parsed successfully (not undefined)
11105
+ parsed.isValid?.() && // Parsed object is valid
11106
+ parsed.country && // Parsed object with a country code
11107
+ this.#isAllowedCountry(parsed.country) && // Parsed with allowed country code
11108
+ (this.defaultCode ? this.defaultCode === parsed.country : true) // In case default country code is set validate parsed country matches it
10913
11109
  );
10914
11110
  }
10915
11111
 
10916
- initInputs() {
10917
- // Sanitize phone input value to filter everything but digits
10918
- this.phoneNumberInput.addEventListener('input', (e) => {
10919
- if (e.target.value.length === 1 && e.target.value === '-') {
10920
- e.target.value = '';
10921
- }
11112
+ #isAllowedCountry(countryCode) {
11113
+ if (!this.restrictCountries) {
11114
+ return true;
11115
+ }
10922
11116
 
10923
- e.target.value = e.target.value
10924
- .replace(/(?!^)\+/g, '')
10925
- .replace('--', '-')
10926
- .replace('+-', '+');
11117
+ return this.restrictCountries.includes(countryCode);
11118
+ }
10927
11119
 
10928
- let sanitizedInput = e.target.value;
10929
- if (!this.allowAlphanumericInput) {
10930
- const telDigitsRegExp = /^[+\d-]+$/;
10931
- sanitizedInput = e.target.value
10932
- .split('')
10933
- .filter((char) => telDigitsRegExp.test(char))
10934
- .join('');
10935
- }
11120
+ #sanitizeInput(val) {
11121
+ val = val
11122
+ .replace(/^-+/, '') // dash as first char
11123
+ .replace(/(?!^)\+/g, '') // multiple plus symbols
11124
+ .replace('--', '-') // consecutive dashes
11125
+ .replace('+-', '+'); // dash following plus symbol
10936
11126
 
10937
- e.target.value = sanitizedInput;
10938
- });
11127
+ if (!this.allowAlphanumericInput) {
11128
+ const telDigitsRegExp = /^[+\d-\(\)]+$/;
11129
+ val = val
11130
+ .split('')
11131
+ .filter((char) => telDigitsRegExp.test(char))
11132
+ .join('');
11133
+ }
10939
11134
 
10940
- this.handleFocusEventsDispatching([this.phoneNumberInput]);
11135
+ return val;
10941
11136
  }
10942
11137
 
10943
- attributeChangedCallback(attrName, oldValue, newValue) {
10944
- super.attributeChangedCallback(attrName, oldValue, newValue);
11138
+ #formatPhoneNumber(phoneNumber = '') {
11139
+ // Get country code from `default-code or` from phone number
11140
+ const countryCode = this.defaultCode || this.#getCountryCodeFromValue(phoneNumber);
10945
11141
 
10946
- if (oldValue !== newValue && observedAttributes$3.includes(attrName)) {
10947
- const attr = mapAttrs[attrName] || attrName;
10948
- this.phoneNumberInput.setAttribute(attr, newValue);
11142
+ // Skip formatting if no country code is available
11143
+ if (!countryCode) {
11144
+ return phoneNumber;
10949
11145
  }
11146
+
11147
+ // Update AsYouType country code if needed
11148
+ if (!this.#ayt || this.#ayt.country !== countryCode) {
11149
+ this.#ayt = new AsYouType(countryCode);
11150
+ }
11151
+
11152
+ // We need to reset AsYouType instance before setting new input
11153
+ this.#ayt.reset();
11154
+
11155
+ // Set AsYouType input
11156
+ const formattedVal = this.#ayt.input(phoneNumber) || phoneNumber;
11157
+
11158
+ return formattedVal;
11159
+ }
11160
+
11161
+ #getCountryCodeFromValue(val) {
11162
+ const parsed = parsePhoneNumberFromString$1(val);
11163
+ return parsed?.country || '';
11164
+ }
11165
+
11166
+ #canFormat(val) {
11167
+ if (!matchingParenthesis(val)) return false;
11168
+ return true;
10950
11169
  }
10951
11170
  }
10952
11171
 
@@ -10957,7 +11176,7 @@ const textVars = TextFieldClass.cssVarList;
10957
11176
  const componentName$M = getComponentName$1('phone-input-box-field');
10958
11177
 
10959
11178
  const customMixin$9 = (superclass) =>
10960
- class PhoneInputBoxFieldMixinClass extends superclass {
11179
+ class PhoneFieldInputBoxMixinClass extends superclass {
10961
11180
  static get CountryCodes() {
10962
11181
  return CountryCodes;
10963
11182
  }
@@ -10981,8 +11200,6 @@ const customMixin$9 = (superclass) =>
10981
11200
  forwardAttrs$1(this.shadowRoot.host, this.inputElement, {
10982
11201
  includeAttrs: [
10983
11202
  'size',
10984
- 'bordered',
10985
- 'invalid',
10986
11203
  'minlength',
10987
11204
  'maxlength',
10988
11205
  'default-code',
@@ -10991,6 +11208,10 @@ const customMixin$9 = (superclass) =>
10991
11208
  'label',
10992
11209
  'label-type',
10993
11210
  'allow-alphanumeric-input',
11211
+ 'restrict-countries',
11212
+ 'format-value',
11213
+ 'strict-validation',
11214
+ 'data-errormessage-type-mismatch',
10994
11215
  ],
10995
11216
  });
10996
11217
  }
@@ -11006,7 +11227,8 @@ const {
11006
11227
  inputElement: inputElement$1,
11007
11228
  requiredIndicator: requiredIndicator$3,
11008
11229
  inputField: inputField$1,
11009
- inputFieldInternal,
11230
+ internalComponent,
11231
+ internalComponentAfter,
11010
11232
  phoneInput,
11011
11233
  errorMessage: errorMessage$5,
11012
11234
  helperText: helperText$3,
@@ -11017,8 +11239,11 @@ const {
11017
11239
  inputElement: { selector: 'input' },
11018
11240
  requiredIndicator: { selector: '[required]::part(required-indicator)::after' },
11019
11241
  inputField: { selector: () => 'vaadin-text-field::part(input-field)' },
11020
- inputFieldInternal: {
11021
- selector: () => 'descope-phone-field-internal-input-box vaadin-text-field::part(input-field)',
11242
+ internalComponent: {
11243
+ selector: 'descope-phone-field-internal-input-box',
11244
+ },
11245
+ internalComponentAfter: {
11246
+ selector: 'descope-phone-field-internal-input-box::after',
11022
11247
  },
11023
11248
  phoneInput: { selector: () => 'descope-text-field' },
11024
11249
  helperText: { selector: '::part(helper-text)' },
@@ -11041,14 +11266,6 @@ const PhoneFieldInputBoxClass = compose$1(
11041
11266
  hostMinWidth: { ...host$f, property: 'min-width' },
11042
11267
  hostDirection: { ...host$f, property: 'direction' },
11043
11268
 
11044
- inputBorderStyle: { ...inputFieldInternal, property: 'border-style' },
11045
- inputBorderWidth: { ...inputFieldInternal, property: 'border-width' },
11046
- inputBorderColor: { ...inputFieldInternal, property: 'border-color' },
11047
- inputBorderRadius: [
11048
- { ...inputField$1, property: 'border-radius' },
11049
- { ...inputFieldInternal, property: 'border-radius' },
11050
- ],
11051
-
11052
11269
  inputHorizontalPadding: [
11053
11270
  { ...phoneInput, property: 'padding-left' },
11054
11271
  { ...phoneInput, property: 'padding-right' },
@@ -11073,10 +11290,18 @@ const PhoneFieldInputBoxClass = compose$1(
11073
11290
 
11074
11291
  inputPlaceholderTextColor: { ...phoneInput, property: textVars.inputPlaceholderColor },
11075
11292
 
11076
- inputOutlineStyle: { ...inputField$1, property: 'outline-style' },
11077
- inputOutlineColor: { ...inputField$1, property: 'outline-color' },
11078
- inputOutlineWidth: { ...inputField$1, property: 'outline-width' },
11079
- inputOutlineOffset: { ...inputField$1, property: 'outline-offset' },
11293
+ inputBorderStyle: { ...internalComponentAfter, property: 'border-style' },
11294
+ inputBorderWidth: { ...internalComponentAfter, property: 'border-width' },
11295
+ inputBorderColor: { ...internalComponentAfter, property: 'border-color' },
11296
+ inputBorderRadius: [
11297
+ { ...internalComponent, property: 'border-radius' },
11298
+ { ...internalComponentAfter, property: 'border-radius' },
11299
+ ],
11300
+
11301
+ inputOutlineStyle: { ...internalComponent, property: 'outline-style' },
11302
+ inputOutlineColor: { ...internalComponent, property: 'outline-color' },
11303
+ inputOutlineWidth: { ...internalComponent, property: 'outline-width' },
11304
+ inputOutlineOffset: { ...internalComponent, property: 'outline-offset' },
11080
11305
 
11081
11306
  labelPosition: { ...label$3, property: 'position' },
11082
11307
  labelTopPosition: { ...label$3, property: 'top' },
@@ -11095,6 +11320,7 @@ const PhoneFieldInputBoxClass = compose$1(
11095
11320
  },
11096
11321
  }),
11097
11322
  draggableMixin$1,
11323
+ inputOverrideValidConstraintsMixin,
11098
11324
  composedProxyInputMixin$1({ proxyProps: ['value', 'selectionStart'] }),
11099
11325
  customMixin$9
11100
11326
  )(
@@ -11126,7 +11352,6 @@ const PhoneFieldInputBoxClass = compose$1(
11126
11352
  vaadin-text-field::part(input-field) {
11127
11353
  padding: 0;
11128
11354
  background: transparent;
11129
- overflow: hidden;
11130
11355
  -webkit-mask-image: none;
11131
11356
  }
11132
11357
  descope-phone-field-internal-input-box {
@@ -11137,14 +11362,20 @@ const PhoneFieldInputBoxClass = compose$1(
11137
11362
  descope-phone-field-internal-input-box > div {
11138
11363
  width: 100%;
11139
11364
  }
11140
- descope-phone-field-internal-input-box .separator {
11141
- flex: 0;
11142
- border: none;
11143
- }
11144
11365
  descope-phone-field-internal-input-box descope-text-field {
11145
11366
  ${textVars.inputOutlineWidth}: 0;
11146
11367
  ${textVars.inputOutlineOffset}: 0;
11147
11368
  }
11369
+ descope-phone-field-internal-input-box::after {
11370
+ content: '';
11371
+ position: absolute;
11372
+ width: 100%;
11373
+ height: 100%;
11374
+ top: 0;
11375
+ left: 0;
11376
+ box-sizing: border-box;
11377
+ pointer-events: none;
11378
+ }
11148
11379
  descope-text-field {
11149
11380
  flex-grow: 1;
11150
11381
  width: 100%;
@@ -17714,6 +17945,8 @@ const attrs = {
17714
17945
  'restrict-countries',
17715
17946
  'default-code',
17716
17947
  'phone-minlength',
17948
+ 'phone-format-value',
17949
+ 'phone-strict-validation',
17717
17950
  'data-errormessage-value-missing-phone',
17718
17951
  ],
17719
17952
  inputBox: [
@@ -17721,6 +17954,8 @@ const attrs = {
17721
17954
  'restrict-countries',
17722
17955
  'default-code',
17723
17956
  'phone-minlength',
17957
+ 'phone-format-value',
17958
+ 'phone-strict-validation',
17724
17959
  'data-errormessage-value-missing-phone',
17725
17960
  ],
17726
17961
  },
@@ -17734,6 +17969,8 @@ const attrMap = {
17734
17969
  phone: {
17735
17970
  'phone-input-label': 'label',
17736
17971
  'phone-minlength': 'minlength',
17972
+ 'phone-format-value': 'format-value',
17973
+ 'phone-strict-validation': 'strict-validation',
17737
17974
  'data-errormessage-value-missing-phone': 'data-errormessage-value-missing',
17738
17975
  },
17739
17976
  };