Files
chatwoot/app/javascript/dashboard/components-next/phonenumberinput/PhoneNumberInput.vue

226 lines
6.3 KiB
Vue

<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 =
'outline-n-ruby-8 dark:outline-n-ruby-8 hover:outline-n-ruby-9 dark:hover:outline-n-ruby-9 disabled:outline-n-ruby-8 dark:disabled:outline-n-ruby-8';
const focusClass =
'has-[:focus]:outline-n-brand dark:has-[:focus]:outline-n-brand';
if (!props.showBorder) {
if (hasError.value) return errorClass;
return `outline-transparent ${focusClass}`;
}
if (hasError.value) {
return errorClass;
}
return `${focusClass} outline-n-weak dark:outline-n-weak hover:outline-n-slate-6 dark:hover:outline-n-slate-6 disabled:outline-n-weak dark:disabled:outline-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 outline outline-1 outline-offset-[-1px] 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 !outline-none 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"
type="button"
class="!h-[1.875rem] top-1 ltr:ml-px rtl:mr-px !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>