// 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. #ifndef CHROME_BROWSER_UI_AUTOFILL_AUTOFILL_DIALOG_CONTROLLER_IMPL_H_ #define CHROME_BROWSER_UI_AUTOFILL_AUTOFILL_DIALOG_CONTROLLER_IMPL_H_ #include <set> #include <vector> #include "base/callback.h" #include "base/memory/scoped_ptr.h" #include "base/memory/weak_ptr.h" #include "base/strings/string16.h" #include "base/time.h" #include "chrome/browser/ui/autofill/account_chooser_model.h" #include "chrome/browser/ui/autofill/autofill_dialog_controller.h" #include "chrome/browser/ui/autofill/autofill_dialog_models.h" #include "chrome/browser/ui/autofill/autofill_dialog_types.h" #include "chrome/browser/ui/autofill/autofill_popup_controller_impl.h" #include "chrome/browser/ui/autofill/country_combobox_model.h" #include "components/autofill/content/browser/wallet/wallet_client.h" #include "components/autofill/content/browser/wallet/wallet_client_delegate.h" #include "components/autofill/content/browser/wallet/wallet_items.h" #include "components/autofill/content/browser/wallet/wallet_signin_helper_delegate.h" #include "components/autofill/core/browser/autofill_manager_delegate.h" #include "components/autofill/core/browser/autofill_metrics.h" #include "components/autofill/core/browser/autofill_popup_delegate.h" #include "components/autofill/core/browser/field_types.h" #include "components/autofill/core/browser/form_structure.h" #include "components/autofill/core/browser/personal_data_manager.h" #include "components/autofill/core/browser/personal_data_manager_observer.h" #include "content/public/browser/notification_observer.h" #include "content/public/browser/notification_registrar.h" #include "content/public/common/ssl_status.h" #include "googleurl/src/gurl.h" #include "ui/base/models/simple_menu_model.h" #include "ui/base/ui_base_types.h" class Profile; namespace content { class WebContents; } namespace user_prefs { class PrefRegistrySyncable; } namespace autofill { class AutofillDataModel; class AutofillDialogView; class AutofillPopupControllerImpl; class DataModelWrapper; class TestableAutofillDialogView; namespace risk { class Fingerprint; } namespace wallet { class WalletSigninHelper; } // This class drives the dialog that appears when a site uses the imperative // autocomplete API to fill out a form. class AutofillDialogControllerImpl : public AutofillDialogController, public AutofillPopupDelegate, public content::NotificationObserver, public SuggestionsMenuModelDelegate, public wallet::WalletClientDelegate, public wallet::WalletSigninHelperDelegate, public PersonalDataManagerObserver, public AccountChooserModelDelegate { public: virtual ~AutofillDialogControllerImpl(); static base::WeakPtr<AutofillDialogControllerImpl> Create( content::WebContents* contents, const FormData& form_structure, const GURL& source_url, const DialogType dialog_type, const base::Callback<void(const FormStructure*, const std::string&)>& callback); static void RegisterUserPrefs(user_prefs::PrefRegistrySyncable* registry); void Show(); void Hide(); // Updates the progress bar based on the Autocheckout progress. |value| should // be in [0.0, 1.0]. void UpdateProgressBar(double value); // Whether Autocheckout is currently running. bool AutocheckoutIsRunning() const; // Called when there is an error in an active Autocheckout flow. void OnAutocheckoutError(); // Called when an Autocheckout flow completes successfully. void OnAutocheckoutSuccess(); // Returns |view_| as a testable version of itself (if |view_| exists and // actually implements |AutofillDialogView::GetTestableView()|). TestableAutofillDialogView* GetTestableView(); // AutofillDialogController implementation. virtual string16 DialogTitle() const OVERRIDE; virtual string16 AccountChooserText() const OVERRIDE; virtual string16 SignInLinkText() const OVERRIDE; virtual string16 EditSuggestionText() const OVERRIDE; virtual string16 CancelButtonText() const OVERRIDE; virtual string16 ConfirmButtonText() const OVERRIDE; virtual string16 SaveLocallyText() const OVERRIDE; virtual string16 ProgressBarText() const OVERRIDE; virtual string16 LegalDocumentsText() OVERRIDE; virtual DialogSignedInState SignedInState() const OVERRIDE; virtual bool ShouldShowSpinner() const OVERRIDE; virtual bool ShouldOfferToSaveInChrome() const OVERRIDE; virtual ui::MenuModel* MenuModelForAccountChooser() OVERRIDE; virtual gfx::Image AccountChooserImage() OVERRIDE; virtual bool ShouldShowDetailArea() const OVERRIDE; virtual bool ShouldShowProgressBar() const OVERRIDE; virtual int GetDialogButtons() const OVERRIDE; virtual bool IsDialogButtonEnabled(ui::DialogButton button) const OVERRIDE; virtual const std::vector<ui::Range>& LegalDocumentLinks() OVERRIDE; virtual bool SectionIsActive(DialogSection section) const OVERRIDE; virtual const DetailInputs& RequestedFieldsForSection(DialogSection section) const OVERRIDE; virtual ui::ComboboxModel* ComboboxModelForAutofillType( AutofillFieldType type) OVERRIDE; virtual ui::MenuModel* MenuModelForSection(DialogSection section) OVERRIDE; #if defined(OS_ANDROID) virtual ui::MenuModel* MenuModelForSectionHack(DialogSection section) OVERRIDE; #endif virtual string16 LabelForSection(DialogSection section) const OVERRIDE; virtual SuggestionState SuggestionStateForSection( DialogSection section) OVERRIDE; virtual void EditClickedForSection(DialogSection section) OVERRIDE; virtual void EditCancelledForSection(DialogSection section) OVERRIDE; virtual gfx::Image IconForField(AutofillFieldType type, const string16& user_input) const OVERRIDE; virtual string16 InputValidityMessage(DialogSection section, AutofillFieldType type, const string16& value) OVERRIDE; virtual ValidityData InputsAreValid( DialogSection section, const DetailOutputMap& inputs, ValidationType validation_type) OVERRIDE; virtual void UserEditedOrActivatedInput(DialogSection section, const DetailInput* input, gfx::NativeView parent_view, const gfx::Rect& content_bounds, const string16& field_contents, bool was_edit) OVERRIDE; virtual bool HandleKeyPressEventInInput( const content::NativeWebKeyboardEvent& event) OVERRIDE; virtual void FocusMoved() OVERRIDE; virtual gfx::Image SplashPageImage() const OVERRIDE; virtual void ViewClosed() OVERRIDE; virtual std::vector<DialogNotification> CurrentNotifications() OVERRIDE; virtual void SignInLinkClicked() OVERRIDE; virtual void NotificationCheckboxStateChanged(DialogNotification::Type type, bool checked) OVERRIDE; virtual void LegalDocumentLinkClicked(const ui::Range& range) OVERRIDE; virtual void OnCancel() OVERRIDE; virtual void OnAccept() OVERRIDE; virtual Profile* profile() OVERRIDE; virtual content::WebContents* web_contents() OVERRIDE; // AutofillPopupDelegate implementation. virtual void OnPopupShown(content::KeyboardListener* listener) OVERRIDE; virtual void OnPopupHidden(content::KeyboardListener* listener) OVERRIDE; virtual void DidSelectSuggestion(int identifier) OVERRIDE; virtual void DidAcceptSuggestion(const string16& value, int identifier) OVERRIDE; virtual void RemoveSuggestion(const string16& value, int identifier) OVERRIDE; virtual void ClearPreviewedForm() OVERRIDE; // content::NotificationObserver implementation. virtual void Observe(int type, const content::NotificationSource& source, const content::NotificationDetails& details) OVERRIDE; // SuggestionsMenuModelDelegate implementation. virtual void SuggestionItemSelected(SuggestionsMenuModel* model, size_t index) OVERRIDE; // wallet::WalletClientDelegate implementation. virtual const AutofillMetrics& GetMetricLogger() const OVERRIDE; virtual DialogType GetDialogType() const OVERRIDE; virtual std::string GetRiskData() const OVERRIDE; virtual void OnDidAcceptLegalDocuments() OVERRIDE; virtual void OnDidAuthenticateInstrument(bool success) OVERRIDE; virtual void OnDidGetFullWallet( scoped_ptr<wallet::FullWallet> full_wallet) OVERRIDE; virtual void OnDidGetWalletItems( scoped_ptr<wallet::WalletItems> wallet_items) OVERRIDE; virtual void OnDidSaveAddress( const std::string& address_id, const std::vector<wallet::RequiredAction>& required_actions, const std::vector<wallet::FormFieldError>& form_field_errors) OVERRIDE; virtual void OnDidSaveInstrument( const std::string& instrument_id, const std::vector<wallet::RequiredAction>& required_actions, const std::vector<wallet::FormFieldError>& form_field_errors) OVERRIDE; virtual void OnDidSaveInstrumentAndAddress( const std::string& instrument_id, const std::string& address_id, const std::vector<wallet::RequiredAction>& required_actions, const std::vector<wallet::FormFieldError>& form_field_errors) OVERRIDE; virtual void OnDidUpdateAddress( const std::string& address_id, const std::vector<wallet::RequiredAction>& required_actions, const std::vector<wallet::FormFieldError>& form_field_errors) OVERRIDE; virtual void OnDidUpdateInstrument( const std::string& instrument_id, const std::vector<wallet::RequiredAction>& required_actions, const std::vector<wallet::FormFieldError>& form_field_errors) OVERRIDE; virtual void OnWalletError( wallet::WalletClient::ErrorType error_type) OVERRIDE; // PersonalDataManagerObserver implementation. virtual void OnPersonalDataChanged() OVERRIDE; // AccountChooserModelDelegate implementation. virtual void AccountChoiceChanged() OVERRIDE; virtual void UpdateAccountChooserView() OVERRIDE; // wallet::WalletSigninHelperDelegate implementation. virtual void OnPassiveSigninSuccess(const std::string& username) OVERRIDE; virtual void OnPassiveSigninFailure( const GoogleServiceAuthError& error) OVERRIDE; virtual void OnAutomaticSigninSuccess(const std::string& username) OVERRIDE; virtual void OnAutomaticSigninFailure( const GoogleServiceAuthError& error) OVERRIDE; virtual void OnUserNameFetchSuccess(const std::string& username) OVERRIDE; virtual void OnUserNameFetchFailure( const GoogleServiceAuthError& error) OVERRIDE; DialogType dialog_type() const { return dialog_type_; } protected: // Exposed for testing. AutofillDialogControllerImpl( content::WebContents* contents, const FormData& form_structure, const GURL& source_url, const DialogType dialog_type, const base::Callback<void(const FormStructure*, const std::string&)>& callback); // Exposed for testing. AutofillDialogView* view() { return view_.get(); } virtual AutofillDialogView* CreateView(); const DetailInput* input_showing_popup() const { return input_showing_popup_; } // Returns the PersonalDataManager for |profile_|. virtual PersonalDataManager* GetManager(); // Returns the WalletClient* this class uses to talk to Online Wallet. Exposed // for testing. virtual wallet::WalletClient* GetWalletClient(); // Call to disable communication to Online Wallet for this dialog. // Exposed for testing. void DisableWallet(wallet::WalletClient::ErrorType error_type); // Returns whether Wallet is the current data source. Exposed for testing. virtual bool IsPayingWithWallet() const; // Asks risk module to asynchronously load fingerprint data. Data will be // returned via |OnDidLoadRiskFingerprintData()|. Exposed for testing. virtual void LoadRiskFingerprintData(); virtual void OnDidLoadRiskFingerprintData( scoped_ptr<risk::Fingerprint> fingerprint); // Opens the given URL in a new foreground tab. virtual void OpenTabWithUrl(const GURL& url); // Whether |section| was sent into edit mode based on existing data. This // happens when a user clicks "Edit" or a suggestion is invalid. virtual bool IsEditingExistingData(DialogSection section) const; // Should be called on the Wallet sign-in error. virtual void OnWalletSigninError(); private: // Whether or not the current request wants credit info back. bool RequestingCreditCardInfo() const; // Whether the information input in this dialog will be securely transmitted // to the requesting site. bool TransmissionWillBeSecure() const; // Initializes |suggested_email_| et al. void SuggestionsUpdated(); // Whether the user's wallet items have at least one address and instrument. bool HasCompleteWallet() const; // Starts fetching the wallet items from Online Wallet. void GetWalletItems(); // Stop showing sign in flow. void HideSignIn(); // Handles the SignedInState() on Wallet or sign-in state update. // Triggers the user name fetch and the passive/automatic sign-in. void SignedInStateUpdated(); // Refreshes the model on Wallet or sign-in state update. void OnWalletOrSigninUpdate(); // Called when a Save or Update call to Wallet has validation errors. void OnWalletFormFieldError( const std::vector<wallet::FormFieldError>& form_field_errors); // Calculates |legal_documents_text_| and |legal_document_link_ranges_| if // they have not already been calculated. void EnsureLegalDocumentsText(); // Clears previously entered manual input, shows editing UI if the current // suggestion is invalid, and updates the |view_| (if it exists). void PrepareDetailInputsForSection(DialogSection section); // Creates a DataModelWrapper item for the item that's checked in the // suggestion model for |section|. This may represent Autofill // data or Wallet data, depending on whether Wallet is currently enabled. scoped_ptr<DataModelWrapper> CreateWrapper(DialogSection section); // Helper to return the current Wallet instrument or address. If the dialog // isn't using Wallet or the user is adding a new instrument or address, NULL // will be returned. const wallet::WalletItems::MaskedInstrument* ActiveInstrument() const; const wallet::Address* ActiveShippingAddress() const; // Fills in |section|-related fields in |output_| according to the state of // |view_|. void FillOutputForSection(DialogSection section); // As above, but uses |compare| to determine whether a DetailInput matches // a field. Saves any new Autofill data to the PersonalDataManager. void FillOutputForSectionWithComparator(DialogSection section, const InputFieldComparator& compare); // Returns whether |form_structure|_| has any fields that match the fieldset // represented by |section|. bool FormStructureCaresAboutSection(DialogSection section) const; // Sets the CVC result on |form_structure_| to the value in |cvc|. void SetCvcResult(const string16& cvc); // Gets the value for |type| in |section|, whether it comes from manual user // input or the active suggestion. string16 GetValueFromSection(DialogSection section, AutofillFieldType type); // Saves the data in |profile| to the personal data manager. This may add // a new profile or tack onto an existing profile. void SaveProfileGleanedFromSection(const AutofillProfile& profile, DialogSection section); // Gets the SuggestionsMenuModel for |section|. SuggestionsMenuModel* SuggestionsMenuModelForSection(DialogSection section); const SuggestionsMenuModel* SuggestionsMenuModelForSection( DialogSection section) const; // And the reverse. DialogSection SectionForSuggestionsMenuModel( const SuggestionsMenuModel& model); // Suggested text and icons for sections. Suggestion text is used to show an // abidged overview of the currently used suggestion. Extra text is used when // part of a section is suggested but part must be manually input (e.g. during // a CVC challenge or when using Autofill's CC section [never stores CVC]). string16 SuggestionTextForSection(DialogSection section); gfx::Font::FontStyle SuggestionTextStyleForSection(DialogSection section) const; string16 RequiredActionTextForSection(DialogSection section) const; gfx::Image SuggestionIconForSection(DialogSection section); string16 ExtraSuggestionTextForSection(DialogSection section) const; gfx::Image ExtraSuggestionIconForSection(DialogSection section) const; // Loads profiles that can suggest data for |type|. |field_contents| is the // part the user has already typed. |inputs| is the rest of section. // Identifying info is loaded into the last three outparams as well as // |popup_guids_|. void GetProfileSuggestions( AutofillFieldType type, const string16& field_contents, const DetailInputs& inputs, std::vector<string16>* popup_values, std::vector<string16>* popup_labels, std::vector<string16>* popup_icons); // Like RequestedFieldsForSection, but returns a pointer. DetailInputs* MutableRequestedFieldsForSection(DialogSection section); // Hides |popup_controller_|'s popup view, if it exists. void HidePopup(); // Set whether the currently editing |section| was originally based on // existing Wallet or Autofill data. void SetEditingExistingData(DialogSection section, bool editing); // Whether the user has chosen to enter all new data in |section|. This // happens via choosing "Add a new X..." from a section's suggestion menu. bool IsManuallyEditingSection(DialogSection section) const; // Whether the user has chosen to enter all new data in at least one section. bool IsManuallyEditingAnySection() const; // Returns validity message for a given credit card number. base::string16 CreditCardNumberValidityMessage( const base::string16& number) const; // Whether a particular DetailInput in |section| should be edited or not. bool InputIsEditable(const DetailInput& input, DialogSection section) const; // Whether all of the input fields currently showing in the dialog have valid // contents. bool AllSectionsAreValid(); // Whether all of the input fields currently showing in the given |section| of // the dialog have valid contents. bool SectionIsValid(DialogSection section); // Whether the currently active credit card expiration date is valid. bool IsCreditCardExpirationValid(const base::string16& year, const base::string16& month) const; // Returns true if |key| refers to a suggestion, as opposed to some control // menu item. bool IsASuggestionItemKey(const std::string& key) const; // Whether the billing section should be used to fill in the shipping details. bool ShouldUseBillingForShipping(); // Whether the user wishes to save information locally to Autofill. bool ShouldSaveDetailsLocally(); // Change whether the controller is currently submitting details to Autofill // or Online Wallet (|is_submitting_|) and update the view. void SetIsSubmitting(bool submitting); // Whether the user has accepted all the current legal documents' terms. bool AreLegalDocumentsCurrent() const; // Accepts any pending legal documents now that the user has pressed Submit. void AcceptLegalDocuments(); // Start the submit proccess to interact with Online Wallet (might do various // things like accept documents, save details, update details, respond to // required actions, etc.). void SubmitWithWallet(); // Creates an instrument based on |views_|' contents. scoped_ptr<wallet::Instrument> CreateTransientInstrument(); // Creates an update request based on |instrument|. May return NULL. scoped_ptr<wallet::WalletClient::UpdateInstrumentRequest> CreateUpdateInstrumentRequest(const wallet::Instrument* instrument, const std::string& instrument_id); // Creates an address based on the contents of |view_|. scoped_ptr<wallet::Address> CreateTransientAddress(); // Gets a full wallet from Online Wallet so the user can purchase something. // This information is decoded to reveal a fronting (proxy) card. void GetFullWallet(); // Calls |GetFullWallet()| if the required members (|risk_data_|, // |active_instrument_id_|, and |active_address_id_|) are populated. void GetFullWalletIfReady(); // Updates the state of the controller and |view_| based on any required // actions returned by Save or Update calls to Wallet. void HandleSaveOrUpdateRequiredActions( const std::vector<wallet::RequiredAction>& required_actions); // Whether submission is currently waiting for |action| to be handled. bool IsSubmitPausedOn(wallet::RequiredAction action) const; // Called when there's nothing left to accept, update, save, or authenticate // in order to fill |form_structure_| and pass data back to the invoking page. void FinishSubmit(); // Writes to prefs the choice of AutofillDataModel for |section|. void PersistAutofillChoice(DialogSection section, const std::string& guid, int variant); // Sets the outparams to the default AutofillDataModel for |section| (which is // the first one in the menu that is a suggestion item). void GetDefaultAutofillChoice(DialogSection section, std::string* guid, int* variant); // Reads from prefs the choice of AutofillDataModel for |section|. Returns // whether there was a setting to read. bool GetAutofillChoice(DialogSection section, std::string* guid, int* variant); // Calculates which AutofillDataModel variant |model| is referring to. size_t GetSelectedVariantForModel(const SuggestionsMenuModel& model); // Logs metrics when the dialog is submitted. void LogOnFinishSubmitMetrics(); // Logs metrics when the dialog is canceled. void LogOnCancelMetrics(); // Logs metrics when the edit ui is shown for the given |section|. void LogEditUiShownMetric(DialogSection section); // Logs metrics when a suggestion item from the given |model| is selected. void LogSuggestionItemSelectedMetric(const SuggestionsMenuModel& model); // Logs the time elapsed from when the dialog was shown to when the user could // interact with it. void LogDialogLatencyToShow(); // Returns the metric corresponding to the user's initial state when // interacting with this dialog. AutofillMetrics::DialogInitialUserStateMetric GetInitialUserState() const; // The |profile| for |contents_|. Profile* const profile_; // The WebContents where the Autofill action originated. content::WebContents* const contents_; // For logging UMA metrics. const AutofillMetrics metric_logger_; base::Time dialog_shown_timestamp_; AutofillMetrics::DialogInitialUserStateMetric initial_user_state_; // The time that Autocheckout started running. Reset on error. While this is // a valid time, |AutocheckoutIsRunning()| will return true. base::Time autocheckout_started_timestamp_; // Whether this is an Autocheckout or a requestAutocomplete dialog. const DialogType dialog_type_; FormStructure form_structure_; // Whether the URL visible to the user when this dialog was requested to be // invoked is the same as |source_url_|. bool invoked_from_same_origin_; // The URL of the invoking site. GURL source_url_; // The SSL info from the invoking site. content::SSLStatus ssl_status_; // The callback via which we return the collected data and, if Online Wallet // was used, the Google transaction id. base::Callback<void(const FormStructure*, const std::string&)> callback_; // The AccountChooserModel acts as the MenuModel for the account chooser, // and also tracks which data source the dialog is using. AccountChooserModel account_chooser_model_; // The sign-in helper to fetch the user info and perform passive sign-in. // The helper is set only during fetch/sign-in, and NULL otherwise. scoped_ptr<wallet::WalletSigninHelper> signin_helper_; // A client to talk to the Online Wallet API. wallet::WalletClient wallet_client_; // Recently received items retrieved via |wallet_client_|. scoped_ptr<wallet::WalletItems> wallet_items_; scoped_ptr<wallet::FullWallet> full_wallet_; // Local machine signals to pass along on each request to trigger (or // discourage) risk challenges; sent if the user is up to date on legal docs. std::string risk_data_; // The text to display when the user is accepting new terms of service, etc. string16 legal_documents_text_; // The ranges within |legal_documents_text_| to linkify. std::vector<ui::Range> legal_document_link_ranges_; // The instrument and address IDs from the Online Wallet server to be used // when getting a full wallet. std::string active_instrument_id_; std::string active_address_id_; // The fields for billing and shipping which the page has actually requested. DetailInputs requested_email_fields_; DetailInputs requested_cc_fields_; DetailInputs requested_billing_fields_; DetailInputs requested_cc_billing_fields_; DetailInputs requested_shipping_fields_; // Models for the credit card expiration inputs. MonthComboboxModel cc_exp_month_combobox_model_; YearComboboxModel cc_exp_year_combobox_model_; // Model for the country input. CountryComboboxModel country_combobox_model_; // Models for the suggestion views. SuggestionsMenuModel suggested_email_; SuggestionsMenuModel suggested_cc_; SuggestionsMenuModel suggested_billing_; SuggestionsMenuModel suggested_cc_billing_; SuggestionsMenuModel suggested_shipping_; // |DialogSection|s that are in edit mode that are based on existing data. std::set<DialogSection> section_editing_state_; // Whether |form_structure_| has asked for any details that would indicate // we should show a shipping section. bool cares_about_shipping_; // The GUIDs for the currently showing unverified profiles popup. std::vector<PersonalDataManager::GUIDPair> popup_guids_; // The controller for the currently showing popup (which helps users when // they're manually filling the dialog). base::WeakPtr<AutofillPopupControllerImpl> popup_controller_; // The input for which |popup_controller_| is currently showing a popup // (if any). const DetailInput* input_showing_popup_; scoped_ptr<AutofillDialogView> view_; // A NotificationRegistrar for tracking the completion of sign-in. content::NotificationRegistrar signin_registrar_; base::WeakPtrFactory<AutofillDialogControllerImpl> weak_ptr_factory_; // Whether the wallet promos should be shown in the notification area. Based // on whether the user has paid with Wallet or has signed into this dialog. bool should_show_wallet_promo_; bool has_shown_wallet_usage_confirmation_; // Whether a user accepted legal documents while this dialog is running. bool has_accepted_legal_documents_; // True after the user first accepts the dialog and presses "Submit". May // continue to be true while processing required actions. bool is_submitting_; // True if the last call to |GetFullWallet()| returned a // CHOOSE_ANOTHER_INSTRUMENT_OR_ADDRESS required action, indicating that the // selected instrument or address had become invalid since it was originally // returned in |GetWalletItems()|. bool choose_another_instrument_or_address_; // Whether or not the server side validation errors returned by Wallet were // recoverable. bool wallet_server_validation_recoverable_; typedef std::map<AutofillFieldType, std::pair<base::string16, base::string16> > TypeErrorInputMap; typedef std::map<DialogSection, TypeErrorInputMap> WalletValidationErrors; // Wallet validation errors. section->type->(error_msg, input_value). WalletValidationErrors wallet_errors_; // The current state of the Autocheckout flow. AutocheckoutState autocheckout_state_; // Whether the latency to display to the UI was logged to UMA yet. bool was_ui_latency_logged_; DISALLOW_COPY_AND_ASSIGN(AutofillDialogControllerImpl); }; } // namespace autofill #endif // CHROME_BROWSER_UI_AUTOFILL_AUTOFILL_DIALOG_CONTROLLER_IMPL_H_