// 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_POPUP_CONTROLLER_IMPL_H_ #define CHROME_BROWSER_UI_AUTOFILL_AUTOFILL_POPUP_CONTROLLER_IMPL_H_ #include "base/gtest_prod_util.h" #include "base/i18n/rtl.h" #include "base/memory/weak_ptr.h" #include "base/strings/string16.h" #include "chrome/browser/ui/autofill/autofill_popup_controller.h" #include "content/public/browser/render_widget_host.h" #include "ui/gfx/font.h" #include "ui/gfx/rect.h" #include "ui/gfx/rect_f.h" namespace content { struct NativeWebKeyboardEvent; class RenderViewHost; class WebContents; } namespace gfx { class Display; } namespace ui { class KeyEvent; } namespace autofill { class AutofillPopupDelegate; class AutofillPopupView; // This class is a controller for an AutofillPopupView. It implements // AutofillPopupController to allow calls from AutofillPopupView. The // other, public functions are available to its instantiator. class AutofillPopupControllerImpl : public AutofillPopupController { public: // Creates a new |AutofillPopupControllerImpl|, or reuses |previous| if the // construction arguments are the same. |previous| may be invalidated by this // call. The controller will listen for keyboard input routed to // |web_contents| while the popup is showing, unless |web_contents| is NULL. static base::WeakPtr GetOrCreate( base::WeakPtr previous, base::WeakPtr delegate, content::WebContents* web_contents, gfx::NativeView container_view, const gfx::RectF& element_bounds, base::i18n::TextDirection text_direction); // Shows the popup, or updates the existing popup with the given values. void Show(const std::vector& names, const std::vector& subtexts, const std::vector& icons, const std::vector& identifiers); // Updates the data list values currently shown with the popup. void UpdateDataListValues(const std::vector& values, const std::vector& labels); // Hides the popup and destroys the controller. This also invalidates // |delegate_|. virtual void Hide() OVERRIDE; // Invoked when the view was destroyed by by someone other than this class. virtual void ViewDestroyed() OVERRIDE; bool HandleKeyPressEvent(const content::NativeWebKeyboardEvent& event); // Tells the view to capture mouse events. Must be called before |Show()|. void set_hide_on_outside_click(bool hide_on_outside_click); protected: FRIEND_TEST_ALL_PREFIXES(AutofillExternalDelegateBrowserTest, CloseWidgetAndNoLeaking); FRIEND_TEST_ALL_PREFIXES(AutofillPopupControllerUnitTest, ProperlyResetController); AutofillPopupControllerImpl(base::WeakPtr delegate, content::WebContents* web_contents, gfx::NativeView container_view, const gfx::RectF& element_bounds, base::i18n::TextDirection text_direction); virtual ~AutofillPopupControllerImpl(); // AutofillPopupController implementation. virtual void UpdateBoundsAndRedrawPopup() OVERRIDE; virtual void LineSelectedAtPoint(int x, int y) OVERRIDE; virtual void LineAcceptedAtPoint(int x, int y) OVERRIDE; virtual void SelectionCleared() OVERRIDE; virtual bool ShouldRepostEvent(const ui::MouseEvent& event) OVERRIDE; virtual void AcceptSuggestion(size_t index) OVERRIDE; virtual int GetIconResourceID(const string16& resource_name) const OVERRIDE; virtual bool CanDelete(size_t index) const OVERRIDE; virtual bool IsWarning(size_t index) const OVERRIDE; virtual gfx::Rect GetRowBounds(size_t index) OVERRIDE; virtual void SetPopupBounds(const gfx::Rect& bounds) OVERRIDE; virtual const gfx::Rect& popup_bounds() const OVERRIDE; virtual gfx::NativeView container_view() const OVERRIDE; virtual const gfx::RectF& element_bounds() const OVERRIDE; virtual bool IsRTL() const OVERRIDE; virtual const std::vector& names() const OVERRIDE; virtual const std::vector& subtexts() const OVERRIDE; virtual const std::vector& icons() const OVERRIDE; virtual const std::vector& identifiers() const OVERRIDE; #if !defined(OS_ANDROID) virtual const gfx::Font& GetNameFontForRow(size_t index) const OVERRIDE; virtual const gfx::Font& subtext_font() const OVERRIDE; #endif virtual int selected_line() const OVERRIDE; virtual bool hide_on_outside_click() const OVERRIDE; // Change which line is currently selected by the user. void SetSelectedLine(int selected_line); // Increase the selected line by 1, properly handling wrapping. void SelectNextLine(); // Decrease the selected line by 1, properly handling wrapping. void SelectPreviousLine(); // The user has choosen the selected line. bool AcceptSelectedLine(); // The user has removed a suggestion. bool RemoveSelectedLine(); // Convert a y-coordinate to the closest line. int LineFromY(int y); // Returns the height of a row depending on its type. int GetRowHeightFromId(int identifier) const; // Returns true if the given id refers to an element that can be accepted. bool CanAccept(int id); // Returns true if the popup still has non-options entries to show the user. bool HasSuggestions(); // Set the Autofill entry values. Exposed to allow tests to set these values // without showing the popup. void SetValues(const std::vector& names, const std::vector& subtexts, const std::vector& icons, const std::vector& identifier); AutofillPopupView* view() { return view_; } // |view_| pass throughs (virtual for testing). virtual void ShowView(); virtual void InvalidateRow(size_t row); // Protected so tests can access. #if !defined(OS_ANDROID) // Calculates the desired width of the popup based on its contents. int GetDesiredPopupWidth() const; // Calculates the desired height of the popup based on its contents. int GetDesiredPopupHeight() const; // Calculate the width of the row, excluding all the text. This provides // the size of the row that won't be reducible (since all the text can be // elided if there isn't enough space). int RowWidthWithoutText(int row) const; #endif base::WeakPtr GetWeakPtr(); private: // Clear the internal state of the controller. This is needed to ensure that // when the popup is reused it doesn't leak values between uses. void ClearState(); const gfx::Rect RoundedElementBounds() const; #if !defined(OS_ANDROID) // Calculates and sets the bounds of the popup, including placing it properly // to prevent it from going off the screen. void UpdatePopupBounds(); #endif // A helper function to get the display closest to the given point (virtual // for testing). virtual gfx::Display GetDisplayNearestPoint(const gfx::Point& point) const; // Calculates the width of the popup and the x position of it. These values // will stay on the screen. std::pair CalculatePopupXAndWidth( const gfx::Display& left_display, const gfx::Display& right_display, int popup_required_width) const; // Calculates the height of the popup and the y position of it. These values // will stay on the screen. std::pair CalculatePopupYAndHeight( const gfx::Display& top_display, const gfx::Display& bottom_display, int popup_required_height) const; AutofillPopupView* view_; // Weak reference. base::WeakPtr delegate_; // The WebContents in which this object should listen for keyboard events // while showing the popup. Can be NULL, in which case this object will not // listen for keyboard events. content::WebContents* web_contents_; gfx::NativeView container_view_; // Weak reference. // The bounds of the text element that is the focus of the Autofill. // These coordinates are in screen space. const gfx::RectF element_bounds_; // The bounds of the Autofill popup. gfx::Rect popup_bounds_; // The text direction of the popup. base::i18n::TextDirection text_direction_; // The RenderViewHost that this object has registered its keyboard press // callback with. content::RenderViewHost* registered_key_press_event_callback_with_; // The current Autofill query values. std::vector names_; std::vector subtexts_; std::vector icons_; std::vector identifiers_; // Since names_ can be elided to ensure that it fits on the screen, we need to // keep an unelided copy of the names to be able to pass to the delegate. std::vector full_names_; #if !defined(OS_ANDROID) // The fonts for the popup text. gfx::Font name_font_; gfx::Font subtext_font_; gfx::Font warning_font_; #endif // The line that is currently selected by the user. // |kNoSelection| indicates that no line is currently selected. int selected_line_; // Whether the popup view should hide on mouse presses outside of it. bool hide_on_outside_click_; content::RenderWidgetHost::KeyPressEventCallback key_press_event_callback_; base::WeakPtrFactory weak_ptr_factory_; }; } // namespace autofill #endif // CHROME_BROWSER_UI_AUTOFILL_AUTOFILL_POPUP_CONTROLLER_IMPL_H_