Do not behave like bookmarkbar menu
This commit is contained in:
parent
48451032e3
commit
43bfce26a7
4 changed files with 60 additions and 65 deletions
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
};
|
};
|
||||||
|
|
|
@ -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
|
||||||
|
|
|
@ -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);
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in a new issue