blob: ebb79a2a6f06603a609ff5f3c9ac315df2e2aeac [file] [log] [blame]
// 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();
std::vector<std::string> 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<ServerFieldType>& 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<base::string16> 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