A wrapper around libphonenumber with added functionality merged from the following libs:
Uses the following native libraries:
Platform | Library | Version |
---|---|---|
Android | libphonenumber | 8.12.24 |
iOS | PhoneNumberKit | 3.3 |
The main advantage to this lib is it lets you optionally format a phone number synchronously without making calls into libphonenumber with platform calls.
First you need to call the init
function. This will load all of the available regions available on the device from libphonenumber to build a formatting mask for each country using its example number from libphonenumber.
If you don't run the init function then formatNumberSync
will simply return the same thing passed into it without formatting anything as there won't be any masks to utilize.
We use the same approach from flutter_multi_formatter
for masking but instead of statically defining all of our country masks, we pull them on the fly from libphonenubmer so that the masks will be automatically maintained over time.
You can either do this during your app init before calling RunApp
, or with a FutureBuilder
as the example app demonstrates.
await FlutterLibphonenumber().init();
Normally calls to libphonenumber's format function are asynchronous, and you might not want your UI rebuilding every time you need to format a phone number.
To get around this, we load a mask of every supported phone region's example number from libphonenumber during the init()
call. We can then use this mask to format an e164 phone number synchronously like this:
final rawNumber = '+14145556666';
final formattedNumber = FlutterLibphonenumber().formatNumberSync(rawNumber); // +1 414-555-6666
When you call init, this lib will store a list of the countries and phone metadata with the following class:
class CountryWithPhoneCode {
final String phoneCode // '44',
final String countryCode // 'GB',
final String exampleNumberMobileNational // '07400 123456',
final String exampleNumberFixedLineNational // '0121 234 5678',
final String phoneMaskMobileNational // '00000 000000',
final String phoneMaskFixedLineNational // '0000 000 0000',
final String exampleNumberMobileInternational // '+44 7400 123456',
final String exampleNumberFixedLineInternational // '+44 121 234 5678',
final String phoneMaskMobileInternational // '+00 0000 000000',
final String phoneMaskFixedLineInternational // '+00 000 000 0000',
final String countryName // 'United Kingdom';
}
To access this list of countries you can get at it like this:
final countries = CountryManager().countries; // List<CountryWithPhoneCode>
Here is a reference of all of the available functions.
Must be called before we can format anything. This loads all available countries on the device and calls getAllSupportedRegions()
to then cross-reference and combine everything and save a List<CountryWithPhoneCode>
of every available country with its phone code / mask.
Optionally provide a map of overrides where the key is the country code (ex: GB
or US
) and the value is a CountryWithPhoneCode
object that should replace the data pulled from libphonenumber. This is useful if you want to customize the mask data for a given country.
Returns all of the available regions on the device in a map with each key as the region code, and the value as a CountryWithPhoneCode
object containing all of the mobile and landline phone masks / example numbers for national / international format, phone code, region code, and country name.
Example response:
{
"UK": [CountryWithPhoneCode()],
"US": [CountryWithPhoneCode()]
}
Formats a number using libphonenumber. Under the hood this is using the AsYouType formatter. Depending on your result, you might want to use formatNumberSync()
to utilize the phone number masks.
Will return the parsed / formatted number like this:
{
formatted: "1 (414) 444-4444",
}
Parses a number and if it is full and complete, returns some metadata associated with the number. The number must be a valid and compelte e164 formatted number to be considered valid.
Will throw an error if the number isn't valid.
Example response:
{
country_code: 49,
e164: '+4930123123123',
national: '030 123 123 123',
type: 'mobile',
international: '+49 30 123 123 123',
national_number: '030123123123',
}
String formatNumberSync(String number, {CountryWithPhoneCode? country, PhoneNumberType phoneNumberType = PhoneNumberType.mobile, phoneNumberFormat = PhoneNumberFormat.international, bool removeCountryCodeFromResult = false, bool inputContainsCountryCode = true})
Format a number synchronously using masks to format it. Must have ran the init()
function to pre-populate the mask data or else the original phone
value will be returned.
Optionally specify the phone number type to format it as (mobile
vs fixedLine
). This is useful when a country has more than one phone number format and you want to format it to either fit the fixed line or mobile pattern.
Use removeCountryCodeFromResult
to remove the country code from the formatted result. Set inputContainsCountryCode
accordingly based on if the inputted number to format contains the country code or not.
Example response:
"1 (414) 444-4444"
Future<FormatPhoneResult?> getFormattedParseResult(String phoneNumber,CountryWithPhoneCode country, {PhoneNumberType phoneNumberType = PhoneNumberType.mobile, PhoneNumberFormat phoneNumberFormat = PhoneNumberFormat.international,})
Asynchronously formats a phone number with libphonenumber. Will return the formatted number and if it's a valid/complete number, will return the e164 value as well in the e164
field. Uses libphonenumber's parse function to verify if it's a valid number or not. This is useful if you want to format a number and also check if it's valid, in one step.
Optionally pass a PhoneNumberType
to format the number using either the mobile (default) or fixed line mask.
e164
will be null if the number is not valid.
class FormatPhoneResult {
String formattedNumber; // 1 (414) 444-4444
String e164; // +14144444444
}
required String country
You must provide the country used to format the text accurately.
The text formatter also takes 5 optional arguments:
-
PhoneNumberType? phoneNumberType
specify whether to format the phone number in the mobile format using the mask for mobile numbers, or the fixed line format. Can either bePhoneNumberType.mobile
orPhoneNumberType.fixedLine
. Defaults toPhoneNumberType.mobile
. -
PhoneNumberFormat? phoneNumberFormat
specify to format using the national or international format. Can either bePhoneNumberFormat.international
orPhoneNumberFormat.national
. Defaults toPhoneNumberFormat.international
. -
FutureOr Function(String val) onFormatFinished
Optionally get a notification at this callback of the final formatted value once all formatting is completed. This is useful if you want to do something else that is triggered after formatting is done. -
bool? inputContainsCountryCode
When true, mask will be applied assuming the input contains a country code in it. -
int? additionalDigits
You can tell the formatter to allow additional digits on the end of the mask. This is useful for some countries which have a similar mask but varying length of numbers. It's safe to set this to a value like 3 or 5 and you won't have to think about it again.
To use it, simply add LibPhonenumberTextFormatter()
to your TextField
's inputFormatters
list:
TextField(inputFormatters: [LibPhonenumberTextFormatter(
country: 'GB',
)])