• Home
  • Line#
  • Scopes#
  • Navigate#
  • Raw
  • Download
1 // Copyright (c) 2012 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 #ifndef UI_VIEWS_CONTROLS_MENU_SUBMENU_VIEW_H_
6 #define UI_VIEWS_CONTROLS_MENU_SUBMENU_VIEW_H_
7 
8 #include <string>
9 
10 #include "base/compiler_specific.h"
11 #include "ui/views/animation/scroll_animator.h"
12 #include "ui/views/controls/menu/menu_delegate.h"
13 #include "ui/views/view.h"
14 
15 namespace views {
16 
17 class MenuHost;
18 class MenuItemView;
19 class MenuScrollViewContainer;
20 
21 // SubmenuView is the parent of all menu items.
22 //
23 // SubmenuView has the following responsibilities:
24 // . It positions and sizes all child views (any type of View may be added,
25 //   not just MenuItemViews).
26 // . Forwards the appropriate events to the MenuController. This allows the
27 //   MenuController to update the selection as the user moves the mouse around.
28 // . Renders the drop indicator during a drop operation.
29 // . Shows and hides the window (a NativeWidgetWin) when the menu is shown on
30 //   screen.
31 //
32 // SubmenuView is itself contained in a MenuScrollViewContainer.
33 // MenuScrollViewContainer handles showing as much of the SubmenuView as the
34 // screen allows. If the SubmenuView is taller than the screen, scroll buttons
35 // are provided that allow the user to see all the menu items.
36 class VIEWS_EXPORT SubmenuView : public View,
37                                  public ScrollDelegate {
38  public:
39   // The submenu's class name.
40   static const char kViewClassName[];
41 
42   // Creates a SubmenuView for the specified menu item.
43   explicit SubmenuView(MenuItemView* parent);
44   virtual ~SubmenuView();
45 
46   // Returns the number of child views that are MenuItemViews.
47   // MenuItemViews are identified by ID.
48   int GetMenuItemCount();
49 
50   // Returns the MenuItemView at the specified index.
51   MenuItemView* GetMenuItemAt(int index);
52 
53   // Positions and sizes the child views. This tiles the views vertically,
54   // giving each child the available width.
55   virtual void Layout() OVERRIDE;
56   virtual gfx::Size GetPreferredSize() OVERRIDE;
57 
58   // Override from View.
59   virtual void GetAccessibleState(ui::AccessibleViewState* state) OVERRIDE;
60 
61   // Painting.
62   virtual void PaintChildren(gfx::Canvas* canvas) OVERRIDE;
63 
64   // Drag and drop methods. These are forwarded to the MenuController.
65   virtual bool GetDropFormats(
66       int* formats,
67       std::set<OSExchangeData::CustomFormat>* custom_formats) OVERRIDE;
68   virtual bool AreDropTypesRequired() OVERRIDE;
69   virtual bool CanDrop(const OSExchangeData& data) OVERRIDE;
70   virtual void OnDragEntered(const ui::DropTargetEvent& event) OVERRIDE;
71   virtual int OnDragUpdated(const ui::DropTargetEvent& event) OVERRIDE;
72   virtual void OnDragExited() OVERRIDE;
73   virtual int OnPerformDrop(const ui::DropTargetEvent& event) OVERRIDE;
74 
75   // Scrolls on menu item boundaries.
76   virtual bool OnMouseWheel(const ui::MouseWheelEvent& e) OVERRIDE;
77 
78   // Overridden from ui::EventHandler.
79   // Scrolls on menu item boundaries.
80   virtual void OnGestureEvent(ui::GestureEvent* event) OVERRIDE;
81 
82   // Returns true if the menu is showing.
83   bool IsShowing();
84 
85   // Shows the menu at the specified location. Coordinates are in screen
86   // coordinates. max_width gives the max width the view should be.
87   void ShowAt(Widget* parent, const gfx::Rect& bounds, bool do_capture);
88 
89   // Resets the bounds of the submenu to |bounds|.
90   void Reposition(const gfx::Rect& bounds);
91 
92   // Closes the menu, destroying the host.
93   void Close();
94 
95   // Hides the hosting window.
96   //
97   // The hosting window is hidden first, then deleted (Close) when the menu is
98   // done running. This is done to avoid deletion ordering dependencies. In
99   // particular, during drag and drop (and when a modal dialog is shown as
100   // a result of choosing a context menu) it is possible that an event is
101   // being processed by the host, so that host is on the stack when we need to
102   // close the window. If we closed the window immediately (and deleted it),
103   // when control returned back to host we would crash as host was deleted.
104   void Hide();
105 
106   // If mouse capture was grabbed, it is released. Does nothing if mouse was
107   // not captured.
108   void ReleaseCapture();
109 
110   // Overriden from View to prevent tab from doing anything.
111   virtual bool SkipDefaultKeyEventProcessing(const ui::KeyEvent& e) OVERRIDE;
112 
113   // Returns the parent menu item we're showing children for.
114   MenuItemView* GetMenuItem() const;
115 
116   // Set the drop item and position.
117   void SetDropMenuItem(MenuItemView* item,
118                        MenuDelegate::DropPosition position);
119 
120   // Returns whether the selection should be shown for the specified item.
121   // The selection is NOT shown during drag and drop when the drop is over
122   // the menu.
123   bool GetShowSelection(MenuItemView* item);
124 
125   // Returns the container for the SubmenuView.
126   MenuScrollViewContainer* GetScrollViewContainer();
127 
128   // Invoked if the menu is prematurely destroyed. This can happen if the window
129   // closes while the menu is shown. If invoked the SubmenuView must drop all
130   // references to the MenuHost as the MenuHost is about to be deleted.
131   void MenuHostDestroyed();
132 
133   // Max width of minor text (accelerator or subtitle) in child menu items. This
134   // doesn't include children's children, only direct children.
max_minor_text_width()135   int max_minor_text_width() const { return max_minor_text_width_; }
136 
137   // Minimum width of menu in pixels (default 0).  This becomes the smallest
138   // width returned by GetPreferredSize().
set_minimum_preferred_width(int minimum_preferred_width)139   void set_minimum_preferred_width(int minimum_preferred_width) {
140     minimum_preferred_width_ = minimum_preferred_width;
141   }
142 
143   // Automatically resize menu if a subview's preferred size changes.
resize_open_menu()144   bool resize_open_menu() const { return resize_open_menu_; }
set_resize_open_menu(bool resize_open_menu)145   void set_resize_open_menu(bool resize_open_menu) {
146     resize_open_menu_ = resize_open_menu;
147   }
148 
149  protected:
150   // Overridden from View:
151   virtual const char* GetClassName() const OVERRIDE;
152 
153   // View method. Overridden to schedule a paint. We do this so that when
154   // scrolling occurs, everything is repainted correctly.
155   virtual void OnBoundsChanged(const gfx::Rect& previous_bounds) OVERRIDE;
156 
157   virtual void ChildPreferredSizeChanged(View* child) OVERRIDE;
158 
159  private:
160   // Paints the drop indicator. This is only invoked if item is non-NULL and
161   // position is not DROP_NONE.
162   void PaintDropIndicator(gfx::Canvas* canvas,
163                           MenuItemView* item,
164                           MenuDelegate::DropPosition position);
165 
166   void SchedulePaintForDropIndicator(MenuItemView* item,
167                                      MenuDelegate::DropPosition position);
168 
169   // Calculates the location of th edrop indicator.
170   gfx::Rect CalculateDropIndicatorBounds(MenuItemView* item,
171                                          MenuDelegate::DropPosition position);
172 
173   // Implementation of ScrollDelegate
174   virtual bool OnScroll(float dx, float dy) OVERRIDE;
175 
176   // Parent menu item.
177   MenuItemView* parent_menu_item_;
178 
179   // Widget subclass used to show the children. This is deleted when we invoke
180   // |DestroyMenuHost|, or |MenuHostDestroyed| is invoked back on us.
181   MenuHost* host_;
182 
183   // If non-null, indicates a drop is in progress and drop_item is the item
184   // the drop is over.
185   MenuItemView* drop_item_;
186 
187   // Position of the drop.
188   MenuDelegate::DropPosition drop_position_;
189 
190   // Ancestor of the SubmenuView, lazily created.
191   MenuScrollViewContainer* scroll_view_container_;
192 
193   // See description above getter.
194   int max_minor_text_width_;
195 
196   // Minimum width returned in GetPreferredSize().
197   int minimum_preferred_width_;
198 
199   // Reposition open menu when contained views change size.
200   bool resize_open_menu_;
201 
202   // The submenu's scroll animator
203   scoped_ptr<ScrollAnimator> scroll_animator_;
204 
205   DISALLOW_COPY_AND_ASSIGN(SubmenuView);
206 };
207 
208 }  // namespace views
209 
210 #endif  // UI_VIEWS_CONTROLS_MENU_SUBMENU_VIEW_H_
211