// Copyright (c) 2010 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/autofill/autofill_profile.h" #include #include #include "base/stl_util-inl.h" #include "base/utf_string_conversions.h" #include "chrome/browser/autofill/address.h" #include "chrome/browser/autofill/autofill_manager.h" #include "chrome/browser/autofill/contact_info.h" #include "chrome/browser/autofill/fax_number.h" #include "chrome/browser/autofill/home_address.h" #include "chrome/browser/autofill/home_phone_number.h" #include "chrome/common/guid.h" #include "grit/generated_resources.h" #include "ui/base/l10n/l10n_util.h" namespace { void InitPersonalInfo(FormGroupMap* personal_info) { (*personal_info)[AutoFillType::CONTACT_INFO] = new ContactInfo(); (*personal_info)[AutoFillType::PHONE_HOME] = new HomePhoneNumber(); (*personal_info)[AutoFillType::PHONE_FAX] = new FaxNumber(); (*personal_info)[AutoFillType::ADDRESS_HOME] = new HomeAddress(); } // Maps |field_type| to a field type that can be directly stored in a profile // (in the sense that it makes sense to call |AutoFillProfile::SetInfo()| with // the returned field type as the first parameter. AutoFillFieldType GetEquivalentFieldType(AutoFillFieldType field_type) { // When billing information is requested from the profile we map to the // home address equivalents. switch (field_type) { case ADDRESS_BILLING_LINE1: return ADDRESS_HOME_LINE1; case ADDRESS_BILLING_LINE2: return ADDRESS_HOME_LINE2; case ADDRESS_BILLING_APT_NUM: return ADDRESS_HOME_APT_NUM; case ADDRESS_BILLING_CITY: return ADDRESS_HOME_CITY; case ADDRESS_BILLING_STATE: return ADDRESS_HOME_STATE; case ADDRESS_BILLING_ZIP: return ADDRESS_HOME_ZIP; case ADDRESS_BILLING_COUNTRY: return ADDRESS_HOME_COUNTRY; default: return field_type; } } // Like |GetEquivalentFieldType()| above, but also returns |NAME_FULL| for // first, middle, and last name field types. AutoFillFieldType GetEquivalentFieldTypeCollapsingNames( AutoFillFieldType field_type) { if (field_type == NAME_FIRST || field_type == NAME_MIDDLE || field_type == NAME_LAST) return NAME_FULL; return GetEquivalentFieldType(field_type); } // Fills |distinguishing_fields| with a list of fields to use when creating // labels that can help to distinguish between two profiles. Draws fields from // |suggested_fields| if it is non-NULL; otherwise returns a default list. // If |suggested_fields| is non-NULL, does not include |excluded_field| in the // list. Otherwise, |excluded_field| is ignored, and should be set to // |UNKNOWN_TYPE| by convention. The resulting list of fields is sorted in // decreasing order of importance. void GetFieldsForDistinguishingProfiles( const std::vector* suggested_fields, AutoFillFieldType excluded_field, std::vector* distinguishing_fields) { static const AutoFillFieldType kDefaultDistinguishingFields[] = { NAME_FULL, ADDRESS_HOME_LINE1, ADDRESS_HOME_CITY, ADDRESS_HOME_STATE, ADDRESS_HOME_ZIP, ADDRESS_HOME_COUNTRY, EMAIL_ADDRESS, PHONE_HOME_WHOLE_NUMBER, PHONE_FAX_WHOLE_NUMBER, COMPANY_NAME, }; if (!suggested_fields) { DCHECK_EQ(excluded_field, UNKNOWN_TYPE); distinguishing_fields->assign( kDefaultDistinguishingFields, kDefaultDistinguishingFields + arraysize(kDefaultDistinguishingFields)); return; } // Keep track of which fields we've seen so that we avoid duplicate entries. // Always ignore fields of unknown type and the excluded field. std::set seen_fields; seen_fields.insert(UNKNOWN_TYPE); seen_fields.insert(GetEquivalentFieldTypeCollapsingNames(excluded_field)); distinguishing_fields->clear(); for (std::vector::const_iterator it = suggested_fields->begin(); it != suggested_fields->end(); ++it) { AutoFillFieldType suggested_type = GetEquivalentFieldTypeCollapsingNames(*it); if (seen_fields.insert(suggested_type).second) distinguishing_fields->push_back(suggested_type); } // Special case: If the excluded field is a partial name (e.g. first name) and // the suggested fields include other name fields, include |NAME_FULL| in the // list of distinguishing fields as a last-ditch fallback. This allows us to // distinguish between profiles that are identical except for the name. if (excluded_field != NAME_FULL && GetEquivalentFieldTypeCollapsingNames(excluded_field) == NAME_FULL) { for (std::vector::const_iterator it = suggested_fields->begin(); it != suggested_fields->end(); ++it) { if (*it != excluded_field && GetEquivalentFieldTypeCollapsingNames(*it) == NAME_FULL) { distinguishing_fields->push_back(NAME_FULL); break; } } } } } // namespace AutoFillProfile::AutoFillProfile(const std::string& guid) : guid_(guid) { InitPersonalInfo(&personal_info_); } AutoFillProfile::AutoFillProfile() : guid_(guid::GenerateGUID()) { InitPersonalInfo(&personal_info_); } AutoFillProfile::AutoFillProfile(const AutoFillProfile& source) : FormGroup() { operator=(source); } AutoFillProfile::~AutoFillProfile() { STLDeleteContainerPairSecondPointers(personal_info_.begin(), personal_info_.end()); } void AutoFillProfile::GetPossibleFieldTypes( const string16& text, FieldTypeSet* possible_types) const { for (FormGroupMap::const_iterator iter = personal_info_.begin(); iter != personal_info_.end(); ++iter) { FormGroup* data = iter->second; DCHECK(data != NULL); data->GetPossibleFieldTypes(text, possible_types); } } void AutoFillProfile::GetAvailableFieldTypes( FieldTypeSet* available_types) const { for (FormGroupMap::const_iterator iter = personal_info_.begin(); iter != personal_info_.end(); ++iter) { FormGroup* data = iter->second; DCHECK(data != NULL); data->GetAvailableFieldTypes(available_types); } } string16 AutoFillProfile::GetFieldText(const AutoFillType& type) const { AutoFillType return_type(GetEquivalentFieldType(type.field_type())); FormGroupMap::const_iterator iter = personal_info_.find(return_type.group()); if (iter == personal_info_.end() || iter->second == NULL) return string16(); return iter->second->GetFieldText(return_type); } void AutoFillProfile::FindInfoMatches( const AutoFillType& type, const string16& info, std::vector* matched_text) const { if (matched_text == NULL) { DLOG(ERROR) << "NULL matched text passed in"; return; } string16 clean_info = StringToLowerASCII(CollapseWhitespace(info, false)); // If the field_type is unknown, then match against all field types. if (type.field_type() == UNKNOWN_TYPE) { FormGroupMap::const_iterator iter; for (iter = personal_info_.begin(); iter != personal_info_.end(); ++iter) { iter->second->FindInfoMatches(type, clean_info, matched_text); } } else { FormGroupMap::const_iterator iter = personal_info_.find(type.group()); DCHECK(iter != personal_info_.end() && iter->second != NULL); if (iter != personal_info_.end() && iter->second != NULL) iter->second->FindInfoMatches(type, clean_info, matched_text); } } void AutoFillProfile::SetInfo(const AutoFillType& type, const string16& value) { FormGroupMap::const_iterator iter = personal_info_.find(type.group()); if (iter == personal_info_.end() || iter->second == NULL) return; iter->second->SetInfo(type, CollapseWhitespace(value, false)); } FormGroup* AutoFillProfile::Clone() const { return new AutoFillProfile(*this); } const string16 AutoFillProfile::Label() const { return label_; } // static bool AutoFillProfile::AdjustInferredLabels( std::vector* profiles) { const size_t kMinimalFieldsShown = 2; std::vector created_labels; CreateInferredLabels(profiles, NULL, UNKNOWN_TYPE, kMinimalFieldsShown, &created_labels); DCHECK_EQ(profiles->size(), created_labels.size()); bool updated_labels = false; for (size_t i = 0; i < profiles->size(); ++i) { if ((*profiles)[i]->Label() != created_labels[i]) { updated_labels = true; (*profiles)[i]->set_label(created_labels[i]); } } return updated_labels; } // static void AutoFillProfile::CreateInferredLabels( const std::vector* profiles, const std::vector* suggested_fields, AutoFillFieldType excluded_field, size_t minimal_fields_shown, std::vector* created_labels) { DCHECK(profiles); DCHECK(created_labels); std::vector fields_to_use; GetFieldsForDistinguishingProfiles(suggested_fields, excluded_field, &fields_to_use); // Construct the default label for each profile. Also construct a map that // associates each label with the profiles that have this label. This map is // then used to detect which labels need further differentiating fields. std::map > labels; for (size_t i = 0; i < profiles->size(); ++i) { string16 label = (*profiles)[i]->ConstructInferredLabel(fields_to_use, minimal_fields_shown); labels[label].push_back(i); } created_labels->resize(profiles->size()); for (std::map >::const_iterator it = labels.begin(); it != labels.end(); ++it) { if (it->second.size() == 1) { // This label is unique, so use it without any further ado. string16 label = it->first; size_t profile_index = it->second.front(); (*created_labels)[profile_index] = label; } else { // We have more than one profile with the same label, so add // differentiating fields. CreateDifferentiatingLabels(*profiles, it->second, fields_to_use, minimal_fields_shown, created_labels); } } } bool AutoFillProfile::IsEmpty() const { FieldTypeSet types; GetAvailableFieldTypes(&types); return types.empty(); } void AutoFillProfile::operator=(const AutoFillProfile& source) { if (this == &source) return; label_ = source.label_; guid_ = source.guid_; STLDeleteContainerPairSecondPointers(personal_info_.begin(), personal_info_.end()); personal_info_.clear(); FormGroupMap::const_iterator iter; for (iter = source.personal_info_.begin(); iter != source.personal_info_.end(); ++iter) { personal_info_[iter->first] = iter->second->Clone(); } } int AutoFillProfile::Compare(const AutoFillProfile& profile) const { // The following AutoFill field types are the only types we store in the WebDB // so far, so we're only concerned with matching these types in the profile. const AutoFillFieldType types[] = { NAME_FIRST, NAME_MIDDLE, NAME_LAST, EMAIL_ADDRESS, COMPANY_NAME, ADDRESS_HOME_LINE1, ADDRESS_HOME_LINE2, ADDRESS_HOME_CITY, ADDRESS_HOME_STATE, ADDRESS_HOME_ZIP, ADDRESS_HOME_COUNTRY, PHONE_HOME_NUMBER, PHONE_FAX_NUMBER }; for (size_t index = 0; index < arraysize(types); ++index) { int comparison = GetFieldText(AutoFillType(types[index])).compare( profile.GetFieldText(AutoFillType(types[index]))); if (comparison != 0) return comparison; } return 0; } bool AutoFillProfile::operator==(const AutoFillProfile& profile) const { if (label_ != profile.label_ || guid_ != profile.guid_) return false; return Compare(profile) == 0; } bool AutoFillProfile::operator!=(const AutoFillProfile& profile) const { return !operator==(profile); } const string16 AutoFillProfile::PrimaryValue() const { return GetFieldText(AutoFillType(NAME_FULL)) + GetFieldText(AutoFillType(ADDRESS_HOME_LINE1)) + GetFieldText(AutoFillType(ADDRESS_HOME_LINE2)) + GetFieldText(AutoFillType(EMAIL_ADDRESS)); } string16 AutoFillProfile::ConstructInferredLabel( const std::vector& included_fields, size_t num_fields_to_use) const { const string16 separator = l10n_util::GetStringUTF16(IDS_AUTOFILL_DIALOG_ADDRESS_SUMMARY_SEPARATOR); string16 label; size_t num_fields_used = 0; for (std::vector::const_iterator it = included_fields.begin(); it != included_fields.end() && num_fields_used < num_fields_to_use; ++it) { string16 field = GetFieldText(AutoFillType(*it)); if (field.empty()) continue; if (!label.empty()) label.append(separator); // Fax number has special format, to indicate that this is a fax number. if (*it == PHONE_FAX_WHOLE_NUMBER) { field = l10n_util::GetStringFUTF16( IDS_AUTOFILL_DIALOG_ADDRESS_SUMMARY_FAX_FORMAT, field); } label.append(field); ++num_fields_used; } return label; } // static void AutoFillProfile::CreateDifferentiatingLabels( const std::vector& profiles, const std::list& indices, const std::vector& fields, size_t num_fields_to_include, std::vector* created_labels) { // For efficiency, we first construct a map of fields to their text values and // each value's frequency. std::map > field_text_frequencies_by_field; for (std::vector::const_iterator field = fields.begin(); field != fields.end(); ++field) { std::map& field_text_frequencies = field_text_frequencies_by_field[*field]; for (std::list::const_iterator it = indices.begin(); it != indices.end(); ++it) { const AutoFillProfile* profile = profiles[*it]; string16 field_text = profile->GetFieldText(AutoFillType(*field)); // If this label is not already in the map, add it with frequency 0. if (!field_text_frequencies.count(field_text)) field_text_frequencies[field_text] = 0; // Now, increment the frequency for this label. ++field_text_frequencies[field_text]; } } // Now comes the meat of the algorithm. For each profile, we scan the list of // fields to use, looking for two things: // 1. A (non-empty) field that differentiates the profile from all others // 2. At least |num_fields_to_include| non-empty fields // Before we've satisfied condition (2), we include all fields, even ones that // are identical across all the profiles. Once we've satisfied condition (2), // we only include fields that that have at last two distinct values. for (std::list::const_iterator it = indices.begin(); it != indices.end(); ++it) { const AutoFillProfile* profile = profiles[*it]; std::vector label_fields; bool found_differentiating_field = false; for (std::vector::const_iterator field = fields.begin(); field != fields.end(); ++field) { // Skip over empty fields. string16 field_text = profile->GetFieldText(AutoFillType(*field)); if (field_text.empty()) continue; std::map& field_text_frequencies = field_text_frequencies_by_field[*field]; found_differentiating_field |= !field_text_frequencies.count(string16()) && (field_text_frequencies[field_text] == 1); // Once we've found enough non-empty fields, skip over any remaining // fields that are identical across all the profiles. if (label_fields.size() >= num_fields_to_include && (field_text_frequencies.size() == 1)) continue; label_fields.push_back(*field); // If we've (1) found a differentiating field and (2) found at least // |num_fields_to_include| non-empty fields, we're done! if (found_differentiating_field && label_fields.size() >= num_fields_to_include) break; } (*created_labels)[*it] = profile->ConstructInferredLabel(label_fields, label_fields.size()); } } // So we can compare AutoFillProfiles with EXPECT_EQ(). std::ostream& operator<<(std::ostream& os, const AutoFillProfile& profile) { return os << UTF16ToUTF8(profile.Label()) << " " << profile.guid() << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(NAME_FIRST))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(NAME_MIDDLE))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(NAME_LAST))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(EMAIL_ADDRESS))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(COMPANY_NAME))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(ADDRESS_HOME_LINE1))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(ADDRESS_HOME_LINE2))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(ADDRESS_HOME_CITY))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(ADDRESS_HOME_STATE))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(ADDRESS_HOME_ZIP))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType(ADDRESS_HOME_COUNTRY))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType( PHONE_HOME_WHOLE_NUMBER))) << " " << UTF16ToUTF8(profile.GetFieldText(AutoFillType( PHONE_FAX_WHOLE_NUMBER))); }