Do not behave like bookmarkbar menu

This commit is contained in:
Cheng Zhao 2016-01-05 11:57:58 +08:00
parent 48451032e3
commit 43bfce26a7
4 changed files with 60 additions and 65 deletions

View file

@ -134,6 +134,16 @@ bool MenuBar::GetMenuButtonFromScreenPoint(const gfx::Point& point,
return false; return false;
} }
void MenuBar::RunMenu(views::MenuButton* button) {
int id = button->tag();
ui::MenuModel::ItemType type = menu_model_->GetTypeAt(id);
if (type != ui::MenuModel::TYPE_SUBMENU)
return;
MenuDelegate menu_delegate(this);
menu_delegate.RunMenu(menu_model_->GetSubmenuModelAt(id), button);
}
const char* MenuBar::GetClassName() const { const char* MenuBar::GetClassName() const {
return kViewClassName; return kViewClassName;
} }
@ -150,13 +160,7 @@ void MenuBar::OnMenuButtonClicked(views::View* source,
return; return;
views::MenuButton* button = static_cast<views::MenuButton*>(source); views::MenuButton* button = static_cast<views::MenuButton*>(source);
int id = button->tag(); RunMenu(button);
ui::MenuModel::ItemType type = menu_model_->GetTypeAt(id);
if (type != ui::MenuModel::TYPE_SUBMENU)
return;
menu_delegate_.reset(new MenuDelegate(this));
menu_delegate_->RunMenu(menu_model_->GetSubmenuModelAt(id), button);
} }
} // namespace atom } // namespace atom

View file

@ -49,6 +49,9 @@ class MenuBar : public views::View,
ui::MenuModel** menu_model, ui::MenuModel** menu_model,
views::MenuButton** button); views::MenuButton** button);
// Shows the menu with |button|.
void RunMenu(views::MenuButton* button);
protected: protected:
// views::View: // views::View:
const char* GetClassName() const override; const char* GetClassName() const override;
@ -71,7 +74,6 @@ class MenuBar : public views::View,
#endif #endif
ui::MenuModel* menu_model_; ui::MenuModel* menu_model_;
scoped_ptr<MenuDelegate> menu_delegate_;
DISALLOW_COPY_AND_ASSIGN(MenuBar); DISALLOW_COPY_AND_ASSIGN(MenuBar);
}; };

View file

