Chromium Code Reviews| Index: chrome/browser/autofill/phone_number_i18n.cc |
| =================================================================== |
| --- chrome/browser/autofill/phone_number_i18n.cc (revision 89739) |
| +++ chrome/browser/autofill/phone_number_i18n.cc (working copy) |
| @@ -4,6 +4,8 @@ |
| #include "chrome/browser/autofill/phone_number_i18n.h" |
| +#include <map> |
|
Ilya Sherman
2011/06/21 00:31:26
Do you still need this include?
GeorgeY
2011/06/21 23:24:44
no, removed
|
| + |
| #include "base/basictypes.h" |
| #include "base/logging.h" |
| #include "base/stringprintf.h" |
| @@ -38,41 +40,16 @@ |
| return i18n::phonenumbers::PhoneNumberUtil::NATIONAL; |
| } |
| -} // namespace |
| - |
| -namespace autofill_i18n { |
| - |
| -string16 NormalizePhoneNumber(const string16& value, |
| - std::string const& locale) { |
| - string16 number; |
| - string16 city_code; |
| - string16 country_code; |
| - string16 result; |
| - // Full number - parse it, split it and re-combine into canonical form. |
| - if (!ParsePhoneNumber(value, locale, &country_code, &city_code, &number)) |
| - return string16(); // Parsing failed - do not store phone. |
| - if (!autofill_i18n::ConstructPhoneNumber( |
| - country_code, city_code, number, |
| - locale, |
| - (country_code.empty() ? |
| - autofill_i18n::NATIONAL : autofill_i18n::INTERNATIONAL), |
| - &result)) { |
| - // Reconstruction failed - do not store phone. |
| - return string16(); |
| - } |
| - std::string result_utf8(UTF16ToUTF8(result)); |
| - i18n::phonenumbers::PhoneNumberUtil::NormalizeDigitsOnly(&result_utf8); |
| - return UTF8ToUTF16(result_utf8); |
| -} |
| - |
| -bool ParsePhoneNumber(const string16& value, |
| - const std::string& locale, |
| - string16* country_code, |
| - string16* city_code, |
| - string16* number) { |
| +bool ParsePhoneNumberInternal(const string16& value, |
| + const std::string& locale, |
| + string16* country_code, |
| + string16* city_code, |
| + string16* number, |
| + i18n::phonenumbers::PhoneNumber* i18n_number) { |
| DCHECK(number); |
| DCHECK(city_code); |
| DCHECK(country_code); |
| + DCHECK(i18n_number); |
| number->clear(); |
| city_code->clear(); |
| @@ -81,38 +58,37 @@ |
| std::string number_text(UTF16ToUTF8(value)); |
| // Parse phone number based on the locale. |
| - i18n::phonenumbers::PhoneNumber i18n_number; |
| i18n::phonenumbers::PhoneNumberUtil* phone_util = |
| i18n::phonenumbers::PhoneNumberUtil::GetInstance(); |
| DCHECK(phone_util); |
| if (phone_util->Parse(number_text, SanitizeLocaleCode(locale).c_str(), |
| - &i18n_number) != |
| + i18n_number) != |
| i18n::phonenumbers::PhoneNumberUtil::NO_PARSING_ERROR) { |
| return false; |
| } |
| i18n::phonenumbers::PhoneNumberUtil::ValidationResult validation = |
| - phone_util->IsPossibleNumberWithReason(i18n_number); |
| + phone_util->IsPossibleNumberWithReason(*i18n_number); |
| if (validation != i18n::phonenumbers::PhoneNumberUtil::IS_POSSIBLE) |
| return false; |
| // This verifies that number has a valid area code (that in some cases could |
| // be empty) for parsed country code. Also verifies that this is a valid |
| // number (in US 1234567 is not valid, because numbers do not start with 1). |
| - if (!phone_util->IsValidNumber(i18n_number)) |
| + if (!phone_util->IsValidNumber(*i18n_number)) |
| return false; |
| std::string national_significant_number; |
| - phone_util->GetNationalSignificantNumber(i18n_number, |
| + phone_util->GetNationalSignificantNumber(*i18n_number, |
| &national_significant_number); |
| std::string area_code; |
| std::string subscriber_number; |
| - int area_length = phone_util->GetLengthOfGeographicalAreaCode(i18n_number); |
| + int area_length = phone_util->GetLengthOfGeographicalAreaCode(*i18n_number); |
| int destination_length = |
| - phone_util->GetLengthOfNationalDestinationCode(i18n_number); |
| + phone_util->GetLengthOfNationalDestinationCode(*i18n_number); |
| // Some phones have a destination code in lieu of area code: mobile operators |
| // in Europe, toll and toll-free numbers in USA, etc. From our point of view |
| // these two types of codes are the same. |
| @@ -132,9 +108,9 @@ |
| string16 normalized_number(UTF8ToUTF16(number_text)); |
| // Check if parsed number has country code and it was not inferred from the |
| // locale. |
| - if (i18n_number.has_country_code()) { |
| - *country_code = UTF8ToUTF16(base::StringPrintf("%d", |
| - i18n_number.country_code())); |
| + if (i18n_number->has_country_code()) { |
| + *country_code = UTF8ToUTF16( |
| + base::StringPrintf("%d", i18n_number->country_code())); |
| if (normalized_number.length() <= national_significant_number.length() && |
| (normalized_number.length() < country_code->length() || |
| normalized_number.compare(0, country_code->length(), *country_code))) { |
| @@ -145,6 +121,43 @@ |
| return true; |
| } |
| +} // namespace |
| + |
| +namespace autofill_i18n { |
| + |
| +string16 NormalizePhoneNumber(const string16& value, |
| + std::string const& locale) { |
| + string16 number; |
| + string16 city_code; |
| + string16 country_code; |
| + string16 result; |
| + // Full number - parse it, split it and re-combine into canonical form. |
| + if (!ParsePhoneNumber(value, locale, &country_code, &city_code, &number)) |
| + return string16(); // Parsing failed - do not store phone. |
| + if (!autofill_i18n::ConstructPhoneNumber( |
| + country_code, city_code, number, |
| + locale, |
| + (country_code.empty() ? |
| + autofill_i18n::NATIONAL : autofill_i18n::INTERNATIONAL), |
| + &result)) { |
| + // Reconstruction failed - do not store phone. |
| + return string16(); |
| + } |
| + std::string result_utf8(UTF16ToUTF8(result)); |
| + i18n::phonenumbers::PhoneNumberUtil::NormalizeDigitsOnly(&result_utf8); |
| + return UTF8ToUTF16(result_utf8); |
| +} |
| + |
| +bool ParsePhoneNumber(const string16& value, |
| + const std::string& locale, |
| + string16* country_code, |
| + string16* city_code, |
| + string16* number) { |
| + i18n::phonenumbers::PhoneNumber i18n_number; |
| + return ParsePhoneNumberInternal(value, locale, country_code, city_code, |
| + number, &i18n_number); |
| +} |
| + |
| bool ConstructPhoneNumber(const string16& country_code, |
| const string16& city_code, |
| const string16& number, |
| @@ -265,5 +278,105 @@ |
| return ComparePhones(number_a, number_b, country_code) == PHONES_EQUAL; |
| } |
| +PhoneObject::PhoneObject(const string16& number, const std::string& locale) |
| + : locale_(SanitizeLocaleCode(locale)), |
| + i18n_number_(NULL) { |
| + scoped_ptr<i18n::phonenumbers::PhoneNumber> |
| + i18n_number(new i18n::phonenumbers::PhoneNumber); |
| + if (ParsePhoneNumberInternal(number, locale_, &country_code_, &city_code_, |
| + &number_, i18n_number.get())) { |
| + // Phone successfully parsed - set |i18n_number_| object, |whole_number_| |
| + // will be set on the first call to GetWholeNumber(). |
| + i18n_number_.reset(i18n_number.release()); |
| + } else { |
| + // Parsing failed. Store passed phone "as is" into |whole_number_|. |
| + whole_number_ = number; |
| + } |
| +} |
| + |
| +PhoneObject::PhoneObject(const PhoneObject& other) |
| + : i18n_number_(NULL) { |
| + *this = other; |
| +} |
| + |
| +PhoneObject::PhoneObject() |
| + : i18n_number_(NULL) { |
| +} |
| + |
| +PhoneObject::~PhoneObject() { |
| +} |
| + |
| +string16 PhoneObject::GetCountryCode() const { |
| + return country_code_; |
| +} |
| + |
| +string16 PhoneObject::GetCityCode() const { |
| + return city_code_; |
| +} |
| + |
| +string16 PhoneObject::GetNumber() const { |
| + return number_; |
| +} |
| + |
| +string16 PhoneObject::GetWholeNumber() const { |
| + if (i18n_number_.get() && whole_number_.empty()) { |
| + i18n::phonenumbers::PhoneNumberUtil::PhoneNumberFormat format = |
| + i18n::phonenumbers::PhoneNumberUtil::INTERNATIONAL; |
| + if (country_code_.empty()) |
| + format = i18n::phonenumbers::PhoneNumberUtil::NATIONAL; |
| + |
| + std::string formatted_number; |
| + i18n::phonenumbers::PhoneNumberUtil* phone_util = |
| + i18n::phonenumbers::PhoneNumberUtil::GetInstance(); |
| + phone_util->Format(*i18n_number_, format, &formatted_number); |
| + i18n::phonenumbers::PhoneNumberUtil::NormalizeDigitsOnly(&formatted_number); |
| + whole_number_ = UTF8ToUTF16(formatted_number); |
| + } |
| + return whole_number_; |
| +} |
| + |
| +PhoneMatch PhoneObject::ComparePhones(const string16& phone) const { |
| + return ComparePhones(PhoneObject(phone, locale_)); |
| +} |
| + |
| +PhoneMatch PhoneObject::ComparePhones(const PhoneObject& phone) const { |
| + if (!i18n_number_.get() || !phone.i18n_number_.get()) { |
| + if (GetWholeNumber().empty()) |
| + return PHONES_NOT_EQUAL; |
| + return (GetWholeNumber() == phone.GetWholeNumber()) ? PHONES_EQUAL : |
| + PHONES_NOT_EQUAL; |
| + } |
| + |
| + i18n::phonenumbers::PhoneNumberUtil* phone_util = |
| + i18n::phonenumbers::PhoneNumberUtil::GetInstance(); |
| + switch (phone_util->IsNumberMatch(*i18n_number_, *(phone.i18n_number_))) { |
| + case i18n::phonenumbers::PhoneNumberUtil::INVALID_NUMBER: |
| + case i18n::phonenumbers::PhoneNumberUtil::NO_MATCH: |
| + return PHONES_NOT_EQUAL; |
| + case i18n::phonenumbers::PhoneNumberUtil::SHORT_NSN_MATCH: |
| + return PHONES_SUBMATCH; |
| + case i18n::phonenumbers::PhoneNumberUtil::NSN_MATCH: |
| + case i18n::phonenumbers::PhoneNumberUtil::EXACT_MATCH: |
| + return PHONES_EQUAL; |
| + default: |
| + NOTREACHED(); |
| + } |
| + return PHONES_NOT_EQUAL; |
| +} |
| + |
| +PhoneObject& PhoneObject::operator=(const PhoneObject& other) { |
| + if (this == &other) |
| + return *this; |
| + country_code_ = other.country_code_; |
| + city_code_ = other.city_code_; |
| + number_ = other.number_; |
| + locale_ = other.locale_; |
| + if (other.i18n_number_.get()) { |
| + i18n_number_.reset(new i18n::phonenumbers::PhoneNumber( |
| + *other.i18n_number_)); |
| + } |
| + return *this; |
| +} |
| + |
| } // namespace autofill_i18n |