@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.
- package/dist/cjs/index.cjs.js +58 -24
- package/dist/cjs/index.cjs.js.map +1 -1
- package/dist/index.esm.js +464 -225
- package/dist/index.esm.js.map +1 -1
- package/dist/umd/3620.js +1 -1
- package/dist/umd/3620.js.map +1 -1
- package/dist/umd/5348.js +2 -0
- package/dist/umd/5348.js.map +1 -0
- package/dist/umd/6477.js +149 -0
- package/dist/umd/6477.js.map +1 -0
- package/dist/umd/9365.js +1 -1
- package/dist/umd/9365.js.map +1 -1
- package/dist/umd/DescopeDev.js +1 -1
- package/dist/umd/DescopeDev.js.map +1 -1
- package/dist/umd/descope-hybrid-field-index-js.js +3 -3
- package/dist/umd/descope-hybrid-field-index-js.js.map +1 -1
- package/dist/umd/descope-passcode-index-js.js +1 -1
- package/dist/umd/descope-passcode-index-js.js.map +1 -1
- package/dist/umd/index.js +1 -1
- package/dist/umd/index.js.map +1 -1
- package/dist/umd/phone-fields-descope-phone-field-descope-phone-field-internal-index-js.js +1 -1
- package/dist/umd/phone-fields-descope-phone-field-descope-phone-field-internal-index-js.js.map +1 -1
- package/dist/umd/phone-fields-descope-phone-field-index-js.js +1 -1
- package/dist/umd/phone-fields-descope-phone-field-index-js.js.map +1 -1
- package/dist/umd/phone-fields-descope-phone-input-box-field-descope-phone-input-box-internal-index-js.js +2 -2
- package/dist/umd/phone-fields-descope-phone-input-box-field-descope-phone-input-box-internal-index-js.js.map +1 -1
- package/dist/umd/phone-fields-descope-phone-input-box-field-index-js.js +2 -113
- package/dist/umd/phone-fields-descope-phone-input-box-field-index-js.js.LICENSE.txt +0 -6
- package/dist/umd/phone-fields-descope-phone-input-box-field-index-js.js.map +1 -1
- package/package.json +7 -6
- package/src/components/descope-hybrid-field/HybridFieldClass.js +6 -0
- package/src/components/descope-passcode/PasscodeClass.js +2 -0
- package/src/components/phone-fields/descope-phone-field/PhoneFieldClass.js +10 -2
- package/src/components/phone-fields/descope-phone-field/descope-phone-field-internal/PhoneFieldInternal.js +229 -125
- package/src/components/phone-fields/descope-phone-input-box-field/PhoneFieldInputBoxClass.js +42 -24
- package/src/components/phone-fields/descope-phone-input-box-field/descope-phone-input-box-internal/PhoneFieldInternalInputBox.js +176 -79
- package/src/components/phone-fields/descope-phone-input-box-field/index.js +0 -1
- package/src/components/phone-fields/helpers.js +7 -0
- package/src/mixins/index.js +1 -0
- package/src/mixins/inputOverrideValidConstraints.js +12 -0
- package/dist/umd/6424.js +0 -149
- package/dist/umd/6424.js.map +0 -1
- /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.
|
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
|
47
|
+
get defaultDialCode() {
|
43
48
|
return getCountryByCodeId(this.getAttribute('default-code'));
|
44
49
|
}
|
45
50
|
|
46
|
-
get
|
47
|
-
return
|
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
|
55
|
-
|
56
|
-
|
57
|
-
}
|
59
|
+
get minLength() {
|
60
|
+
return parseInt(this.getAttribute('minlength'), 10) || 0;
|
61
|
+
}
|
58
62
|
|
59
|
-
|
60
|
-
|
61
|
-
|
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
|
-
|
67
|
+
get restrictCountries() {
|
68
|
+
return this.getAttribute('restrict-countries')?.split(',').filter(Boolean) || [];
|
74
69
|
}
|
75
70
|
|
76
|
-
|
77
|
-
this.
|
71
|
+
get isFormatValue() {
|
72
|
+
return this.getAttribute('format-value') === 'true';
|
78
73
|
}
|
79
74
|
|
80
|
-
get
|
81
|
-
return this.
|
75
|
+
get isStrictValidation() {
|
76
|
+
return this.getAttribute('strict-validation') === 'true';
|
82
77
|
}
|
83
78
|
|
84
|
-
get
|
85
|
-
|
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
|
-
|
89
|
-
|
96
|
+
set value(val) {
|
97
|
+
this.textField.value = val;
|
90
98
|
}
|
91
99
|
|
92
|
-
|
93
|
-
|
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
|
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 (
|
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
|
-
|
120
|
-
this.
|
121
|
-
|
122
|
-
if (e.isTrusted) this.phoneNumberInput.focus();
|
123
|
-
});
|
140
|
+
setSelectionRange(...args) {
|
141
|
+
this.textField.setSelectionRange(...args);
|
142
|
+
}
|
124
143
|
|
125
|
-
|
126
|
-
|
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
|
-
|
130
|
-
|
131
|
-
|
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
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
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.
|
214
|
+
return this.restrictCountries.includes(countryCode);
|
160
215
|
}
|
161
216
|
|
162
|
-
|
163
|
-
|
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
|
-
|
166
|
-
|
167
|
-
|
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,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
|
+
};
|
package/src/mixins/index.js
CHANGED
@@ -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
|
+
};
|