diff options
Diffstat (limited to 'views/controls')
-rw-r--r-- | views/controls/menu/menu_controller.cc | 157 | ||||
-rw-r--r-- | views/controls/menu/menu_controller.h | 14 |
2 files changed, 123 insertions, 48 deletions
diff --git a/views/controls/menu/menu_controller.cc b/views/controls/menu/menu_controller.cc index 276226c..c3b111d 100644 --- a/views/controls/menu/menu_controller.cc +++ b/views/controls/menu/menu_controller.cc @@ -49,6 +49,26 @@ static const int kSubmenuHorizontalInset = 3; namespace views { +namespace { + +// Returns true if the mnemonic of |menu| matches key. +bool MatchesMnemonic(MenuItemView* menu, wchar_t key) { + return menu->GetMnemonic() == key; +} + +// Returns true if |menu| doesn't have a mnemonic and first character of the its +// title is |key|. +bool TitleMatchesMnemonic(MenuItemView* menu, wchar_t key) { + if (menu->GetMnemonic()) + return false; + + std::wstring lower_title = UTF16ToWide( + l10n_util::ToLower(WideToUTF16(menu->GetTitle()))); + return !lower_title.empty() && lower_title[0] == key; +} + +} // namespace + // Convenience for scrolling the view such that the origin is visible. static void ScrollToVisible(View* view) { view->ScrollRectToVisible(gfx::Rect(gfx::Point(), view->size())); @@ -200,6 +220,31 @@ class MenuController::MenuScrollTask { DISALLOW_COPY_AND_ASSIGN(MenuScrollTask); }; +// MenuController:SelectByCharDetails ---------------------------------------- + +struct MenuController::SelectByCharDetails { + SelectByCharDetails() + : first_match(-1), + has_multiple(false), + index_of_item(-1), + next_match(-1) { + } + + // Index of the first menu with the specified mnemonic. + int first_match; + + // If true there are multiple menu items with the same mnemonic. + bool has_multiple; + + // Index of the selected item; may remain -1. + int index_of_item; + + // If there are multiple matches this is the index of the item after the + // currently selected item whose mnemonic matches. This may remain -1 even + // though there are matches. + int next_match; +}; + // MenuController ------------------------------------------------------------ // static @@ -1527,6 +1572,57 @@ void MenuController::CloseSubmenu() { } } +MenuController::SelectByCharDetails MenuController::FindChildForMnemonic( + MenuItemView* parent, + wchar_t key, + bool (*match_function)(MenuItemView* menu, wchar_t mnemonic)) { + SubmenuView* submenu = parent->GetSubmenu(); + DCHECK(submenu); + SelectByCharDetails details; + + for (int i = 0, menu_item_count = submenu->GetMenuItemCount(); + i < menu_item_count; ++i) { + MenuItemView* child = submenu->GetMenuItemAt(i); + if (child->IsEnabled() && child->IsVisible()) { + if (child == pending_state_.item) + details.index_of_item = i; + if (match_function(child, key)) { + if (details.first_match == -1) + details.first_match = i; + else + details.has_multiple = true; + if (details.next_match == -1 && details.index_of_item != -1 && + i > details.index_of_item) + details.next_match = i; + } + } + } + return details; +} + +bool MenuController::AcceptOrSelect(MenuItemView* parent, + const SelectByCharDetails& details) { + // This should only be invoked if there is a match. + DCHECK(details.first_match != -1); + DCHECK(parent->HasSubmenu()); + SubmenuView* submenu = parent->GetSubmenu(); + DCHECK(submenu); + if (!details.has_multiple) { + // There's only one match, activate it (or open if it has a submenu). + if (submenu->GetMenuItemAt(details.first_match)->HasSubmenu()) { + SetSelection(submenu->GetMenuItemAt(details.first_match), true, false); + } else { + Accept(submenu->GetMenuItemAt(details.first_match), 0); + return true; + } + } else if (details.index_of_item == -1 || details.next_match == -1) { + SetSelection(submenu->GetMenuItemAt(details.first_match), false, false); + } else { + SetSelection(submenu->GetMenuItemAt(details.next_match), false, false); + } + return false; +} + bool MenuController::SelectByChar(wchar_t character) { wchar_t char_array[1] = { character }; wchar_t key = UTF16ToWide(l10n_util::ToLower(WideToUTF16(char_array)))[0]; @@ -1535,61 +1631,26 @@ bool MenuController::SelectByChar(wchar_t character) { item = item->GetParentMenuItem(); DCHECK(item); DCHECK(item->HasSubmenu()); - SubmenuView* submenu = item->GetSubmenu(); - DCHECK(submenu); - int menu_item_count = submenu->GetMenuItemCount(); - if (!menu_item_count) + DCHECK(item->GetSubmenu()); + if (item->GetSubmenu()->GetMenuItemCount() == 0) return false; - for (int i = 0; i < menu_item_count; ++i) { - MenuItemView* child = submenu->GetMenuItemAt(i); - if (child->GetMnemonic() == key && child->IsEnabled()) { - Accept(child, 0); - return true; - } - } + + // Look for matches based on mnemonic first. + SelectByCharDetails details = + FindChildForMnemonic(item, key, &MatchesMnemonic); + if (details.first_match != -1) + return AcceptOrSelect(item, details); if (item->GetRootMenuItem()->has_mnemonics()) { // Don't guess at mnemonics if the menu explicitly has them. return false; } - // No matching mnemonic, search through items that don't have mnemonic - // based on first character of the title. - int first_match = -1; - bool has_multiple = false; - int next_match = -1; - int index_of_item = -1; - for (int i = 0; i < menu_item_count; ++i) { - MenuItemView* child = submenu->GetMenuItemAt(i); - if (!child->GetMnemonic() && child->IsEnabled()) { - std::wstring lower_title = UTF16ToWide( - l10n_util::ToLower(WideToUTF16(child->GetTitle()))); - if (child == pending_state_.item) - index_of_item = i; - if (lower_title.length() && lower_title[0] == key) { - if (first_match == -1) - first_match = i; - else - has_multiple = true; - if (next_match == -1 && index_of_item != -1 && i > index_of_item) - next_match = i; - } - } - } - if (first_match != -1) { - if (!has_multiple) { - if (submenu->GetMenuItemAt(first_match)->HasSubmenu()) { - SetSelection(submenu->GetMenuItemAt(first_match), true, false); - } else { - Accept(submenu->GetMenuItemAt(first_match), 0); - return true; - } - } else if (index_of_item == -1 || next_match == -1) { - SetSelection(submenu->GetMenuItemAt(first_match), false, false); - } else { - SetSelection(submenu->GetMenuItemAt(next_match), false, false); - } - } + // If no mnemonics found, look at first character of titles. + details = FindChildForMnemonic(item, key, &TitleMatchesMnemonic); + if (details.first_match != -1) + return AcceptOrSelect(item, details); + return false; } diff --git a/views/controls/menu/menu_controller.h b/views/controls/menu/menu_controller.h index a370904..3c641c1 100644 --- a/views/controls/menu/menu_controller.h +++ b/views/controls/menu/menu_controller.h @@ -120,6 +120,8 @@ class MenuController : public MessageLoopForUI::Dispatcher { private: class MenuScrollTask; + struct SelectByCharDetails; + // Tracks selection information. struct State { State() : item(NULL), submenu_open(false) {} @@ -316,6 +318,18 @@ class MenuController : public MessageLoopForUI::Dispatcher { // If possible, closes the submenu. void CloseSubmenu(); + // Returns details about which menu items match the mnemonic |key|. + // |match_function| is used to determine which menus match. + SelectByCharDetails FindChildForMnemonic( + MenuItemView* parent, + wchar_t key, + bool (*match_function)(MenuItemView* menu, wchar_t mnemonic)); + + // Selects or accepts the appropriate menu item based on |details|. Returns + // true if |Accept| was invoked (which happens if there aren't multiple item + // with the same mnemonic and the item to select does not have a submenu). + bool AcceptOrSelect(MenuItemView* parent, const SelectByCharDetails& details); + // Selects by mnemonic, and if that doesn't work tries the first character of // the title. Returns true if a match was selected and the menu should exit. bool SelectByChar(wchar_t key); |