@descope/web-components-ui 1.44.0 → 1.45.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 +464 -225
  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 -125
  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 +176 -79
  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
@@ -1,20 +1,18 @@
1
1
  import { createBaseInputClass } from '../../../../baseClasses/createBaseInputClass';
2
2
  import { getComponentName } from '../../../../helpers/componentHelpers';
3
- import { getCountryByCodeId } from '../../helpers';
3
+ import { getCountryByCodeId, matchingParenthesis } from '../../helpers';
4
+ import parsePhoneNumberFromString, { AsYouType } from 'libphonenumber-js/min';
4
5
 
5
6
  export const componentName = getComponentName('phone-field-internal-input-box');
6
7
 
7
8
  const observedAttributes = [
8
9
  'disabled',
9
10
  'size',
10
- 'bordered',
11
- 'invalid',
12
11
  'readonly',
13
12
  'phone-input-placeholder',
14
13
  'name',
15
14
  'autocomplete',
16
15
  'label-type',
17
- 'allow-alphanumeric-input',
18
16
  ];
19
17
  const mapAttrs = {
20
18
  'phone-input-placeholder': 'placeholder',
@@ -27,77 +25,95 @@ class PhoneFieldInternal extends BaseInputClass {
27
25
  return [].concat(BaseInputClass.observedAttributes || [], observedAttributes);
28
26
  }
29
27
 
28
+ #ayt;
29
+
30
30
  constructor() {
31
31
  super();
32
32
 
33
33
  this.innerHTML = `
34
34
  <div>
35
- <descope-text-field tabindex="1"></descope-text-field>
35
+ <descope-text-field tabindex="1" type="tel" bordered="false"></descope-text-field>
36
36
  </div>
37
37
  `;
38
38
 
39
- this.phoneNumberInput = this.querySelector('descope-text-field');
39
+ this.textField = this.querySelector('descope-text-field');
40
+ }
41
+
42
+ // notice: this function is exposed in parent component
43
+ get phoneNumberInputEle() {
44
+ return this.textField.shadowRoot.querySelector('input');
40
45
  }
41
46
 
42
- get defaultCountryCode() {
47
+ get defaultDialCode() {
43
48
  return getCountryByCodeId(this.getAttribute('default-code'));
44
49
  }
45
50
 
46
- get hasDefaultCode() {
47
- return !!this.getAttribute('default-code');
51
+ get defaultCode() {
52
+ return this.getAttribute('default-code');
48
53
  }
49
54
 
50
55
  get allowAlphanumericInput() {
51
56
  return this.getAttribute('allow-alphanumeric-input') === 'true';
52
57
  }
53
58
 
54
- get value() {
55
- if (!this.phoneNumberValue) {
56
- return '';
57
- }
59
+ get minLength() {
60
+ return parseInt(this.getAttribute('minlength'), 10) || 0;
61
+ }
58
62
 
59
- if (this.hasDefaultCode) {
60
- // we want to transform phone numbers to a valid {dialCode}-{phoneNumber} format
61
- // e.g.:
62
- // +972-12345 => +972-12345
63
- // 972-12345 => +972-12345
64
- // 12345 => +972-12345
65
- //
66
- // we also want to handle any extra dash if added in the start of the phone number
67
- // e.g.:
68
- // +972--12345 => +972-12345
69
- const pattern = new RegExp(`\\+?${parseInt(this.defaultCountryCode, 10)}--?`);
70
- return `${this.defaultCountryCode}-${this.phoneNumberInput.value.replace(pattern, '')}`;
71
- }
63
+ get maxLength() {
64
+ return parseInt(this.getAttribute('maxlength'), 10) || 50;
65
+ }
72
66
 
73
- return this.phoneNumberInput.value;
67
+ get restrictCountries() {
68
+ return this.getAttribute('restrict-countries')?.split(',').filter(Boolean) || [];
74
69
  }
75
70
 
76
- set value(val) {
77
- this.phoneNumberInput.value = val;
71
+ get isFormatValue() {
72
+ return this.getAttribute('format-value') === 'true';
78
73
  }
79
74
 
80
- get phoneNumberValue() {
81
- return this.phoneNumberInput.value;
75
+ get isStrictValidation() {
76
+ return this.getAttribute('strict-validation') === 'true';
82
77
  }
83
78
 
84
- get phoneNumberInputEle() {
85
- return this.phoneNumberInput.shadowRoot.querySelector('input');
79
+ get value() {
80
+ if (!this.textField.value) return '';
81
+
82
+ if (!this.isStrictValidation) {
83
+ return this.#nonParsedValue();
84
+ }
85
+
86
+ const parsedVal = this.#parseWithCountryCode();
87
+
88
+ if (parsedVal?.country && parsedVal?.countryCallingCode && parsedVal?.nationalNumber) {
89
+ return `+${[parsedVal?.countryCallingCode, parsedVal?.nationalNumber].join('-')}`;
90
+ }
91
+
92
+ // if failed to parse or to find country code return text field value
93
+ return this.textField.value;
86
94
  }
87
95
 
88
- get minLength() {
89
- return parseInt(this.getAttribute('minlength'), 10) || 0;
96
+ set value(val) {
97
+ this.textField.value = val;
90
98
  }
91
99
 
92
- get maxLength() {
93
- return parseInt(this.getAttribute('maxlength'), 10) || 50;
100
+ init() {
101
+ this.addEventListener('focus', (e) => {
102
+ // We want to ignore focus events we are dispatching
103
+ if (e.isTrusted) this.textField.focus();
104
+ });
105
+
106
+ super.init?.();
107
+
108
+ this.textField.addEventListener('input', this.#onInput.bind(this));
109
+ this.handleFocusEventsDispatching([this.textField]);
94
110
  }
95
111
 
96
112
  getValidity() {
97
113
  const validPhonePattern = /^\+?\d{1,4}-?(?:\d-?){1,15}$/;
98
- const stripValue = this.value.replace(/\D/g, '');
114
+ const stripValue = this.#sanitizeVal(this.textField.value);
99
115
 
100
- if (this.isRequired && !this.value) {
116
+ if (this.isRequired && !this.textField.value) {
101
117
  return { valueMissing: true };
102
118
  }
103
119
 
@@ -109,63 +125,144 @@ class PhoneFieldInternal extends BaseInputClass {
109
125
  return { tooLong: true };
110
126
  }
111
127
 
112
- if (this.value && !validPhonePattern.test(this.value)) {
128
+ if (
129
+ // has `strict-validation` and not properly parsed
130
+ (this.isStrictValidation && this.textField.value && !this.#isValidParsedValue()) ||
131
+ // if no `strict-validation` then conform with naive pattern
132
+ (!this.isStrictValidation && this.textField.value && !validPhonePattern.test(this.value))
133
+ ) {
113
134
  return { patternMismatch: true };
114
135
  }
115
136
 
116
137
  return {};
117
138
  }
118
139
 
119
- init() {
120
- this.addEventListener('focus', (e) => {
121
- // we want to ignore focus events we are dispatching
122
- if (e.isTrusted) this.phoneNumberInput.focus();
123
- });
140
+ setSelectionRange(...args) {
141
+ this.textField.setSelectionRange(...args);
142
+ }
124
143
 
125
- super.init?.();
126
- this.initInputs();
144
+ attributeChangedCallback(attrName, oldValue, newValue) {
145
+ super.attributeChangedCallback(attrName, oldValue, newValue);
146
+
147
+ if (oldValue !== newValue && observedAttributes.includes(attrName)) {
148
+ const attr = mapAttrs[attrName] || attrName;
149
+ this.textField.setAttribute(attr, newValue);
150
+ }
127
151
  }
128
152
 
129
- getCountryByDialCode(countryDialCode) {
130
- return this.countryCodeInput.items?.find(
131
- (c) => c.getAttribute('data-country-code') === countryDialCode
153
+ #onInput(e) {
154
+ let sanitizedInput = this.#sanitizeInput(e.target.value);
155
+
156
+ if (this.isFormatValue && this.#canFormat(sanitizedInput)) {
157
+ sanitizedInput = this.#formatPhoneNumber(sanitizedInput);
158
+ }
159
+
160
+ e.target.value = sanitizedInput;
161
+ }
162
+
163
+ #nonParsedValue() {
164
+ if (!this.defaultDialCode) {
165
+ return this.textField.value;
166
+ }
167
+
168
+ const nationalNumber = this.#trimDuplicateCountryCode(this.textField.value);
169
+ const sanitizedVal = this.#sanitizeVal(nationalNumber);
170
+
171
+ return [this.defaultDialCode, sanitizedVal].join('-');
172
+ }
173
+
174
+ #parseWithCountryCode() {
175
+ if (this.defaultDialCode) {
176
+ return parsePhoneNumberFromString(
177
+ [this.defaultDialCode, this.#sanitizeVal(this.textField.value)].filter(Boolean).join('')
178
+ );
179
+ }
180
+
181
+ // if default-code or not parsed - try to extract country code from value
182
+ return parsePhoneNumberFromString(this.textField.value);
183
+ }
184
+
185
+ #sanitizeVal(val) {
186
+ return val.replace(/\D/g, '');
187
+ }
188
+
189
+ #trimDuplicateCountryCode(val) {
190
+ if (this.textField.value?.[0] === '+') {
191
+ const dialCodePrefixPattern = new RegExp(`^\\${this.defaultDialCode}`);
192
+ const trimmed = val.replace(dialCodePrefixPattern, '');
193
+ return trimmed;
194
+ }
195
+ return val;
196
+ }
197
+
198
+ #isValidParsedValue() {
199
+ const parsed = parsePhoneNumberFromString(this.value);
200
+ return (
201
+ parsed && // parsed successfully (not undefined)
202
+ parsed.isValid?.() && // Parsed object is valid
203
+ parsed.country && // Parsed object with a country code
204
+ this.#isAllowedCountry(parsed.country) && // Parsed with allowed country code
205
+ (this.defaultCode ? this.defaultCode === parsed.country : true) // In case default country code is set validate parsed country matches it
132
206
  );
133
207
  }
134
208
 
135
- initInputs() {
136
- // Sanitize phone input value to filter everything but digits
137
- this.phoneNumberInput.addEventListener('input', (e) => {
138
- if (e.target.value.length === 1 && e.target.value === '-') {
139
- e.target.value = '';
140
- }
141
-
142
- e.target.value = e.target.value
143
- .replace(/(?!^)\+/g, '')
144
- .replace('--', '-')
145
- .replace('+-', '+');
146
-
147
- let sanitizedInput = e.target.value;
148
- if (!this.allowAlphanumericInput) {
149
- const telDigitsRegExp = /^[+\d-]+$/;
150
- sanitizedInput = e.target.value
151
- .split('')
152
- .filter((char) => telDigitsRegExp.test(char))
153
- .join('');
154
- }
155
-
156
- e.target.value = sanitizedInput;
157
- });
209
+ #isAllowedCountry(countryCode) {
210
+ if (!this.restrictCountries) {
211
+ return true;
212
+ }
158
213
 
159
- this.handleFocusEventsDispatching([this.phoneNumberInput]);
214
+ return this.restrictCountries.includes(countryCode);
160
215
  }
161
216
 
162
- attributeChangedCallback(attrName, oldValue, newValue) {
163
- super.attributeChangedCallback(attrName, oldValue, newValue);
217
+ #sanitizeInput(val) {
218
+ val = val
219
+ .replace(/^-+/, '') // dash as first char
220
+ .replace(/(?!^)\+/g, '') // multiple plus symbols
221
+ .replace('--', '-') // consecutive dashes
222
+ .replace('+-', '+'); // dash following plus symbol
223
+
224
+ if (!this.allowAlphanumericInput) {
225
+ const telDigitsRegExp = /^[+\d-\(\)]+$/;
226
+ val = val
227
+ .split('')
228
+ .filter((char) => telDigitsRegExp.test(char))
229
+ .join('');
230
+ }
164
231
 
165
- if (oldValue !== newValue && observedAttributes.includes(attrName)) {
166
- const attr = mapAttrs[attrName] || attrName;
167
- this.phoneNumberInput.setAttribute(attr, newValue);
232
+ return val;
233
+ }
234
+
235
+ #formatPhoneNumber(phoneNumber = '') {
236
+ // Get country code from `default-code or` from phone number
237
+ const countryCode = this.defaultCode || this.#getCountryCodeFromValue(phoneNumber);
238
+
239
+ // Skip formatting if no country code is available
240
+ if (!countryCode) {
241
+ return phoneNumber;
242
+ }
243
+
244
+ // Update AsYouType country code if needed
245
+ if (!this.#ayt || this.#ayt.country !== countryCode) {
246
+ this.#ayt = new AsYouType(countryCode);
168
247
  }
248
+
249
+ // We need to reset AsYouType instance before setting new input
250
+ this.#ayt.reset();
251
+
252
+ // Set AsYouType input
253
+ const formattedVal = this.#ayt.input(phoneNumber) || phoneNumber;
254
+
255
+ return formattedVal;
256
+ }
257
+
258
+ #getCountryCodeFromValue(val) {
259
+ const parsed = parsePhoneNumberFromString(val);
260
+ return parsed?.country || '';
261
+ }
262
+
263
+ #canFormat(val) {
264
+ if (!matchingParenthesis(val)) return false;
265
+ return true;
169
266
  }
170
267
  }
171
268
 
@@ -1,5 +1,4 @@
1
1
  import './descope-phone-input-box-internal';
2
- import '@descope-ui/descope-combo-box';
3
2
  import '../../descope-text-field';
4
3
 
5
4
  import { componentName, PhoneFieldInputBoxClass } from './PhoneFieldInputBoxClass';
@@ -1,5 +1,12 @@
1
+ import parsePhoneNumberFromString from 'libphonenumber-js/min';
1
2
  import CountryCodes from './CountryCodes';
2
3
 
3
4
  export const getCountryByCodeId = (countryCode) => {
4
5
  return CountryCodes.find((c) => c.code === countryCode)?.dialCode;
5
6
  };
7
+
8
+ export const matchingParenthesis = (val) => {
9
+ const openParenMatches = val.match(/\(/g);
10
+ const closeParenMatches = val.match(/\)/g);
11
+ return openParenMatches?.length === closeParenMatches?.length;
12
+ };
@@ -11,3 +11,4 @@ export { normalizeBooleanAttributesMixin } from './normalizeBooleanAttributesMix
11
11
  export { lifecycleEventsMixin } from './lifecycleEventsMixin';
12
12
  export { inputEventsDispatchingMixin } from './inputEventsDispatchingMixin';
13
13
  export { externalInputMixin } from './externalInputMixin';
14
+ export { inputOverrideValidConstraintsMixin } from './inputOverrideValidConstraints';
@@ -0,0 +1,12 @@
1
+ export const inputOverrideValidConstraintsMixin = (superclass) =>
2
+ class InputOverrideValidConstraintsMixinClass extends superclass {
3
+ init() {
4
+ super.init?.();
5
+
6
+ // vaadin uses `validConstraints` (required, pattern, minlength, maxlength) to determine if it should validate
7
+ // the input or not. We want to override this behavior, so we can enforce validation even if these attributes are not present.
8
+ if (this.baseElement._hasValidConstraints) {
9
+ this.baseElement._hasValidConstraints = () => true;
10
+ }
11
+ }
12
+ };