OLD | NEW |
| (Empty) |
1 // Copyright 2013 The Chromium Authors. All rights reserved. | |
2 // Use of this source code is governed by a BSD-style license that can be | |
3 // found in the LICENSE file. | |
4 | |
5 #include "components/autofill/content/browser/autocheckout_manager.h" | |
6 | |
7 #include "base/basictypes.h" | |
8 #include "base/bind.h" | |
9 #include "base/strings/utf_string_conversions.h" | |
10 #include "components/autofill/content/browser/autocheckout_request_manager.h" | |
11 #include "components/autofill/content/browser/autocheckout_statistic.h" | |
12 #include "components/autofill/content/browser/autocheckout_steps.h" | |
13 #include "components/autofill/core/browser/autofill_country.h" | |
14 #include "components/autofill/core/browser/autofill_field.h" | |
15 #include "components/autofill/core/browser/autofill_manager.h" | |
16 #include "components/autofill/core/browser/autofill_metrics.h" | |
17 #include "components/autofill/core/browser/autofill_profile.h" | |
18 #include "components/autofill/core/browser/autofill_type.h" | |
19 #include "components/autofill/core/browser/credit_card.h" | |
20 #include "components/autofill/core/browser/form_structure.h" | |
21 #include "components/autofill/core/common/autofill_messages.h" | |
22 #include "components/autofill/core/common/form_data.h" | |
23 #include "components/autofill/core/common/form_field_data.h" | |
24 #include "components/autofill/core/common/web_element_descriptor.h" | |
25 #include "content/public/browser/browser_context.h" | |
26 #include "content/public/browser/browser_thread.h" | |
27 #include "content/public/browser/render_view_host.h" | |
28 #include "content/public/browser/web_contents.h" | |
29 #include "net/cookies/cookie_options.h" | |
30 #include "net/cookies/cookie_store.h" | |
31 #include "net/url_request/url_request_context.h" | |
32 #include "net/url_request/url_request_context_getter.h" | |
33 #include "ui/gfx/rect.h" | |
34 #include "url/gurl.h" | |
35 | |
36 using content::RenderViewHost; | |
37 using content::WebContents; | |
38 | |
39 namespace autofill { | |
40 | |
41 namespace { | |
42 | |
43 const char kGoogleAccountsUrl[] = "https://accounts.google.com/"; | |
44 | |
45 // Build FormFieldData based on the supplied |autocomplete_attribute|. Will | |
46 // fill rest of properties with default values. | |
47 FormFieldData BuildField(const std::string& autocomplete_attribute) { | |
48 FormFieldData field; | |
49 field.name = base::string16(); | |
50 field.value = base::string16(); | |
51 field.autocomplete_attribute = autocomplete_attribute; | |
52 field.form_control_type = "text"; | |
53 return field; | |
54 } | |
55 | |
56 // Build Autocheckout specific form data to be consumed by | |
57 // AutofillDialogController to show the Autocheckout specific UI. | |
58 FormData BuildAutocheckoutFormData() { | |
59 FormData formdata; | |
60 formdata.fields.push_back(BuildField("email")); | |
61 formdata.fields.push_back(BuildField("cc-name")); | |
62 formdata.fields.push_back(BuildField("cc-number")); | |
63 formdata.fields.push_back(BuildField("cc-exp-month")); | |
64 formdata.fields.push_back(BuildField("cc-exp-year")); | |
65 formdata.fields.push_back(BuildField("cc-csc")); | |
66 formdata.fields.push_back(BuildField("billing address-line1")); | |
67 formdata.fields.push_back(BuildField("billing address-line2")); | |
68 formdata.fields.push_back(BuildField("billing locality")); | |
69 formdata.fields.push_back(BuildField("billing region")); | |
70 formdata.fields.push_back(BuildField("billing country")); | |
71 formdata.fields.push_back(BuildField("billing postal-code")); | |
72 formdata.fields.push_back(BuildField("billing tel")); | |
73 formdata.fields.push_back(BuildField("shipping name")); | |
74 formdata.fields.push_back(BuildField("shipping address-line1")); | |
75 formdata.fields.push_back(BuildField("shipping address-line2")); | |
76 formdata.fields.push_back(BuildField("shipping locality")); | |
77 formdata.fields.push_back(BuildField("shipping region")); | |
78 formdata.fields.push_back(BuildField("shipping country")); | |
79 formdata.fields.push_back(BuildField("shipping postal-code")); | |
80 formdata.fields.push_back(BuildField("shipping tel")); | |
81 return formdata; | |
82 } | |
83 | |
84 AutofillMetrics::AutocheckoutBuyFlowMetric AutocheckoutStatusToUmaMetric( | |
85 AutocheckoutStatus status) { | |
86 switch (status) { | |
87 case SUCCESS: | |
88 return AutofillMetrics::AUTOCHECKOUT_BUY_FLOW_SUCCESS; | |
89 case MISSING_FIELDMAPPING: | |
90 return AutofillMetrics::AUTOCHECKOUT_BUY_FLOW_MISSING_FIELDMAPPING; | |
91 case MISSING_CLICK_ELEMENT_BEFORE_FORM_FILLING: | |
92 return AutofillMetrics:: | |
93 AUTOCHECKOUT_BUY_FLOW_MISSING_CLICK_ELEMENT_BEFORE_FORM_FILLING; | |
94 case MISSING_CLICK_ELEMENT_AFTER_FORM_FILLING: | |
95 return AutofillMetrics:: | |
96 AUTOCHECKOUT_BUY_FLOW_MISSING_CLICK_ELEMENT_AFTER_FORM_FILLING; | |
97 case MISSING_ADVANCE: | |
98 return AutofillMetrics::AUTOCHECKOUT_BUY_FLOW_MISSING_ADVANCE_ELEMENT; | |
99 case CANNOT_PROCEED: | |
100 return AutofillMetrics::AUTOCHECKOUT_BUY_FLOW_CANNOT_PROCEED; | |
101 case AUTOCHECKOUT_STATUS_NUM_STATUS: | |
102 NOTREACHED(); | |
103 } | |
104 | |
105 NOTREACHED(); | |
106 return AutofillMetrics::NUM_AUTOCHECKOUT_BUY_FLOW_METRICS; | |
107 } | |
108 | |
109 // Callback for retrieving Google Account cookies. |callback| is passed the | |
110 // retrieved cookies and posted back to the UI thread. |cookies| is any Google | |
111 // Account cookies. | |
112 void GetGoogleCookiesCallback( | |
113 const base::Callback<void(const std::string&)>& callback, | |
114 const std::string& cookies) { | |
115 content::BrowserThread::PostTask(content::BrowserThread::UI, | |
116 FROM_HERE, | |
117 base::Bind(callback, cookies)); | |
118 } | |
119 | |
120 // Gets Google Account cookies. Must be called on the IO thread. | |
121 // |request_context_getter| is a getter for the current request context. | |
122 // |callback| is called when retrieving cookies is completed. | |
123 void GetGoogleCookies( | |
124 scoped_refptr<net::URLRequestContextGetter> request_context_getter, | |
125 const base::Callback<void(const std::string&)>& callback) { | |
126 net::URLRequestContext* url_request_context = | |
127 request_context_getter->GetURLRequestContext(); | |
128 if (!url_request_context) | |
129 return; | |
130 | |
131 net::CookieStore* cookie_store = url_request_context->cookie_store(); | |
132 | |
133 base::Callback<void(const std::string&)> cookie_callback = base::Bind( | |
134 &GetGoogleCookiesCallback, | |
135 callback); | |
136 | |
137 net::CookieOptions cookie_options; | |
138 cookie_options.set_include_httponly(); | |
139 cookie_store->GetCookiesWithOptionsAsync(GURL(kGoogleAccountsUrl), | |
140 cookie_options, | |
141 cookie_callback); | |
142 } | |
143 | |
144 bool IsBillingGroup(FieldTypeGroup group) { | |
145 return group == ADDRESS_BILLING || | |
146 group == PHONE_BILLING || | |
147 group == NAME_BILLING; | |
148 } | |
149 | |
150 const char kTransactionIdNotSet[] = "transaction id not set"; | |
151 | |
152 } // namespace | |
153 | |
154 AutocheckoutManager::AutocheckoutManager(AutofillManager* autofill_manager) | |
155 : autofill_manager_(autofill_manager), | |
156 metric_logger_(new AutofillMetrics), | |
157 should_show_bubble_(true), | |
158 is_autocheckout_bubble_showing_(false), | |
159 in_autocheckout_flow_(false), | |
160 should_preserve_dialog_(false), | |
161 google_transaction_id_(kTransactionIdNotSet), | |
162 weak_ptr_factory_(this) {} | |
163 | |
164 AutocheckoutManager::~AutocheckoutManager() { | |
165 } | |
166 | |
167 void AutocheckoutManager::FillForms() { | |
168 // |page_meta_data_| should have been set by OnLoadedPageMetaData. | |
169 DCHECK(page_meta_data_); | |
170 | |
171 // Fill the forms on the page with data given by user. | |
172 std::vector<FormData> filled_forms; | |
173 const std::vector<FormStructure*>& form_structures = | |
174 autofill_manager_->GetFormStructures(); | |
175 for (std::vector<FormStructure*>::const_iterator iter = | |
176 form_structures.begin(); iter != form_structures.end(); ++iter) { | |
177 FormStructure* form_structure = *iter; | |
178 form_structure->set_filled_by_autocheckout(true); | |
179 FormData form = form_structure->ToFormData(); | |
180 DCHECK_EQ(form_structure->field_count(), form.fields.size()); | |
181 | |
182 for (size_t i = 0; i < form_structure->field_count(); ++i) { | |
183 const AutofillField* field = form_structure->field(i); | |
184 SetValue(*field, &form.fields[i]); | |
185 } | |
186 | |
187 filled_forms.push_back(form); | |
188 } | |
189 | |
190 // Send filled forms along with proceed descriptor to renderer. | |
191 RenderViewHost* host = | |
192 autofill_manager_->GetWebContents()->GetRenderViewHost(); | |
193 if (!host) | |
194 return; | |
195 | |
196 host->Send(new AutofillMsg_FillFormsAndClick( | |
197 host->GetRoutingID(), | |
198 filled_forms, | |
199 page_meta_data_->click_elements_before_form_fill, | |
200 page_meta_data_->click_elements_after_form_fill, | |
201 page_meta_data_->proceed_element_descriptor)); | |
202 // Record time taken for navigating current page. | |
203 RecordTimeTaken(page_meta_data_->current_page_number); | |
204 } | |
205 | |
206 void AutocheckoutManager::OnAutocheckoutPageCompleted( | |
207 AutocheckoutStatus status) { | |
208 if (!in_autocheckout_flow_) | |
209 return; | |
210 | |
211 DVLOG(2) << "OnAutocheckoutPageCompleted, page_no: " | |
212 << page_meta_data_->current_page_number | |
213 << " status: " | |
214 << status; | |
215 | |
216 DCHECK_NE(MISSING_FIELDMAPPING, status); | |
217 | |
218 SetStepProgressForPage( | |
219 page_meta_data_->current_page_number, | |
220 (status == SUCCESS) ? AUTOCHECKOUT_STEP_COMPLETED : | |
221 AUTOCHECKOUT_STEP_FAILED); | |
222 | |
223 if (page_meta_data_->IsEndOfAutofillableFlow() || status != SUCCESS) | |
224 EndAutocheckout(status); | |
225 } | |
226 | |
227 void AutocheckoutManager::OnLoadedPageMetaData( | |
228 scoped_ptr<AutocheckoutPageMetaData> page_meta_data) { | |
229 scoped_ptr<AutocheckoutPageMetaData> old_meta_data = page_meta_data_.Pass(); | |
230 page_meta_data_ = page_meta_data.Pass(); | |
231 | |
232 // If there is no click element in the last page, then it's the real last page | |
233 // of the flow, and the dialog will be closed when the page navigates. | |
234 // Otherwise, the dialog should be preserved for the page loaded by the click | |
235 // element on the last page of the flow. | |
236 // Note, |should_preserve_dialog_| has to be computed at this point because | |
237 // |in_autocheckout_flow_| may change after |OnLoadedPageMetaData| is called. | |
238 should_preserve_dialog_ = in_autocheckout_flow_ || | |
239 (old_meta_data.get() && | |
240 old_meta_data->IsEndOfAutofillableFlow() && | |
241 old_meta_data->proceed_element_descriptor.retrieval_method != | |
242 WebElementDescriptor::NONE); | |
243 | |
244 // Don't log that the bubble could be displayed if the user entered an | |
245 // Autocheckout flow and sees the first page of the flow again due to an | |
246 // error. | |
247 if (IsStartOfAutofillableFlow() && !in_autocheckout_flow_) { | |
248 metric_logger_->LogAutocheckoutBubbleMetric( | |
249 AutofillMetrics::BUBBLE_COULD_BE_DISPLAYED); | |
250 } | |
251 | |
252 // On the first page of an Autocheckout flow, when this function is called the | |
253 // user won't have opted into the flow yet. | |
254 if (!in_autocheckout_flow_) | |
255 return; | |
256 | |
257 AutocheckoutStatus status = SUCCESS; | |
258 | |
259 // Missing Autofill server results. | |
260 if (!page_meta_data_.get()) { | |
261 status = MISSING_FIELDMAPPING; | |
262 } else if (IsStartOfAutofillableFlow()) { | |
263 // Not possible unless Autocheckout failed to proceed. | |
264 status = CANNOT_PROCEED; | |
265 } else if (!page_meta_data_->IsInAutofillableFlow()) { | |
266 // Missing Autocheckout meta data in the Autofill server results. | |
267 status = MISSING_FIELDMAPPING; | |
268 } else if (page_meta_data_->current_page_number <= | |
269 old_meta_data->current_page_number) { | |
270 // Not possible unless Autocheckout failed to proceed. | |
271 status = CANNOT_PROCEED; | |
272 } | |
273 | |
274 // Encountered an error during the Autocheckout flow, probably to | |
275 // do with a problem on the previous page. | |
276 if (status != SUCCESS) { | |
277 SetStepProgressForPage(old_meta_data->current_page_number, | |
278 AUTOCHECKOUT_STEP_FAILED); | |
279 EndAutocheckout(status); | |
280 return; | |
281 } | |
282 | |
283 SetStepProgressForPage(page_meta_data_->current_page_number, | |
284 AUTOCHECKOUT_STEP_STARTED); | |
285 | |
286 FillForms(); | |
287 } | |
288 | |
289 void AutocheckoutManager::OnFormsSeen() { | |
290 should_show_bubble_ = true; | |
291 } | |
292 | |
293 bool AutocheckoutManager::ShouldIgnoreAjax() { | |
294 return in_autocheckout_flow_ && page_meta_data_->ignore_ajax; | |
295 } | |
296 | |
297 void AutocheckoutManager::MaybeShowAutocheckoutBubble( | |
298 const GURL& frame_url, | |
299 const gfx::RectF& bounding_box) { | |
300 if (!should_show_bubble_ || | |
301 is_autocheckout_bubble_showing_ || | |
302 !IsStartOfAutofillableFlow()) | |
303 return; | |
304 | |
305 base::Callback<void(const std::string&)> callback = base::Bind( | |
306 &AutocheckoutManager::ShowAutocheckoutBubble, | |
307 weak_ptr_factory_.GetWeakPtr(), | |
308 frame_url, | |
309 bounding_box); | |
310 | |
311 content::WebContents* web_contents = autofill_manager_->GetWebContents(); | |
312 if (!web_contents) | |
313 return; | |
314 | |
315 content::BrowserContext* browser_context = web_contents->GetBrowserContext(); | |
316 if(!browser_context) | |
317 return; | |
318 | |
319 scoped_refptr<net::URLRequestContextGetter> request_context = | |
320 scoped_refptr<net::URLRequestContextGetter>( | |
321 browser_context->GetRequestContext()); | |
322 | |
323 if (!request_context.get()) | |
324 return; | |
325 | |
326 base::Closure task = base::Bind(&GetGoogleCookies, request_context, callback); | |
327 | |
328 content::BrowserThread::PostTask(content::BrowserThread::IO, | |
329 FROM_HERE, | |
330 task); | |
331 } | |
332 | |
333 void AutocheckoutManager::ReturnAutocheckoutData( | |
334 const FormStructure* result, | |
335 const std::string& google_transaction_id) { | |
336 if (!result) { | |
337 // When user cancels the dialog, |result| is NULL. | |
338 // TODO(): add AutocheckoutStatus.USER_CANCELLED, and call | |
339 // EndAutocheckout(USER_CANCELLED) instead. | |
340 in_autocheckout_flow_ = false; | |
341 return; | |
342 } | |
343 | |
344 latency_statistics_.clear(); | |
345 last_step_completion_timestamp_ = base::TimeTicks().Now(); | |
346 google_transaction_id_ = google_transaction_id; | |
347 in_autocheckout_flow_ = true; | |
348 should_preserve_dialog_ = true; | |
349 metric_logger_->LogAutocheckoutBuyFlowMetric( | |
350 AutofillMetrics::AUTOCHECKOUT_BUY_FLOW_STARTED); | |
351 | |
352 profile_.reset(new AutofillProfile()); | |
353 credit_card_.reset(new CreditCard()); | |
354 billing_address_.reset(new AutofillProfile()); | |
355 | |
356 for (size_t i = 0; i < result->field_count(); ++i) { | |
357 const AutofillType& type = result->field(i)->Type(); | |
358 const base::string16& value = result->field(i)->value; | |
359 ServerFieldType server_type = type.GetStorableType(); | |
360 if (server_type == CREDIT_CARD_VERIFICATION_CODE) { | |
361 cvv_ = result->field(i)->value; | |
362 continue; | |
363 } | |
364 FieldTypeGroup group = type.group(); | |
365 if (group == CREDIT_CARD) { | |
366 credit_card_->SetRawInfo(server_type, value); | |
367 // TODO(dgwallinga): Find a way of cleanly deprecating CREDIT_CARD_NAME. | |
368 // code.google.com/p/chromium/issues/detail?id=263498 | |
369 if (server_type == CREDIT_CARD_NAME) | |
370 billing_address_->SetRawInfo(NAME_BILLING_FULL, value); | |
371 } else if (server_type == ADDRESS_HOME_COUNTRY) { | |
372 if (IsBillingGroup(group)) | |
373 billing_address_->SetInfo(type, value, autofill_manager_->app_locale()); | |
374 else | |
375 profile_->SetInfo(type, value, autofill_manager_->app_locale()); | |
376 } else if (IsBillingGroup(group)) { | |
377 billing_address_->SetRawInfo(server_type, value); | |
378 } else { | |
379 profile_->SetRawInfo(server_type, value); | |
380 } | |
381 } | |
382 | |
383 // Page types only available in first-page meta data, so save | |
384 // them for use later as we navigate. | |
385 page_types_ = page_meta_data_->page_types; | |
386 SetStepProgressForPage(page_meta_data_->current_page_number, | |
387 AUTOCHECKOUT_STEP_STARTED); | |
388 | |
389 FillForms(); | |
390 } | |
391 | |
392 void AutocheckoutManager::set_metric_logger( | |
393 scoped_ptr<AutofillMetrics> metric_logger) { | |
394 metric_logger_ = metric_logger.Pass(); | |
395 } | |
396 | |
397 void AutocheckoutManager::MaybeShowAutocheckoutDialog( | |
398 const GURL& frame_url, | |
399 AutocheckoutBubbleState state) { | |
400 is_autocheckout_bubble_showing_ = false; | |
401 | |
402 // User has taken action on the bubble, don't offer bubble again. | |
403 if (state != AUTOCHECKOUT_BUBBLE_IGNORED) | |
404 should_show_bubble_ = false; | |
405 | |
406 if (state != AUTOCHECKOUT_BUBBLE_ACCEPTED) | |
407 return; | |
408 | |
409 base::Callback<void(const FormStructure*, const std::string&)> callback = | |
410 base::Bind(&AutocheckoutManager::ReturnAutocheckoutData, | |
411 weak_ptr_factory_.GetWeakPtr()); | |
412 autofill_manager_->ShowRequestAutocompleteDialog(BuildAutocheckoutFormData(), | |
413 frame_url, | |
414 DIALOG_TYPE_AUTOCHECKOUT, | |
415 callback); | |
416 | |
417 for (std::map<int, std::vector<AutocheckoutStepType> >::const_iterator | |
418 it = page_meta_data_->page_types.begin(); | |
419 it != page_meta_data_->page_types.end(); ++it) { | |
420 for (size_t i = 0; i < it->second.size(); ++i) { | |
421 autofill_manager_->delegate()->AddAutocheckoutStep(it->second[i]); | |
422 } | |
423 } | |
424 } | |
425 | |
426 void AutocheckoutManager::ShowAutocheckoutBubble( | |
427 const GURL& frame_url, | |
428 const gfx::RectF& bounding_box, | |
429 const std::string& cookies) { | |
430 DCHECK(thread_checker_.CalledOnValidThread()); | |
431 | |
432 base::Callback<void(AutocheckoutBubbleState)> callback = base::Bind( | |
433 &AutocheckoutManager::MaybeShowAutocheckoutDialog, | |
434 weak_ptr_factory_.GetWeakPtr(), | |
435 frame_url); | |
436 is_autocheckout_bubble_showing_ = | |
437 autofill_manager_->delegate()->ShowAutocheckoutBubble( | |
438 bounding_box, | |
439 cookies.find("LSID") != std::string::npos, | |
440 callback); | |
441 } | |
442 | |
443 bool AutocheckoutManager::IsStartOfAutofillableFlow() const { | |
444 return page_meta_data_ && page_meta_data_->IsStartOfAutofillableFlow(); | |
445 } | |
446 | |
447 bool AutocheckoutManager::IsInAutofillableFlow() const { | |
448 return page_meta_data_ && page_meta_data_->IsInAutofillableFlow(); | |
449 } | |
450 | |
451 void AutocheckoutManager::SetValue(const AutofillField& field, | |
452 FormFieldData* field_to_fill) { | |
453 // No-op if Autofill server doesn't know about the field. | |
454 if (field.server_type() == NO_SERVER_DATA) | |
455 return; | |
456 | |
457 const AutofillType& type = field.Type(); | |
458 | |
459 ServerFieldType server_type = type.GetStorableType(); | |
460 if (server_type == FIELD_WITH_DEFAULT_VALUE) { | |
461 // For a form with radio buttons, like: | |
462 // <form> | |
463 // <input type="radio" name="sex" value="male">Male<br> | |
464 // <input type="radio" name="sex" value="female">Female | |
465 // </form> | |
466 // If the default value specified at the server is "female", then | |
467 // Autofill server responds back with following field mappings | |
468 // (fieldtype: FIELD_WITH_DEFAULT_VALUE, value: "female") | |
469 // (fieldtype: FIELD_WITH_DEFAULT_VALUE, value: "female") | |
470 // Note that, the field mapping is repeated twice to respond to both the | |
471 // input elements with the same name/signature in the form. | |
472 // | |
473 // FIELD_WITH_DEFAULT_VALUE can also be used for selects, the correspondent | |
474 // example of the radio buttons example above is: | |
475 // <SELECT name="sex"> | |
476 // <OPTION value="female">Female</OPTION> | |
477 // <OPTION value="male">Male</OPTION> | |
478 // </SELECT> | |
479 base::string16 default_value = UTF8ToUTF16(field.default_value()); | |
480 if (field.is_checkable) { | |
481 // Mark the field checked if server says the default value of the field | |
482 // to be this field's value. | |
483 field_to_fill->is_checked = (field.value == default_value); | |
484 } else if (field.form_control_type == "select-one") { | |
485 field_to_fill->value = default_value; | |
486 } else { | |
487 // FIELD_WITH_DEFAULT_VALUE should not be used for other type of fields. | |
488 NOTREACHED(); | |
489 } | |
490 return; | |
491 } | |
492 | |
493 // Handle verification code directly. | |
494 if (server_type == CREDIT_CARD_VERIFICATION_CODE) { | |
495 field_to_fill->value = cvv_; | |
496 return; | |
497 } | |
498 | |
499 if (type.group() == CREDIT_CARD) { | |
500 credit_card_->FillFormField( | |
501 field, 0, autofill_manager_->app_locale(), field_to_fill); | |
502 } else if (IsBillingGroup(type.group())) { | |
503 billing_address_->FillFormField( | |
504 field, 0, autofill_manager_->app_locale(), field_to_fill); | |
505 } else { | |
506 profile_->FillFormField( | |
507 field, 0, autofill_manager_->app_locale(), field_to_fill); | |
508 } | |
509 } | |
510 | |
511 void AutocheckoutManager::SendAutocheckoutStatus(AutocheckoutStatus status) { | |
512 // To ensure stale data isn't being sent. | |
513 DCHECK_NE(kTransactionIdNotSet, google_transaction_id_); | |
514 | |
515 AutocheckoutRequestManager::CreateForBrowserContext( | |
516 autofill_manager_->GetWebContents()->GetBrowserContext()); | |
517 AutocheckoutRequestManager* autocheckout_request_manager = | |
518 AutocheckoutRequestManager::FromBrowserContext( | |
519 autofill_manager_->GetWebContents()->GetBrowserContext()); | |
520 // It is assumed that the domain Autocheckout starts on does not change | |
521 // during the flow. If this proves to be incorrect, the |source_url| from | |
522 // AutofillDialogControllerImpl will need to be provided in its callback in | |
523 // addition to the Google transaction id. | |
524 autocheckout_request_manager->SendAutocheckoutStatus( | |
525 status, | |
526 autofill_manager_->GetWebContents()->GetURL(), | |
527 latency_statistics_, | |
528 google_transaction_id_); | |
529 | |
530 // Log the result of this Autocheckout flow to UMA. | |
531 metric_logger_->LogAutocheckoutBuyFlowMetric( | |
532 AutocheckoutStatusToUmaMetric(status)); | |
533 | |
534 google_transaction_id_ = kTransactionIdNotSet; | |
535 } | |
536 | |
537 void AutocheckoutManager::SetStepProgressForPage( | |
538 int page_number, | |
539 AutocheckoutStepStatus status) { | |
540 if (page_types_.count(page_number) == 1) { | |
541 for (size_t i = 0; i < page_types_[page_number].size(); ++i) { | |
542 autofill_manager_->delegate()->UpdateAutocheckoutStep( | |
543 page_types_[page_number][i], status); | |
544 } | |
545 } | |
546 } | |
547 | |
548 void AutocheckoutManager::RecordTimeTaken(int page_number) { | |
549 AutocheckoutStatistic statistic; | |
550 statistic.page_number = page_number; | |
551 if (page_types_.count(page_number) == 1) { | |
552 for (size_t i = 0; i < page_types_[page_number].size(); ++i) { | |
553 statistic.steps.push_back(page_types_[page_number][i]); | |
554 } | |
555 } | |
556 | |
557 statistic.time_taken = | |
558 base::TimeTicks().Now() - last_step_completion_timestamp_; | |
559 latency_statistics_.push_back(statistic); | |
560 | |
561 // Reset timestamp. | |
562 last_step_completion_timestamp_ = base::TimeTicks().Now(); | |
563 } | |
564 | |
565 void AutocheckoutManager::EndAutocheckout(AutocheckoutStatus status) { | |
566 DCHECK(in_autocheckout_flow_); | |
567 | |
568 DVLOG(2) << "EndAutocheckout at step: " | |
569 << page_meta_data_->current_page_number | |
570 << " with status: " | |
571 << status; | |
572 | |
573 SendAutocheckoutStatus(status); | |
574 if (status == SUCCESS) | |
575 autofill_manager_->delegate()->OnAutocheckoutSuccess(); | |
576 else | |
577 autofill_manager_->delegate()->OnAutocheckoutError(); | |
578 in_autocheckout_flow_ = false; | |
579 } | |
580 | |
581 } // namespace autofill | |
OLD | NEW |