1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
|
// Copyright (c) 2010 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 VIEWS_CONTROLS_MENU_MENU_ITEM_VIEW_H_
#define VIEWS_CONTROLS_MENU_MENU_ITEM_VIEW_H_
#include "third_party/skia/include/core/SkBitmap.h"
#include "views/view.h"
namespace views {
class MenuButton;
class MenuController;
class MenuDelegate;
class SubmenuView;
// MenuItemView --------------------------------------------------------------
// MenuItemView represents a single menu item with a label and optional icon.
// Each MenuItemView may also contain a submenu, which in turn may contain
// any number of child MenuItemViews.
//
// To use a menu create an initial MenuItemView using the constructor that
// takes a MenuDelegate, then create any number of child menu items by way
// of the various AddXXX methods.
//
// MenuItemView is itself a View, which means you can add Views to each
// MenuItemView. This is normally NOT want you want, rather add other child
// Views to the submenu of the MenuItemView. Any child views of the MenuItemView
// that are focusable can be navigated to by way of the up/down arrow and can be
// activated by way of space/return keys. Activating a focusable child results
// in |AcceleratorPressed| being invoked. Note, that as menus try not to steal
// focus from the hosting window child views do not actually get focus. Instead
// |SetHotTracked| is used as the user navigates around.
//
// There are two ways to show a MenuItemView:
// 1. Use RunMenuAt. This blocks the caller, executing the selected command
// on success.
// 2. Use RunMenuForDropAt. This is intended for use during a drop session
// and does NOT block the caller. Instead the delegate is notified when the
// menu closes via the DropMenuClosed method.
class MenuItemView : public View {
public:
friend class MenuController;
// ID used to identify menu items.
static const int kMenuItemViewID;
// ID used to identify empty menu items.
static const int kEmptyMenuItemViewID;
// If true SetNestableTasksAllowed(true) is invoked before MessageLoop::Run
// is invoked. This is only useful for testing and defaults to false.
static bool allow_task_nesting_during_run_;
// Different types of menu items.
enum Type {
NORMAL,
SUBMENU,
CHECKBOX,
RADIO,
SEPARATOR
};
// Where the menu should be anchored to.
enum AnchorPosition {
TOPLEFT,
TOPRIGHT
};
// Constructor for use with the top level menu item. This menu is never
// shown to the user, rather its use as the parent for all menu items.
explicit MenuItemView(MenuDelegate* delegate);
virtual ~MenuItemView();
// Overridden from View:
virtual bool GetTooltipText(const gfx::Point& p, std::wstring* tooltip);
virtual bool GetAccessibleRole(AccessibilityTypes::Role* role);
// Returns the preferred height of menu items. This is only valid when the
// menu is about to be shown.
static int pref_menu_height() { return pref_menu_height_; }
// X-coordinate of where the label starts.
static int label_start() { return label_start_; }
// Run methods. See description above class for details. Both Run methods take
// a rectangle, which is used to position the menu. |has_mnemonics| indicates
// whether the items have mnemonics. Mnemonics are identified by way of the
// character following the '&'.
void RunMenuAt(gfx::NativeWindow parent,
MenuButton* button,
const gfx::Rect& bounds,
AnchorPosition anchor,
bool has_mnemonics);
void RunMenuForDropAt(gfx::NativeWindow parent,
const gfx::Rect& bounds,
AnchorPosition anchor);
// Hides and cancels the menu. This does nothing if the menu is not open.
void Cancel();
// Adds an item to this menu.
// item_id The id of the item, used to identify it in delegate callbacks
// or (if delegate is NULL) to identify the command associated
// with this item with the controller specified in the ctor. Note
// that this value should not be 0 as this has a special meaning
// ("NULL command, no item selected")
// label The text label shown.
// type The type of item.
void AppendMenuItem(int item_id,
const std::wstring& label,
Type type) {
AppendMenuItemImpl(item_id, label, SkBitmap(), type);
}
// Append a submenu to this menu.
// The returned pointer is owned by this menu.
MenuItemView* AppendSubMenu(int item_id,
const std::wstring& label) {
return AppendMenuItemImpl(item_id, label, SkBitmap(), SUBMENU);
}
// Append a submenu with an icon to this menu.
// The returned pointer is owned by this menu.
MenuItemView* AppendSubMenuWithIcon(int item_id,
const std::wstring& label,
const SkBitmap& icon) {
return AppendMenuItemImpl(item_id, label, icon, SUBMENU);
}
// This is a convenience for standard text label menu items where the label
// is provided with this call.
void AppendMenuItemWithLabel(int item_id,
const std::wstring& label) {
AppendMenuItem(item_id, label, NORMAL);
}
// This is a convenience for text label menu items where the label is
// provided by the delegate.
void AppendDelegateMenuItem(int item_id) {
AppendMenuItem(item_id, std::wstring(), NORMAL);
}
// Adds a separator to this menu
void AppendSeparator() {
AppendMenuItemImpl(0, std::wstring(), SkBitmap(), SEPARATOR);
}
// Appends a menu item with an icon. This is for the menu item which
// needs an icon. Calling this function forces the Menu class to draw
// the menu, instead of relying on Windows.
void AppendMenuItemWithIcon(int item_id,
const std::wstring& label,
const SkBitmap& icon) {
AppendMenuItemImpl(item_id, label, icon, NORMAL);
}
// All the AppendXXX methods funnel into this.
MenuItemView* AppendMenuItemImpl(int item_id,
const std::wstring& label,
const SkBitmap& icon,
Type type);
// Returns the view that contains child menu items. If the submenu has
// not been creates, this creates it.
virtual SubmenuView* CreateSubmenu();
// Returns true if this menu item has a submenu.
virtual bool HasSubmenu() const { return (submenu_ != NULL); }
// Returns the view containing child menu items.
virtual SubmenuView* GetSubmenu() const { return submenu_; }
// Returns the parent menu item.
MenuItemView* GetParentMenuItem() const { return parent_menu_item_; }
// Sets the title
void SetTitle(const std::wstring& title) {
title_ = title;
}
// Returns the title.
const std::wstring& GetTitle() const { return title_; }
// Sets whether this item is selected. This is invoked as the user moves
// the mouse around the menu while open.
void SetSelected(bool selected);
// Returns true if the item is selected.
bool IsSelected() const { return selected_; }
// Sets the |tooltip| for a menu item view with |item_id| identifier.
void SetTooltip(const std::wstring& tooltip, int item_id);
// Sets the icon for the descendant identified by item_id.
void SetIcon(const SkBitmap& icon, int item_id);
// Sets the icon of this menu item.
void SetIcon(const SkBitmap& icon);
// Returns the icon.
const SkBitmap& GetIcon() const { return icon_; }
// Sets the command id of this menu item.
void SetCommand(int command) { command_ = command; }
// Returns the command id of this item.
int GetCommand() const { return command_; }
// Paints the menu item.
virtual void Paint(gfx::Canvas* canvas);
// Returns the preferred size of this item.
virtual gfx::Size GetPreferredSize();
// Returns the object responsible for controlling showing the menu.
MenuController* GetMenuController();
// Returns the delegate. This returns the delegate of the root menu item.
MenuDelegate* GetDelegate();
// Returns the root parent, or this if this has no parent.
MenuItemView* GetRootMenuItem();
// Returns the mnemonic for this MenuItemView, or 0 if this MenuItemView
// doesn't have a mnemonic.
wchar_t GetMnemonic();
// Do we have icons? This only has effect on the top menu. Turning this on
// makes the menus slightly wider and taller.
void set_has_icons(bool has_icons) {
has_icons_ = has_icons;
}
// Returns the descendant with the specified command.
MenuItemView* GetMenuItemByID(int id);
// Invoke if you remove/add children to the menu while it's showing. This
// recalculates the bounds.
void ChildrenChanged();
// Sizes any child views.
virtual void Layout();
// Returns the amount of space needed to accomodate the accelerator. The
// space needed for the accelerator is NOT included in the preferred width.
int GetAcceleratorTextWidth();
protected:
// Creates a MenuItemView. This is used by the various AddXXX methods.
MenuItemView(MenuItemView* parent, int command, Type type);
private:
// Calculates all sizes that we can from the OS.
//
// This is invoked prior to Running a menu.
static void UpdateMenuPartSizes(bool has_icons);
// Called by the two constructors to initialize this menu item.
void Init(MenuItemView* parent,
int command,
MenuItemView::Type type,
MenuDelegate* delegate);
// Invoked by the MenuController when the menu closes as the result of
// drag and drop run.
void DropMenuClosed(bool notify_delegate);
// The RunXXX methods call into this to set up the necessary state before
// running.
void PrepareForRun(bool has_mnemonics);
// Returns the flags passed to DrawStringInt.
int GetDrawStringFlags();
// If this menu item has no children a child is added showing it has no
// children. Otherwise AddEmtpyMenus is recursively invoked on child menu
// items that have children.
void AddEmptyMenus();
// Undoes the work of AddEmptyMenus.
void RemoveEmptyMenus();
// Given bounds within our View, this helper routine mirrors the bounds if
// necessary.
void AdjustBoundsForRTLUI(gfx::Rect* rect) const;
// Actual paint implementation. If for_drag is true, portions of the menu
// are not rendered.
void Paint(gfx::Canvas* canvas, bool for_drag);
// Paints the accelerator.
void PaintAccelerator(gfx::Canvas* canvas);
// Destroys the window used to display this menu and recursively destroys
// the windows used to display all descendants.
void DestroyAllMenuHosts();
// Returns the accelerator text.
std::wstring GetAcceleratorText();
// Returns the various margins.
int GetTopMargin();
int GetBottomMargin();
// Returns the preferred width (and padding) of any children.
int GetChildPreferredWidth();
// The delegate. This is only valid for the root menu item. You shouldn't
// use this directly, instead use GetDelegate() which walks the tree as
// as necessary.
MenuDelegate* delegate_;
// Returns the controller for the run operation, or NULL if the menu isn't
// showing.
MenuController* controller_;
// Used to detect when Cancel was invoked.
bool canceled_;
// Our parent.
MenuItemView* parent_menu_item_;
// Type of menu. NOTE: MenuItemView doesn't itself represent SEPARATOR,
// that is handled by an entirely different view class.
Type type_;
// Whether we're selected.
bool selected_;
// Command id.
int command_;
// Submenu, created via CreateSubmenu.
SubmenuView* submenu_;
// Title.
std::wstring title_;
// Icon.
SkBitmap icon_;
// Does the title have a mnemonic?
bool has_mnemonics_;
bool has_icons_;
// The tooltip to show on hover for this menu item.
std::wstring tooltip_;
// X-coordinate of where the label starts.
static int label_start_;
// Margins between the right of the item and the label.
static int item_right_margin_;
// Preferred height of menu items. Reset every time a menu is run.
static int pref_menu_height_;
DISALLOW_COPY_AND_ASSIGN(MenuItemView);
};
} // namespace views
#endif // VIEWS_CONTROLS_MENU_MENU_ITEM_VIEW_H_
|