 Chromium Code Reviews
 Chromium Code Reviews Issue 1012363002:
  Autofill: Recognize month/year selects when searching for credit cards.  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@master
    
  
    Issue 1012363002:
  Autofill: Recognize month/year selects when searching for credit cards.  (Closed) 
  Base URL: https://chromium.googlesource.com/chromium/src.git@master| Index: components/autofill/core/browser/credit_card_field.cc | 
| diff --git a/components/autofill/core/browser/credit_card_field.cc b/components/autofill/core/browser/credit_card_field.cc | 
| index ba9c001022dc69ced0d5207c70f5d64b26d4550b..1c8d02ef8b696f9f5bd28ae4d568a206f98fd7c5 100644 | 
| --- a/components/autofill/core/browser/credit_card_field.cc | 
| +++ b/components/autofill/core/browser/credit_card_field.cc | 
| @@ -7,19 +7,55 @@ | 
| #include <stddef.h> | 
| #include "base/memory/scoped_ptr.h" | 
| +#include "base/stl_util.h" | 
| #include "base/strings/string16.h" | 
| +#include "base/strings/string_number_conversions.h" | 
| #include "base/strings/string_util.h" | 
| #include "base/strings/utf_string_conversions.h" | 
| +#include "base/time/time.h" | 
| #include "components/autofill/core/browser/autofill_field.h" | 
| #include "components/autofill/core/browser/autofill_regex_constants.h" | 
| +#include "components/autofill/core/browser/autofill_regexes.h" | 
| #include "components/autofill/core/browser/autofill_scanner.h" | 
| #include "components/autofill/core/browser/field_types.h" | 
| namespace autofill { | 
| +namespace { | 
| + | 
| // Credit card numbers are at most 19 digits in length. | 
| // [Ref: http://en.wikipedia.org/wiki/Bank_card_number] | 
| -static const size_t kMaxValidCardNumberSize = 19; | 
| +const size_t kMaxValidCardNumberSize = 19; | 
| + | 
| +// Look for the vector |regex_needles| in |haystack|. Returns true if a | 
| +// consecutive section of |haystack| matches |regex_needles|. | 
| +bool FindConsecutiveStrings(const std::vector<base::string16>& regex_needles, | 
| + const std::vector<base::string16>& haystack) { | 
| + if (regex_needles.empty() || haystack.empty()) | 
| + return false; | 
| + | 
| + auto haystack_remaining_it = haystack.begin(); | 
| + size_t haystack_remaining_size = haystack.size(); | 
| + while (regex_needles.size() <= haystack_remaining_size) { | 
| 
Evan Stade
2015/03/18 04:29:25
for (size_t i = 0; i < haystack.size() - regex_nee
 
Lei Zhang
2015/03/18 18:08:56
Done.
 | 
| + auto haystack_it = haystack_remaining_it; | 
| + bool found_match = true; | 
| + for (const base::string16& regex : regex_needles) { | 
| + if (!MatchesPattern(*haystack_it, regex)) { | 
| + found_match = false; | 
| + break; | 
| + } | 
| + ++haystack_it; | 
| + } | 
| + if (found_match) | 
| + return true; | 
| + | 
| + ++haystack_remaining_it; | 
| + --haystack_remaining_size; | 
| + } | 
| + return false; | 
| +} | 
| + | 
| +} | 
| // static | 
| scoped_ptr<FormField> CreditCardField::Parse(AutofillScanner* scanner) { | 
| @@ -157,6 +193,68 @@ scoped_ptr<FormField> CreditCardField::Parse(AutofillScanner* scanner) { | 
| return nullptr; | 
| } | 
| +// static | 
| +bool CreditCardField::LikelyCardMonthSelectField(AutofillScanner* scanner) { | 
| + if (scanner->IsEnd()) | 
| + return false; | 
| + | 
| + AutofillField* field = scanner->Cursor(); | 
| + if (!MatchesFormControlType(field->form_control_type, MATCH_SELECT)) | 
| + return false; | 
| + | 
| + if (field->option_values.size() < 12 || field->option_values.size() > 13) | 
| + return false; | 
| + | 
| + // Filter out years. | 
| + const base::string16 kNumericalYearRe = | 
| + base::ASCIIToUTF16("[1-9][0-9][0-9][0-9]"); | 
| + for (const auto& value : field->option_values) { | 
| + if (MatchesPattern(value, kNumericalYearRe)) | 
| + return false; | 
| + } | 
| + for (const auto& value : field->option_contents) { | 
| + if (MatchesPattern(value, kNumericalYearRe)) | 
| + return false; | 
| + } | 
| + | 
| + // Look for numerical months. | 
| + const base::string16 kNumericalMonthRe = base::ASCIIToUTF16("12"); | 
| + if (MatchesPattern(field->option_values.back(), kNumericalMonthRe) || | 
| + MatchesPattern(field->option_contents.back(), kNumericalMonthRe)) { | 
| + return true; | 
| + } | 
| + | 
| + // Maybe do more matches here. e.g. look for (translated) December. | 
| + | 
| + // Unsure? Return false. | 
| + return false; | 
| +} | 
| + | 
| +// static | 
| +bool CreditCardField::LikelyCardYearSelectField(AutofillScanner* scanner) { | 
| + if (scanner->IsEnd()) | 
| + return false; | 
| + | 
| + AutofillField* field = scanner->Cursor(); | 
| + if (!MatchesFormControlType(field->form_control_type, MATCH_SELECT)) | 
| + return false; | 
| + | 
| + const base::Time time_now = base::Time::Now(); | 
| + base::Time::Exploded time_exploded; | 
| + time_now.UTCExplode(&time_exploded); | 
| + | 
| + const int kYearsToMatch = 3; | 
| + std::vector<base::string16> years_to_check; | 
| + for (int year = time_exploded.year; | 
| + year < time_exploded.year + kYearsToMatch; | 
| + ++year) { | 
| + years_to_check.push_back(base::IntToString16(year)); | 
| + } | 
| + if (FindConsecutiveStrings(years_to_check, field->option_values)) | 
| 
Evan Stade
2015/03/18 04:29:25
if X
  return true
return Y
is equivalent to
ret
 
Lei Zhang
2015/03/18 18:08:56
Done.
 | 
| + return true; | 
| + return FindConsecutiveStrings(years_to_check, field->option_contents); | 
| +} | 
| + | 
| CreditCardField::CreditCardField() | 
| : cardholder_(nullptr), | 
| cardholder_last_(nullptr), | 
| @@ -215,8 +313,24 @@ bool CreditCardField::ParseExpirationDate(AutofillScanner* scanner) { | 
| if (expiration_month_ || expiration_date_) | 
| return false; | 
| - // First try to parse split month/year expiration fields. | 
| + // First try to parse split month/year expiration fields by looking for a | 
| + // pair of select fields that look like month/year. | 
| size_t month_year_saved_cursor = scanner->SaveCursor(); | 
| + | 
| + if (LikelyCardMonthSelectField(scanner)) { | 
| + expiration_month_ = scanner->Cursor(); | 
| + scanner->Advance(); | 
| + if (LikelyCardYearSelectField(scanner)) { | 
| + expiration_year_ = scanner->Cursor(); | 
| + scanner->Advance(); | 
| + return true; | 
| + } | 
| + expiration_month_ = nullptr; | 
| + expiration_year_ = nullptr; | 
| + } | 
| + | 
| + // If that fails, do a general regex search. | 
| + scanner->RewindTo(month_year_saved_cursor); | 
| const int kMatchTelAndSelect = MATCH_DEFAULT | MATCH_TELEPHONE | MATCH_SELECT; | 
| if (ParseFieldSpecifics(scanner, | 
| base::UTF8ToUTF16(kExpirationMonthRe), |