@ -5,7 +5,7 @@
#include "atom/browser/ui/views/menu_delegate.h" #include "atom/browser/ui/views/menu_delegate.h"
#include "atom/browser/ui/views/menu_bar.h" #include "atom/browser/ui/views/menu_bar.h"
#include "base/stl_util.h" #include "content/public/browser/browser_thread.h"
#include "ui/views/controls/button/menu_button.h" #include "ui/views/controls/button/menu_button.h"
#include "ui/views/controls/menu/menu_item_view.h" #include "ui/views/controls/menu/menu_item_view.h"
#include "ui/views/controls/menu/menu_model_adapter.h" #include "ui/views/controls/menu/menu_model_adapter.h"
@ -16,13 +16,10 @@ namespace atom {
MenuDelegate::MenuDelegate(MenuBar* menu_bar) MenuDelegate::MenuDelegate(MenuBar* menu_bar)
: menu_bar_(menu_bar), : menu_bar_(menu_bar),
id_(-1), id_(-1) {
items_(menu_bar_->GetItemCount()),
delegates_(menu_bar_->GetItemCount()) {
} }
MenuDelegate::~MenuDelegate() { MenuDelegate::~MenuDelegate() {
STLDeleteElements(&delegates_);
} }
void MenuDelegate::RunMenu(ui::MenuModel* model, views::MenuButton* button) { void MenuDelegate::RunMenu(ui::MenuModel* model, views::MenuButton* button) {
@ -33,12 +30,15 @@ void MenuDelegate::RunMenu(ui::MenuModel* model, views::MenuButton* button) {
button->height() - 1); button->height() - 1);
id_ = button->tag(); id_ = button->tag();
views::MenuItemView* item = BuildMenu(model); adapter_.reset(new views::MenuModelAdapter(model));
views::MenuRunner menu_runner( views::MenuItemView* item = new views::MenuItemView(this);
static_cast<views::MenuModelAdapter*>(adapter_.get())->BuildMenu(item);
menu_runner_.reset(new views::MenuRunner(
item, item,
views::MenuRunner::CONTEXT_MENU | views::MenuRunner::HAS_MNEMONICS); views::MenuRunner::CONTEXT_MENU | views::MenuRunner::HAS_MNEMONICS));
ignore_result(menu_runner.RunMenuAt( ignore_result(menu_runner_->RunMenuAt(
button->GetWidget()->GetTopLevelWidget(), button->GetWidget()->GetTopLevelWidget(),
button, button,
bounds, bounds,
@ -46,68 +46,53 @@ void MenuDelegate::RunMenu(ui::MenuModel* model, views::MenuButton* button) {
ui::MENU_SOURCE_MOUSE)); ui::MENU_SOURCE_MOUSE));
} }
views::MenuItemView* MenuDelegate::BuildMenu(ui::MenuModel* model) {
DCHECK_GE(id_, 0);
if (!items_[id_]) {
views::MenuModelAdapter* delegate = new views::MenuModelAdapter(model);
delegates_[id_] = delegate;
views::MenuItemView* item = new views::MenuItemView(this);
delegate->BuildMenu(item);
items_[id_] = item;
}
return items_[id_];
}
void MenuDelegate::ExecuteCommand(int id) { void MenuDelegate::ExecuteCommand(int id) {
delegate()->ExecuteCommand(id); adapter_->ExecuteCommand(id);
} }
void MenuDelegate::ExecuteCommand(int id, int mouse_event_flags) { void MenuDelegate::ExecuteCommand(int id, int mouse_event_flags) {
delegate()->ExecuteCommand(id, mouse_event_flags); adapter_->ExecuteCommand(id, mouse_event_flags);
} }
bool MenuDelegate::IsTriggerableEvent(views::MenuItemView* source, bool MenuDelegate::IsTriggerableEvent(views::MenuItemView* source,
const ui::Event& e) { const ui::Event& e) {
return delegate()->IsTriggerableEvent(source, e); return adapter_->IsTriggerableEvent(source, e);
} }
bool MenuDelegate::GetAccelerator(int id, ui::Accelerator* accelerator) const { bool MenuDelegate::GetAccelerator(int id, ui::Accelerator* accelerator) const {
return delegate()->GetAccelerator(id, accelerator); return adapter_->GetAccelerator(id, accelerator);
} }
base::string16 MenuDelegate::GetLabel(int id) const { base::string16 MenuDelegate::GetLabel(int id) const {
return delegate()->GetLabel(id); return adapter_->GetLabel(id);
} }
const gfx::FontList* MenuDelegate::GetLabelFontList(int id) const { const gfx::FontList* MenuDelegate::GetLabelFontList(int id) const {
return delegate()->GetLabelFontList(id); return adapter_->GetLabelFontList(id);
} }
bool MenuDelegate::IsCommandEnabled(int id) const { bool MenuDelegate::IsCommandEnabled(int id) const {
return delegate()->IsCommandEnabled(id); return adapter_->IsCommandEnabled(id);
} }
bool MenuDelegate::IsCommandVisible(int id) const { bool MenuDelegate::IsCommandVisible(int id) const {
return delegate()->IsCommandVisible(id); return adapter_->IsCommandVisible(id);
} }
bool MenuDelegate::IsItemChecked(int id) const { bool MenuDelegate::IsItemChecked(int id) const {
return delegate()->IsItemChecked(id); return adapter_->IsItemChecked(id);
} }
void MenuDelegate::SelectionChanged(views::MenuItemView* menu) { void MenuDelegate::SelectionChanged(views::MenuItemView* menu) {
delegate()->SelectionChanged(menu); adapter_->SelectionChanged(menu);
} }
void MenuDelegate::WillShowMenu(views::MenuItemView* menu) { void MenuDelegate::WillShowMenu(views::MenuItemView* menu) {
delegate()->WillShowMenu(menu); adapter_->WillShowMenu(menu);
} }
void MenuDelegate::WillHideMenu(views::MenuItemView* menu) { void MenuDelegate::WillHideMenu(views::MenuItemView* menu) {
delegate()->WillHideMenu(menu); adapter_->WillHideMenu(menu);
} }
views::MenuItemView* MenuDelegate::GetSiblingMenu( views::MenuItemView* MenuDelegate::GetSiblingMenu(
@ -115,16 +100,28 @@ views::MenuItemView* MenuDelegate::GetSiblingMenu(
const gfx::Point& screen_point, const gfx::Point& screen_point,
views::MenuAnchorPosition* anchor, views::MenuAnchorPosition* anchor,
bool* has_mnemonics, bool* has_mnemonics,
views::MenuButton** button) { views::MenuButton**) {
views::MenuButton* button;
ui::MenuModel* model; ui::MenuModel* model;
if (!menu_bar_->GetMenuButtonFromScreenPoint(screen_point, &model, button)) if (menu_bar_->GetMenuButtonFromScreenPoint(screen_point, &model, &button) &&
return NULL; button->tag() != id_) {
// Switch to sibling menu on next tick, otherwise crash may happen.
content::BrowserThread::PostTask(
content::BrowserThread::UI, FROM_HERE,
base::Bind(&MenuDelegate::SwitchToSiblingMenu,
base::Unretained(this), button));
}
*anchor = views::MENU_ANCHOR_TOPLEFT; return nullptr;
*has_mnemonics = true; }
id_ = (*button)->tag(); void MenuDelegate::SwitchToSiblingMenu(views::MenuButton* button) {
return BuildMenu(model); menu_runner_->Cancel();
// After canceling the menu, we need to wait until next tick so we are out of
// nested message loop.
content::BrowserThread::PostTask(
content::BrowserThread::UI, FROM_HERE,
base::Bind(&MenuBar::RunMenu, base::Unretained(menu_bar_), button));
} }
} // namespace atom } // namespace atom

View file

@ -5,12 +5,11 @@
#ifndef ATOM_BROWSER_UI_VIEWS_MENU_DELEGATE_H_ #ifndef ATOM_BROWSER_UI_VIEWS_MENU_DELEGATE_H_
#define ATOM_BROWSER_UI_VIEWS_MENU_DELEGATE_H_ #define ATOM_BROWSER_UI_VIEWS_MENU_DELEGATE_H_
#include <vector> #include "base/memory/scoped_ptr.h"
#include "ui/views/controls/menu/menu_delegate.h" #include "ui/views/controls/menu/menu_delegate.h"
namespace views { namespace views {
class MenuModelAdapter; class MenuRunner;
} }
namespace ui { namespace ui {
@ -51,20 +50,13 @@ class MenuDelegate : public views::MenuDelegate {
views::MenuButton** button) override; views::MenuButton** button) override;
private: private:
// Gets the cached menu item view from the model. // Close this menu and run the menu of |button|.
views::MenuItemView* BuildMenu(ui::MenuModel* model); void SwitchToSiblingMenu(views::MenuButton* button);
// Returns delegate for current item.
views::MenuDelegate* delegate() const { return delegates_[id_]; }
MenuBar* menu_bar_; MenuBar* menu_bar_;
// Current item's id.
int id_; int id_;
// Cached menu items, managed by MenuRunner. scoped_ptr<views::MenuDelegate> adapter_;
std::vector<views::MenuItemView*> items_; scoped_ptr<views::MenuRunner> menu_runner_;
// Cached menu delegates for each menu item, managed by us.
std::vector<views::MenuDelegate*> delegates_;
DISALLOW_COPY_AND_ASSIGN(MenuDelegate); DISALLOW_COPY_AND_ASSIGN(MenuDelegate);
}; };