// Copyright (c) 2012 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. #include "chrome/browser/ui/autofill/data_model_wrapper.h" #include "base/bind.h" #include "base/callback.h" #include "base/strings/string_util.h" #include "base/strings/utf_string_conversions.h" #include "chrome/browser/browser_process.h" #include "chrome/browser/ui/autofill/autofill_dialog_i18n_input.h" #include "chrome/browser/ui/autofill/autofill_dialog_models.h" #include "components/autofill/content/browser/wallet/full_wallet.h" #include "components/autofill/content/browser/wallet/wallet_address.h" #include "components/autofill/content/browser/wallet/wallet_items.h" #include "components/autofill/core/browser/address_i18n.h" #include "components/autofill/core/browser/autofill_country.h" #include "components/autofill/core/browser/autofill_data_model.h" #include "components/autofill/core/browser/autofill_field.h" #include "components/autofill/core/browser/autofill_profile.h" #include "components/autofill/core/browser/autofill_type.h" #include "components/autofill/core/browser/credit_card.h" #include "components/autofill/core/browser/form_structure.h" #include "third_party/libaddressinput/src/cpp/include/libaddressinput/address_data.h" #include "third_party/libaddressinput/src/cpp/include/libaddressinput/address_field.h" #include "third_party/libaddressinput/src/cpp/include/libaddressinput/address_formatter.h" #include "ui/base/resource/resource_bundle.h" #include "ui/gfx/image/image.h" namespace autofill { DataModelWrapper::~DataModelWrapper() {} void DataModelWrapper::FillInputs(DetailInputs* inputs) { for (size_t i = 0; i < inputs->size(); ++i) { (*inputs)[i].initial_value = GetInfoForDisplay(AutofillType((*inputs)[i].type)); } } base::string16 DataModelWrapper::GetInfoForDisplay(const AutofillType& type) const { return GetInfo(type); } gfx::Image DataModelWrapper::GetIcon() { return gfx::Image(); } bool DataModelWrapper::GetDisplayText( base::string16* vertically_compact, base::string16* horizontally_compact) { base::string16 phone = GetInfoForDisplay(AutofillType(PHONE_HOME_WHOLE_NUMBER)); if (phone.empty()) return false; // Format the address. scoped_ptr< ::i18n::addressinput::AddressData> address_data = i18n::CreateAddressData( base::Bind(&DataModelWrapper::GetInfo, base::Unretained(this))); address_data->language_code = GetLanguageCode(); // Interactive autofill dialog does not display organization field. address_data->organization.clear(); std::vector lines; ::i18n::addressinput::GetFormattedNationalAddress(*address_data, &lines); std::string single_line; ::i18n::addressinput::GetFormattedNationalAddressLine( *address_data, &single_line); // Email and phone number aren't part of address formatting. base::string16 non_address_info; base::string16 email = GetInfoForDisplay(AutofillType(EMAIL_ADDRESS)); if (!email.empty()) non_address_info += base::ASCIIToUTF16("\n") + email; non_address_info += base::ASCIIToUTF16("\n") + phone; *vertically_compact = base::UTF8ToUTF16(single_line) + non_address_info; *horizontally_compact = base::UTF8ToUTF16(JoinString(lines, "\n")) + non_address_info; return true; } bool DataModelWrapper::FillFormStructure( const std::vector& types, const FormStructure::InputFieldComparator& compare, FormStructure* form_structure) const { return form_structure->FillFields( types, compare, base::Bind(&DataModelWrapper::GetInfo, base::Unretained(this)), GetLanguageCode(), g_browser_process->GetApplicationLocale()); } DataModelWrapper::DataModelWrapper() {} // AutofillProfileWrapper AutofillProfileWrapper::AutofillProfileWrapper(const AutofillProfile* profile) : profile_(profile), variant_group_(NO_GROUP), variant_(0) {} AutofillProfileWrapper::AutofillProfileWrapper( const AutofillProfile* profile, const AutofillType& type, size_t variant) : profile_(profile), variant_group_(type.group()), variant_(variant) {} AutofillProfileWrapper::~AutofillProfileWrapper() {} base::string16 AutofillProfileWrapper::GetInfo(const AutofillType& type) const { // Requests for the user's credit card are filled from the billing address, // but the AutofillProfile class doesn't know how to fill credit card // fields. So, request for the corresponding profile type instead. AutofillType effective_type = type; if (type.GetStorableType() == CREDIT_CARD_NAME) effective_type = AutofillType(NAME_BILLING_FULL); size_t variant = GetVariantForType(effective_type); const std::string& app_locale = g_browser_process->GetApplicationLocale(); return profile_->GetInfoForVariant(effective_type, variant, app_locale); } base::string16 AutofillProfileWrapper::GetInfoForDisplay( const AutofillType& type) const { // We display the "raw" phone number which contains user-defined formatting. if (type.GetStorableType() == PHONE_HOME_WHOLE_NUMBER) { std::vector values; profile_->GetRawMultiInfo(type.GetStorableType(), &values); const base::string16& phone_number = values[GetVariantForType(type)]; // If there is no user-defined formatting at all, add some standard // formatting. if (base::ContainsOnlyChars(phone_number, base::ASCIIToUTF16("+0123456789"))) { std::string region = base::UTF16ToASCII( GetInfo(AutofillType(HTML_TYPE_COUNTRY_CODE, HTML_MODE_NONE))); i18n::PhoneObject phone(phone_number, region); base::string16 formatted_number = phone.GetFormattedNumber(); // Formatting may fail. if (!formatted_number.empty()) return formatted_number; } return phone_number; } return DataModelWrapper::GetInfoForDisplay(type); } const std::string& AutofillProfileWrapper::GetLanguageCode() const { return profile_->language_code(); } size_t AutofillProfileWrapper::GetVariantForType(const AutofillType& type) const { if (type.group() == variant_group_) return variant_; return 0; } // AutofillShippingAddressWrapper AutofillShippingAddressWrapper::AutofillShippingAddressWrapper( const AutofillProfile* profile) : AutofillProfileWrapper(profile) {} AutofillShippingAddressWrapper::~AutofillShippingAddressWrapper() {} base::string16 AutofillShippingAddressWrapper::GetInfo( const AutofillType& type) const { // Shipping addresses don't have email addresses associated with them. if (type.GetStorableType() == EMAIL_ADDRESS) return base::string16(); return AutofillProfileWrapper::GetInfo(type); } // AutofillCreditCardWrapper AutofillCreditCardWrapper::AutofillCreditCardWrapper(const CreditCard* card) : card_(card) {} AutofillCreditCardWrapper::~AutofillCreditCardWrapper() {} base::string16 AutofillCreditCardWrapper::GetInfo(const AutofillType& type) const { if (type.group() != CREDIT_CARD) return base::string16(); if (type.GetStorableType() == CREDIT_CARD_EXP_MONTH) return MonthComboboxModel::FormatMonth(card_->expiration_month()); return card_->GetInfo(type, g_browser_process->GetApplicationLocale()); } gfx::Image AutofillCreditCardWrapper::GetIcon() { ui::ResourceBundle& rb = ui::ResourceBundle::GetSharedInstance(); return rb.GetImageNamed(CreditCard::IconResourceId(card_->type())); } bool AutofillCreditCardWrapper::GetDisplayText( base::string16* vertically_compact, base::string16* horizontally_compact) { if (!card_->IsValid()) return false; *vertically_compact = *horizontally_compact = card_->TypeAndLastFourDigits(); return true; } const std::string& AutofillCreditCardWrapper::GetLanguageCode() const { // Formatting a credit card for display does not depend on language code. return base::EmptyString(); } // WalletAddressWrapper WalletAddressWrapper::WalletAddressWrapper( const wallet::Address* address) : address_(address) {} WalletAddressWrapper::~WalletAddressWrapper() {} base::string16 WalletAddressWrapper::GetInfo(const AutofillType& type) const { // Reachable from DataModelWrapper::GetDisplayText(). if (type.GetStorableType() == EMAIL_ADDRESS) return base::string16(); return address_->GetInfo(type, g_browser_process->GetApplicationLocale()); } base::string16 WalletAddressWrapper::GetInfoForDisplay(const AutofillType& type) const { if (type.GetStorableType() == PHONE_HOME_WHOLE_NUMBER) return address_->DisplayPhoneNumber(); return DataModelWrapper::GetInfoForDisplay(type); } bool WalletAddressWrapper::GetDisplayText( base::string16* vertically_compact, base::string16* horizontally_compact) { if (!address_->is_complete_address()) return false; return DataModelWrapper::GetDisplayText(vertically_compact, horizontally_compact); } const std::string& WalletAddressWrapper::GetLanguageCode() const { return address_->language_code(); } // WalletInstrumentWrapper WalletInstrumentWrapper::WalletInstrumentWrapper( const wallet::WalletItems::MaskedInstrument* instrument) : instrument_(instrument) {} WalletInstrumentWrapper::~WalletInstrumentWrapper() {} base::string16 WalletInstrumentWrapper::GetInfo(const AutofillType& type) const { // Reachable from DataModelWrapper::GetDisplayText(). if (type.GetStorableType() == EMAIL_ADDRESS) return base::string16(); if (type.GetStorableType() == CREDIT_CARD_EXP_MONTH) return MonthComboboxModel::FormatMonth(instrument_->expiration_month()); return instrument_->GetInfo(type, g_browser_process->GetApplicationLocale()); } base::string16 WalletInstrumentWrapper::GetInfoForDisplay( const AutofillType& type) const { if (type.GetStorableType() == PHONE_HOME_WHOLE_NUMBER) return instrument_->address().DisplayPhoneNumber(); return DataModelWrapper::GetInfoForDisplay(type); } gfx::Image WalletInstrumentWrapper::GetIcon() { return instrument_->CardIcon(); } bool WalletInstrumentWrapper::GetDisplayText( base::string16* vertically_compact, base::string16* horizontally_compact) { // TODO(dbeam): handle other instrument statuses? http://crbug.com/233048 if (instrument_->status() == wallet::WalletItems::MaskedInstrument::EXPIRED || !instrument_->address().is_complete_address()) { return false; } if (!DataModelWrapper::GetDisplayText(vertically_compact, horizontally_compact)) { return false; } // TODO(estade): descriptive_name() is user-provided. Should we use it or // just type + last 4 digits? base::string16 line1 = instrument_->descriptive_name() + base::ASCIIToUTF16("\n"); *vertically_compact = line1 + *vertically_compact; *horizontally_compact = line1 + *horizontally_compact; return true; } const std::string& WalletInstrumentWrapper::GetLanguageCode() const { return instrument_->address().language_code(); } // FullWalletBillingWrapper FullWalletBillingWrapper::FullWalletBillingWrapper( wallet::FullWallet* full_wallet) : full_wallet_(full_wallet) { DCHECK(full_wallet_); } FullWalletBillingWrapper::~FullWalletBillingWrapper() {} base::string16 FullWalletBillingWrapper::GetInfo(const AutofillType& type) const { return full_wallet_->GetInfo(g_browser_process->GetApplicationLocale(), type); } bool FullWalletBillingWrapper::GetDisplayText( base::string16* vertically_compact, base::string16* horizontally_compact) { // TODO(dbeam): handle other required actions? http://crbug.com/163508 if (full_wallet_->HasRequiredAction(wallet::UPDATE_EXPIRATION_DATE)) return false; return DataModelWrapper::GetDisplayText(vertically_compact, horizontally_compact); } const std::string& FullWalletBillingWrapper::GetLanguageCode() const { // Can be NULL if there are required actions. return full_wallet_->billing_address() ? full_wallet_->billing_address()->language_code() : base::EmptyString(); } // FullWalletShippingWrapper FullWalletShippingWrapper::FullWalletShippingWrapper( wallet::FullWallet* full_wallet) : full_wallet_(full_wallet) { DCHECK(full_wallet_); } FullWalletShippingWrapper::~FullWalletShippingWrapper() {} base::string16 FullWalletShippingWrapper::GetInfo( const AutofillType& type) const { return full_wallet_->shipping_address()->GetInfo( type, g_browser_process->GetApplicationLocale()); } const std::string& FullWalletShippingWrapper::GetLanguageCode() const { // Can be NULL if there are required actions or shipping address is not // required. return full_wallet_->shipping_address() ? full_wallet_->shipping_address()->language_code() : base::EmptyString(); } // I18nAddressDataWrapper I18nAddressDataWrapper::I18nAddressDataWrapper( const ::i18n::addressinput::AddressData* address) : address_(address) {} I18nAddressDataWrapper::~I18nAddressDataWrapper() {} base::string16 I18nAddressDataWrapper::GetInfo(const AutofillType& type) const { ::i18n::addressinput::AddressField field; if (!i18n::FieldForType(type.GetStorableType(), &field)) return base::string16(); if (field == ::i18n::addressinput::STREET_ADDRESS) return base::string16(); if (field == ::i18n::addressinput::COUNTRY) { return AutofillCountry(address_->region_code, g_browser_process->GetApplicationLocale()).name(); } return base::UTF8ToUTF16(address_->GetFieldValue(field)); } const std::string& I18nAddressDataWrapper::GetLanguageCode() const { return address_->language_code; } } // namespace autofill