Reallocate static ranges for wxMenuItem

Previously different menus may use conflict ids in range 10000~.

Fix TypesettingTools/Aegisub#53
See also Aegisub/Aegisub#131
This commit is contained in:
wangqr 2020-06-03 23:49:04 -04:00
parent 71c940f933
commit 51648c638b
7 changed files with 45 additions and 26 deletions

View file

@ -55,9 +55,10 @@
#include <wx/scrolbar.h> #include <wx/scrolbar.h>
#include <wx/sizer.h> #include <wx/sizer.h>
// Check menu.h for id range allocation before editing this enum
enum { enum {
GRID_SCROLLBAR = 1730, GRID_SCROLLBAR = 1730,
MENU_SHOW_COL = 1250 // Needs 15 IDs after this MENU_SHOW_COL = (wxID_HIGHEST + 1) + 2000 // Needs 15 IDs after this
}; };
BaseGrid::BaseGrid(wxWindow* parent, agi::Context *context) BaseGrid::BaseGrid(wxWindow* parent, agi::Context *context)
@ -541,7 +542,7 @@ void BaseGrid::OnMouseEvent(wxMouseEvent &event) {
void BaseGrid::OnContextMenu(wxContextMenuEvent &evt) { void BaseGrid::OnContextMenu(wxContextMenuEvent &evt) {
wxPoint pos = evt.GetPosition(); wxPoint pos = evt.GetPosition();
if (pos == wxDefaultPosition || ScreenToClient(pos).y > lineHeight) { if (pos == wxDefaultPosition || ScreenToClient(pos).y > lineHeight) {
if (!context_menu) context_menu = menu::GetMenu("grid_context", context); if (!context_menu) context_menu = menu::GetMenu("grid_context", (wxID_HIGHEST + 1) + 8000, context);
menu::OpenPopupMenu(context_menu.get(), this); menu::OpenPopupMenu(context_menu.get(), this);
} }
else { else {

View file

@ -128,7 +128,7 @@ FrameMain::FrameMain()
EnableToolBar(*OPT_GET("App/Show Toolbar")); EnableToolBar(*OPT_GET("App/Show Toolbar"));
StartupLog("Initialize menu bar"); StartupLog("Initialize menu bar");
menu::GetMenuBar("main", this, context.get()); menu::GetMenuBar("main", this, (wxID_HIGHEST + 1) + 10000, context.get());
StartupLog("Create status bar"); StartupLog("Create status bar");
CreateStatusBar(2); CreateStatusBar(2);

View file

@ -28,6 +28,19 @@ class wxMenu;
class wxMenuBar; class wxMenuBar;
class wxWindow; class wxWindow;
/*
ID allocation for menu items:
... - wxID_ANY(-1), wxID_LOWEST(4999) - wxID_HIGHEST(5999) Reserved by wxWidgets, see documentation for wxID_HIGHEST
(wxID_HIGHEST + 1) + 2000 ~ (wxID_HIGHEST + 1) + 2014 Grid column list, see base_grid.cpp
(wxID_HIGHEST + 1) + 3000 ~ (wxID_HIGHEST + 1) + 3001 Context menu, see timeedit_ctrl.cpp
(wxID_HIGHEST + 1) + 4000 ~ (wxID_HIGHEST + 1) + 7999 Context menu, see subs_edit_ctrl.cpp
(wxID_HIGHEST + 1) + 8000 ~ (wxID_HIGHEST + 1) + 8019 Grid context menu, see base_grid.cpp
(wxID_HIGHEST + 1) + 9000 ~ (wxID_HIGHEST + 1) + 9004 Video context menu, see video_display.cpp
(wxID_HIGHEST + 1) + 10000 ~ (wxID_HIGHEST + 1) + 10999 Main menu
*/
namespace menu { namespace menu {
DEFINE_EXCEPTION(Error, agi::Exception); DEFINE_EXCEPTION(Error, agi::Exception);
DEFINE_EXCEPTION(UnknownMenu, Error); DEFINE_EXCEPTION(UnknownMenu, Error);
@ -39,7 +52,7 @@ namespace menu {
/// Throws: /// Throws:
/// UnknownMenu if no menu with the given name was found /// UnknownMenu if no menu with the given name was found
/// BadMenu if there is a menu with the given name, but it is invalid /// BadMenu if there is a menu with the given name, but it is invalid
void GetMenuBar(std::string const& name, wxFrame *window, agi::Context *c); void GetMenuBar(std::string const& name, wxFrame *window, int id_base, agi::Context *c);
/// @brief Get the menu with the specified name as a wxMenu /// @brief Get the menu with the specified name as a wxMenu
/// @param name Name of the menu /// @param name Name of the menu
@ -47,7 +60,7 @@ namespace menu {
/// Throws: /// Throws:
/// UnknownMenu if no menu with the given name was found /// UnknownMenu if no menu with the given name was found
/// BadMenu if there is a menu with the given name, but it is invalid /// BadMenu if there is a menu with the given name, but it is invalid
std::unique_ptr<wxMenu> GetMenu(std::string const& name, agi::Context *c); std::unique_ptr<wxMenu> GetMenu(std::string const& name, int id_base, agi::Context *c);
/// @brief Open a popup menu at the mouse /// @brief Open a popup menu at the mouse
/// @param menu Menu to open /// @param menu Menu to open

View file

@ -55,10 +55,11 @@
#endif #endif
namespace { namespace {
/// Window ID of first menu item
static const int MENU_ID_BASE = 10000;
class MruMenu final : public wxMenu { class MruMenu final : public wxMenu {
/// Window ID of first menu item
const int id_base;
std::string type; std::string type;
std::vector<wxMenuItem *> items; std::vector<wxMenuItem *> items;
std::vector<std::string> *cmds; std::vector<std::string> *cmds;
@ -70,7 +71,7 @@ class MruMenu final : public wxMenu {
for (size_t i = GetMenuItemCount(); i < new_size; ++i) { for (size_t i = GetMenuItemCount(); i < new_size; ++i) {
if (i >= items.size()) { if (i >= items.size()) {
items.push_back(new wxMenuItem(this, MENU_ID_BASE + cmds->size(), "_")); items.push_back(new wxMenuItem(this, id_base + cmds->size(), "_"));
cmds->push_back(agi::format("recent/%s/%d", boost::to_lower_copy(type), i)); cmds->push_back(agi::format("recent/%s/%d", boost::to_lower_copy(type), i));
} }
Append(items[i]); Append(items[i]);
@ -78,8 +79,8 @@ class MruMenu final : public wxMenu {
} }
public: public:
MruMenu(std::string type, std::vector<std::string> *cmds) MruMenu(int id_base, std::string type, std::vector<std::string> *cmds)
: type(std::move(type)) : id_base(id_base), type(std::move(type))
, cmds(cmds) , cmds(cmds)
{ {
} }
@ -122,6 +123,8 @@ public:
/// on submenus in many cases, and registering large numbers of wxEVT_UPDATE_UI /// on submenus in many cases, and registering large numbers of wxEVT_UPDATE_UI
/// handlers makes everything involves events unusably slow. /// handlers makes everything involves events unusably slow.
class CommandManager { class CommandManager {
/// Window ID of first menu item
const int id_base;
/// Menu items which need to do something on menu open /// Menu items which need to do something on menu open
std::vector<std::pair<std::string, wxMenuItem*>> dynamic_items; std::vector<std::pair<std::string, wxMenuItem*>> dynamic_items;
/// Menu items which need to be updated only when hotkeys change /// Menu items which need to be updated only when hotkeys change
@ -169,8 +172,8 @@ class CommandManager {
} }
public: public:
CommandManager(agi::Context *context) CommandManager(int id_base, agi::Context *context)
: context(context) : id_base(id_base), context(context)
, hotkeys_changed(hotkey::inst->AddHotkeyChangeListener(&CommandManager::OnHotkeysChanged, this)) , hotkeys_changed(hotkey::inst->AddHotkeyChangeListener(&CommandManager::OnHotkeysChanged, this))
{ {
} }
@ -198,7 +201,7 @@ public:
menu_text += to_wx("\t" + hotkey::get_hotkey_str_first("Default", co->name())); menu_text += to_wx("\t" + hotkey::get_hotkey_str_first("Default", co->name()));
wxMenuItem *item = new wxMenuItem(parent, MENU_ID_BASE + items.size(), menu_text, co->StrHelp(), kind); wxMenuItem *item = new wxMenuItem(parent, id_base + items.size(), menu_text, co->StrHelp(), kind);
#if defined(__WXMSW__) #if defined(__WXMSW__)
#if wxCHECK_VERSION(3, 1, 3) #if wxCHECK_VERSION(3, 1, 3)
if (kind == wxITEM_NORMAL) { if (kind == wxITEM_NORMAL) {
@ -243,7 +246,7 @@ public:
/// @param name MRU type /// @param name MRU type
/// @param parent Menu to append the new MRU menu to /// @param parent Menu to append the new MRU menu to
void AddRecent(std::string const& name, wxMenu *parent) { void AddRecent(std::string const& name, wxMenu *parent) {
mru.push_back(new MruMenu(name, &items)); mru.push_back(new MruMenu(id_base, name, &items));
parent->AppendSubMenu(mru.back(), _("&Recent")); parent->AppendSubMenu(mru.back(), _("&Recent"));
} }
@ -257,7 +260,7 @@ public:
void OnMenuClick(wxCommandEvent &evt) { void OnMenuClick(wxCommandEvent &evt) {
// This also gets clicks on unrelated things such as the toolbar, so // This also gets clicks on unrelated things such as the toolbar, so
// the window ID ranges really need to be unique // the window ID ranges really need to be unique
size_t id = static_cast<size_t>(evt.GetId() - MENU_ID_BASE); size_t id = static_cast<size_t>(evt.GetId() - id_base);
if (id < items.size() && context) if (id < items.size() && context)
cmd::call(items[id], context); cmd::call(items[id], context);
@ -290,13 +293,13 @@ public:
/// Wrapper for wxMenu to add a command manager /// Wrapper for wxMenu to add a command manager
struct CommandMenu final : public wxMenu { struct CommandMenu final : public wxMenu {
CommandManager cm; CommandManager cm;
CommandMenu(agi::Context *c) : cm(c) { } CommandMenu(int id_base, agi::Context *c) : cm(id_base, c) { }
}; };
/// Wrapper for wxMenuBar to add a command manager /// Wrapper for wxMenuBar to add a command manager
struct CommandMenuBar final : public wxMenuBar { struct CommandMenuBar final : public wxMenuBar {
CommandManager cm; CommandManager cm;
CommandMenuBar(agi::Context *c) : cm(c) { } CommandMenuBar(int id_base, agi::Context *c) : cm(id_base, c) { }
}; };
/// Read a string from a json object /// Read a string from a json object
@ -512,7 +515,7 @@ public:
} }
namespace menu { namespace menu {
void GetMenuBar(std::string const& name, wxFrame *window, agi::Context *c) { void GetMenuBar(std::string const& name, wxFrame *window, int id_base, agi::Context *c) {
#ifdef __WXMAC__ #ifdef __WXMAC__
auto bind_events = [&](CommandMenuBar *menu) { auto bind_events = [&](CommandMenuBar *menu) {
window->Bind(wxEVT_ACTIVATE, [=](wxActivateEvent&) { menu->cm.SetContext(c); }); window->Bind(wxEVT_ACTIVATE, [=](wxActivateEvent&) { menu->cm.SetContext(c); });
@ -528,7 +531,7 @@ namespace menu {
} }
#endif #endif
auto menu = agi::make_unique<CommandMenuBar>(c); auto menu = agi::make_unique<CommandMenuBar>(id_base, c);
for (auto const& item : get_menu(name)) { for (auto const& item : get_menu(name)) {
std::string submenu, disp; std::string submenu, disp;
read_entry(item, "submenu", &submenu); read_entry(item, "submenu", &submenu);
@ -561,8 +564,8 @@ namespace menu {
menu.release(); menu.release();
} }
std::unique_ptr<wxMenu> GetMenu(std::string const& name, agi::Context *c) { std::unique_ptr<wxMenu> GetMenu(std::string const& name, int id_base, agi::Context *c) {
auto menu = agi::make_unique<CommandMenu>(c); auto menu = agi::make_unique<CommandMenu>(id_base, c);
build_menu(name, c, &menu->cm, menu.get()); build_menu(name, c, &menu->cm, menu.get());
menu->Bind(wxEVT_MENU_OPEN, &CommandManager::OnMenuOpen, &menu->cm); menu->Bind(wxEVT_MENU_OPEN, &CommandManager::OnMenuOpen, &menu->cm);
menu->Bind(wxEVT_MENU, &CommandManager::OnMenuClick, &menu->cm); menu->Bind(wxEVT_MENU, &CommandManager::OnMenuClick, &menu->cm);

View file

@ -61,8 +61,9 @@
#define LANGS_MAX 1000 #define LANGS_MAX 1000
/// Event ids /// Event ids
// Check menu.h for id range allocation before editing this enum
enum { enum {
EDIT_MENU_SPLIT_PRESERVE = 1400, EDIT_MENU_SPLIT_PRESERVE = (wxID_HIGHEST + 1) + 4000,
EDIT_MENU_SPLIT_ESTIMATE, EDIT_MENU_SPLIT_ESTIMATE,
EDIT_MENU_SPLIT_VIDEO, EDIT_MENU_SPLIT_VIDEO,
EDIT_MENU_CUT, EDIT_MENU_CUT,
@ -73,9 +74,9 @@ enum {
EDIT_MENU_REMOVE_FROM_DICT, EDIT_MENU_REMOVE_FROM_DICT,
EDIT_MENU_SUGGESTION, EDIT_MENU_SUGGESTION,
EDIT_MENU_SUGGESTIONS, EDIT_MENU_SUGGESTIONS,
EDIT_MENU_THESAURUS = 1450, EDIT_MENU_THESAURUS = (wxID_HIGHEST + 1) + 5000,
EDIT_MENU_THESAURUS_SUGS, EDIT_MENU_THESAURUS_SUGS,
EDIT_MENU_DIC_LANGUAGE = 1600, EDIT_MENU_DIC_LANGUAGE = (wxID_HIGHEST + 1) + 6000,
EDIT_MENU_DIC_LANGS, EDIT_MENU_DIC_LANGS,
EDIT_MENU_THES_LANGUAGE = EDIT_MENU_DIC_LANGUAGE + LANGS_MAX, EDIT_MENU_THES_LANGUAGE = EDIT_MENU_DIC_LANGUAGE + LANGS_MAX,
EDIT_MENU_THES_LANGS EDIT_MENU_THES_LANGS

View file

@ -48,8 +48,9 @@
#define TimeEditWindowStyle #define TimeEditWindowStyle
// Check menu.h for id range allocation before editing this enum
enum { enum {
Time_Edit_Copy = 1320, Time_Edit_Copy = (wxID_HIGHEST + 1) + 3000,
Time_Edit_Paste Time_Edit_Paste
}; };

View file

@ -383,7 +383,7 @@ void VideoDisplay::OnMouseWheel(wxMouseEvent& event) {
} }
void VideoDisplay::OnContextMenu(wxContextMenuEvent&) { void VideoDisplay::OnContextMenu(wxContextMenuEvent&) {
if (!context_menu) context_menu = menu::GetMenu("video_context", con); if (!context_menu) context_menu = menu::GetMenu("video_context", (wxID_HIGHEST + 1) + 9000, con);
SetCursor(wxNullCursor); SetCursor(wxNullCursor);
menu::OpenPopupMenu(context_menu.get(), this); menu::OpenPopupMenu(context_menu.get(), this);
} }