feat: New phone number input component (#10446)
This commit is contained in:
@@ -0,0 +1,49 @@
|
|||||||
|
<script setup>
|
||||||
|
import { ref } from 'vue';
|
||||||
|
import PhoneNumberInput from './PhoneNumberInput.vue';
|
||||||
|
|
||||||
|
const phoneNumber = ref('+14155552671');
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<Story
|
||||||
|
title="Components/PhoneNumberInput"
|
||||||
|
:layout="{ type: 'grid', width: '400px' }"
|
||||||
|
>
|
||||||
|
<Variant title="Default">
|
||||||
|
<div class="flex flex-col gap-4 p-4 h-[300px]">
|
||||||
|
<PhoneNumberInput
|
||||||
|
v-model="phoneNumber"
|
||||||
|
placeholder="Enter phone number"
|
||||||
|
/>
|
||||||
|
<div class="text-sm">Phone number value: {{ phoneNumber }}</div>
|
||||||
|
</div>
|
||||||
|
</Variant>
|
||||||
|
|
||||||
|
<Variant title="Disabled">
|
||||||
|
<div class="flex flex-col gap-4 p-4 h-[300px]">
|
||||||
|
<PhoneNumberInput
|
||||||
|
v-model="phoneNumber"
|
||||||
|
placeholder="Enter phone number"
|
||||||
|
disabled
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</Variant>
|
||||||
|
|
||||||
|
<Variant title="Without Border">
|
||||||
|
<div class="flex flex-col gap-4 p-4 h-[300px]">
|
||||||
|
<PhoneNumberInput
|
||||||
|
v-model="phoneNumber"
|
||||||
|
placeholder="Enter phone number"
|
||||||
|
:show-border="false"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</Variant>
|
||||||
|
|
||||||
|
<Variant title="Empty State auto select based on time zone">
|
||||||
|
<div class="flex flex-col gap-4 p-4 h-[300px]">
|
||||||
|
<PhoneNumberInput placeholder="Enter phone number" />
|
||||||
|
</div>
|
||||||
|
</Variant>
|
||||||
|
</Story>
|
||||||
|
</template>
|
||||||
@@ -0,0 +1,219 @@
|
|||||||
|
<script setup>
|
||||||
|
import { ref, computed, watch } from 'vue';
|
||||||
|
import parsePhoneNumber from 'libphonenumber-js';
|
||||||
|
import { useI18n } from 'vue-i18n';
|
||||||
|
import countries from 'shared/constants/countries.js';
|
||||||
|
import { useVuelidate } from '@vuelidate/core';
|
||||||
|
import { required, minLength, numeric } from '@vuelidate/validators';
|
||||||
|
import {
|
||||||
|
getActiveCountryCode,
|
||||||
|
getActiveDialCode,
|
||||||
|
} from 'shared/components/PhoneInput/helper';
|
||||||
|
|
||||||
|
import Input from 'dashboard/components-next/input/Input.vue';
|
||||||
|
import Button from 'dashboard/components-next/button/Button.vue';
|
||||||
|
import DropdownMenu from 'dashboard/components-next/dropdown-menu/DropdownMenu.vue';
|
||||||
|
|
||||||
|
const props = defineProps({
|
||||||
|
placeholder: {
|
||||||
|
type: String,
|
||||||
|
default: '',
|
||||||
|
},
|
||||||
|
disabled: {
|
||||||
|
type: Boolean,
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
showBorder: {
|
||||||
|
type: Boolean,
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const modelValue = defineModel({
|
||||||
|
type: [String, Number],
|
||||||
|
default: '',
|
||||||
|
});
|
||||||
|
|
||||||
|
const { t } = useI18n();
|
||||||
|
|
||||||
|
const showDropdown = ref(false);
|
||||||
|
const searchQuery = ref('');
|
||||||
|
const activeCountryCode = ref(getActiveCountryCode());
|
||||||
|
const activeDialCode = ref(getActiveDialCode());
|
||||||
|
const phoneNumber = ref('');
|
||||||
|
|
||||||
|
const rules = {
|
||||||
|
phoneNumber: {
|
||||||
|
minLength: minLength(2),
|
||||||
|
numeric,
|
||||||
|
},
|
||||||
|
activeDialCode: {
|
||||||
|
required,
|
||||||
|
validDialCode: value => {
|
||||||
|
return countries.some(country => country.dial_code === value);
|
||||||
|
},
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
const v$ = useVuelidate(rules, {
|
||||||
|
phoneNumber,
|
||||||
|
activeDialCode,
|
||||||
|
});
|
||||||
|
|
||||||
|
const hasError = computed(() => v$.value.$invalid);
|
||||||
|
|
||||||
|
const countryList = computed(() => {
|
||||||
|
return countries.map(country => ({
|
||||||
|
value: country.id,
|
||||||
|
label: country.name,
|
||||||
|
dialCode: country.dial_code,
|
||||||
|
emoji: country.emoji,
|
||||||
|
isSelected: String(activeCountryCode.value) === String(country.id),
|
||||||
|
action: 'phoneNumberInput',
|
||||||
|
}));
|
||||||
|
});
|
||||||
|
|
||||||
|
const filteredCountries = computed(() => {
|
||||||
|
const query = searchQuery.value.toLowerCase();
|
||||||
|
return countryList.value.filter(({ label, dialCode, value }) =>
|
||||||
|
[label, dialCode, value].some(field => field.toLowerCase().includes(query))
|
||||||
|
);
|
||||||
|
});
|
||||||
|
|
||||||
|
const activeCountry = computed(() =>
|
||||||
|
activeCountryCode.value
|
||||||
|
? countryList.value.find(
|
||||||
|
country => country.value === activeCountryCode.value
|
||||||
|
)
|
||||||
|
: ''
|
||||||
|
);
|
||||||
|
|
||||||
|
const inputBorderClass = computed(() => {
|
||||||
|
const errorClass =
|
||||||
|
'border-n-ruby-8 dark:border-n-ruby-8 hover:border-n-ruby-9 dark:hover:border-n-ruby-9 disabled:border-n-ruby-8 dark:disabled:border-n-ruby-8';
|
||||||
|
if (!props.showBorder) {
|
||||||
|
return hasError.value ? errorClass : 'border-transparent';
|
||||||
|
}
|
||||||
|
if (hasError.value) {
|
||||||
|
return errorClass;
|
||||||
|
}
|
||||||
|
return 'has-[:focus]:border-n-brand dark:has-[:focus]:border-n-brand border-n-weak dark:border-n-weak hover:border-n-slate-6 dark:hover:border-n-slate-6 disabled:border-n-weak dark:disabled:border-n-weak';
|
||||||
|
});
|
||||||
|
|
||||||
|
const phoneNumberError = computed(() => {
|
||||||
|
if (!v$.value.$dirty) return '';
|
||||||
|
return v$.value.activeDialCode.$invalid
|
||||||
|
? t('PHONE_INPUT.DIAL_CODE_ERROR')
|
||||||
|
: v$.value.phoneNumber.$invalid && t('PHONE_INPUT.ERROR');
|
||||||
|
});
|
||||||
|
|
||||||
|
const emitPhoneNumber = value => {
|
||||||
|
const newValue = value ? `${activeDialCode.value}${value}` : '';
|
||||||
|
modelValue.value = newValue;
|
||||||
|
};
|
||||||
|
|
||||||
|
const onSelectCountry = async ({ value, dialCode }) => {
|
||||||
|
if (!value || !showDropdown.value) return;
|
||||||
|
|
||||||
|
activeCountryCode.value = value;
|
||||||
|
activeDialCode.value = dialCode;
|
||||||
|
searchQuery.value = '';
|
||||||
|
showDropdown.value = false;
|
||||||
|
if (!v$.value.$invalid && phoneNumber.value) {
|
||||||
|
emitPhoneNumber(phoneNumber.value);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const toggleCountryDropdown = () => {
|
||||||
|
showDropdown.value = !showDropdown.value;
|
||||||
|
};
|
||||||
|
|
||||||
|
const closeCountryDropdown = () => {
|
||||||
|
showDropdown.value = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
watch(phoneNumber, async value => {
|
||||||
|
await v$.value.$touch();
|
||||||
|
if (!v$.value.$invalid) {
|
||||||
|
emitPhoneNumber(value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
watch(
|
||||||
|
modelValue,
|
||||||
|
newValue => {
|
||||||
|
const number = parsePhoneNumber(newValue);
|
||||||
|
if (number) {
|
||||||
|
if (number?.country) activeCountryCode.value = number.country;
|
||||||
|
if (number?.countryCallingCode)
|
||||||
|
activeDialCode.value = `+${number.countryCallingCode}`;
|
||||||
|
phoneNumber.value = newValue.replace(`+${number.countryCallingCode}`, '');
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ immediate: true }
|
||||||
|
);
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<div>
|
||||||
|
<div
|
||||||
|
v-on-clickaway="() => closeCountryDropdown()"
|
||||||
|
class="relative flex items-center h-8 transition-all duration-500 ease-in-out border rounded-lg bg-n-alpha-black2"
|
||||||
|
:class="[inputBorderClass, { 'cursor-not-allowed opacity-50': disabled }]"
|
||||||
|
>
|
||||||
|
<Input
|
||||||
|
v-model="phoneNumber"
|
||||||
|
type="tel"
|
||||||
|
:placeholder="placeholder"
|
||||||
|
:disabled="disabled"
|
||||||
|
custom-input-class="!border-0 h-8 !py-0.5 !bg-transparent ltr:!pl-1 rtl:!pr-1"
|
||||||
|
class="w-full !flex-row"
|
||||||
|
>
|
||||||
|
<template #prefix>
|
||||||
|
<div class="flex items-center flex-shrink-0">
|
||||||
|
<Button
|
||||||
|
:label="activeCountry?.emoji || ''"
|
||||||
|
color="slate"
|
||||||
|
size="sm"
|
||||||
|
:icon="
|
||||||
|
!activeCountry ? 'i-lucide-globe' : 'i-lucide-chevron-down'
|
||||||
|
"
|
||||||
|
trailing-icon
|
||||||
|
:disabled="disabled"
|
||||||
|
class="!h-[30px] top-1 !px-2 outline-0 !outline-none !rounded-lg border-0 ltr:!rounded-r-none rtl:!rounded-l-none"
|
||||||
|
@click="toggleCountryDropdown"
|
||||||
|
>
|
||||||
|
<span
|
||||||
|
v-if="activeCountry"
|
||||||
|
class="inline-flex justify-center text-sm whitespace-nowrap"
|
||||||
|
>
|
||||||
|
{{ activeCountry?.emoji }}
|
||||||
|
</span>
|
||||||
|
</Button>
|
||||||
|
<span
|
||||||
|
v-if="activeCountry"
|
||||||
|
class="text-sm left-[38px] top-2.5 text-n-slate-11 ltr:!pl-1 rtl:!pr-1"
|
||||||
|
>
|
||||||
|
{{ activeDialCode }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
|
</Input>
|
||||||
|
<DropdownMenu
|
||||||
|
v-if="showDropdown"
|
||||||
|
:menu-items="filteredCountries"
|
||||||
|
show-search
|
||||||
|
class="z-[100] w-48 mt-2 overflow-y-auto ltr:left-0 rtl:right-0 top-full max-h-52"
|
||||||
|
@action="onSelectCountry"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
<template v-if="phoneNumberError">
|
||||||
|
<p
|
||||||
|
v-if="phoneNumberError"
|
||||||
|
class="min-w-0 mt-1 mb-0 text-xs truncate transition-all duration-500 ease-in-out text-n-ruby-9 dark:text-n-ruby-9"
|
||||||
|
>
|
||||||
|
{{ phoneNumberError }}
|
||||||
|
</p>
|
||||||
|
</template>
|
||||||
|
</div>
|
||||||
|
</template>
|
||||||
@@ -18,6 +18,11 @@
|
|||||||
"CONFIRM": "Confirm"
|
"CONFIRM": "Confirm"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"PHONE_INPUT": {
|
||||||
|
"SEARCH_PLACEHOLDER": "Search country",
|
||||||
|
"ERROR": "Phone number should be empty or in E.164 format",
|
||||||
|
"DIAL_CODE_ERROR": "Please select a dial code from the list"
|
||||||
|
},
|
||||||
"THUMBNAIL": {
|
"THUMBNAIL": {
|
||||||
"AUTHOR": {
|
"AUTHOR": {
|
||||||
"NOT_AVAILABLE": "Author is not available"
|
"NOT_AVAILABLE": "Author is not available"
|
||||||
|
|||||||
Reference in New Issue
Block a user