diff --git a/src/base_grid.cpp b/src/base_grid.cpp index 509474ddf8..f058e6e97e 100644 --- a/src/base_grid.cpp +++ b/src/base_grid.cpp @@ -56,9 +56,10 @@ #include #include +// Check menu.h for id range allocation before editing this enum enum { 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) @@ -579,7 +580,7 @@ void BaseGrid::OnMouseEvent(wxMouseEvent &event) { void BaseGrid::OnContextMenu(wxContextMenuEvent &evt) { wxPoint pos = evt.GetPosition(); 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); } else { diff --git a/src/font_file_lister.cpp b/src/font_file_lister.cpp index 813e2ca967..a98c582647 100644 --- a/src/font_file_lister.cpp +++ b/src/font_file_lister.cpp @@ -139,6 +139,8 @@ void FontCollector::ProcessDialogueLine(const AssDialogue *line, int index) { break; } case AssBlockType::DRAWING: + used_styles[style].drawing = true; + break; case AssBlockType::COMMENT: break; } @@ -146,7 +148,11 @@ void FontCollector::ProcessDialogueLine(const AssDialogue *line, int index) { } void FontCollector::ProcessChunk(std::pair const& style) { - if (style.second.chars.empty()) return; + if (style.second.chars.empty() && !style.second.drawing) return; + + if (style.second.chars.empty() && style.second.drawing) { + status_callback(fmt_tl("Font '%s' is used in a drawing, but not in any text.\n", style.first.facename), 3); + } auto res = lister.GetFontPaths(style.first.facename, style.first.bold, style.first.italic, style.second.chars); diff --git a/src/font_file_lister.h b/src/font_file_lister.h index 9c7cd51ad7..82f02cca9d 100644 --- a/src/font_file_lister.h +++ b/src/font_file_lister.h @@ -127,6 +127,7 @@ class FontCollector { /// Data about where each style is used struct UsageData { std::vector chars; ///< Characters used in this style which glyphs will be needed for + bool drawing = false; ///< Whether this style is used for a drawing std::vector lines; ///< Lines on which this style is used via overrides std::vector styles; ///< ASS styles which use this style }; diff --git a/src/frame_main.cpp b/src/frame_main.cpp index b37dafefc0..6e6efe7a21 100644 --- a/src/frame_main.cpp +++ b/src/frame_main.cpp @@ -128,7 +128,7 @@ FrameMain::FrameMain() EnableToolBar(*OPT_GET("App/Show Toolbar")); StartupLog("Initialize menu bar"); - menu::GetMenuBar("main", this, context.get()); + menu::GetMenuBar("main", this, (wxID_HIGHEST + 1) + 10000, context.get()); StartupLog("Create status bar"); CreateStatusBar(2); diff --git a/src/include/aegisub/menu.h b/src/include/aegisub/menu.h index 266a3c669a..22d709cbd6 100644 --- a/src/include/aegisub/menu.h +++ b/src/include/aegisub/menu.h @@ -28,6 +28,19 @@ class wxMenu; class wxMenuBar; 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 { DEFINE_EXCEPTION(Error, agi::Exception); DEFINE_EXCEPTION(UnknownMenu, Error); @@ -39,7 +52,7 @@ namespace menu { /// Throws: /// UnknownMenu if no menu with the given name was found /// 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 /// @param name Name of the menu @@ -47,7 +60,7 @@ namespace menu { /// Throws: /// UnknownMenu if no menu with the given name was found /// BadMenu if there is a menu with the given name, but it is invalid - std::unique_ptr GetMenu(std::string const& name, agi::Context *c); + std::unique_ptr GetMenu(std::string const& name, int id_base, agi::Context *c); /// @brief Open a popup menu at the mouse /// @param menu Menu to open diff --git a/src/libresrc/default_hotkey.json b/src/libresrc/default_hotkey.json index db56f32c18..789ab0d780 100644 --- a/src/libresrc/default_hotkey.json +++ b/src/libresrc/default_hotkey.json @@ -125,10 +125,10 @@ "edit/line/delete" : [ "Ctrl-Delete" ], - "edit/line/duplicate/shift" : [ + "edit/line/split/before" : [ "Ctrl-D" ], - "edit/line/duplicate/shift_back" : [ + "edit/line/split/after" : [ "Ctrl-Shift-D" ], "edit/line/paste" : [ diff --git a/src/libresrc/osx/default_hotkey.json b/src/libresrc/osx/default_hotkey.json index 888d3f02bf..7789f510c0 100644 --- a/src/libresrc/osx/default_hotkey.json +++ b/src/libresrc/osx/default_hotkey.json @@ -128,10 +128,10 @@ "edit/line/delete" : [ "Ctrl-Backspace" ], - "edit/line/duplicate/shift" : [ + "edit/line/split/before" : [ "Ctrl-D" ], - "edit/line/duplicate/shift_back" : [ + "edit/line/split/after" : [ "Ctrl-Shift-D" ], "edit/line/paste" : [ diff --git a/src/menu.cpp b/src/menu.cpp index 99edbcf7d8..7d66ad3d3c 100644 --- a/src/menu.cpp +++ b/src/menu.cpp @@ -51,10 +51,11 @@ #endif namespace { -/// Window ID of first menu item -static const int MENU_ID_BASE = 10000; class MruMenu final : public wxMenu { + /// Window ID of first menu item + const int id_base; + std::string type; std::vector items; std::vector *cmds; @@ -66,7 +67,7 @@ class MruMenu final : public wxMenu { for (size_t i = GetMenuItemCount(); i < new_size; ++i) { 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)); } Append(items[i]); @@ -74,8 +75,8 @@ class MruMenu final : public wxMenu { } public: - MruMenu(std::string type, std::vector *cmds) - : type(std::move(type)) + MruMenu(int id_base, std::string type, std::vector *cmds) + : id_base(id_base), type(std::move(type)) , cmds(cmds) { } @@ -118,6 +119,8 @@ class MruMenu final : public wxMenu { /// on submenus in many cases, and registering large numbers of wxEVT_UPDATE_UI /// handlers makes everything involves events unusably slow. class CommandManager { + /// Window ID of first menu item + const int id_base; /// Menu items which need to do something on menu open std::vector> dynamic_items; /// Menu items which need to be updated only when hotkeys change @@ -165,8 +168,8 @@ class CommandManager { } public: - CommandManager(agi::Context *context) - : context(context) + CommandManager(int id_base, agi::Context *context) + : id_base(id_base), context(context) , hotkeys_changed(hotkey::inst->AddHotkeyChangeListener(&CommandManager::OnHotkeysChanged, this)) { } @@ -194,7 +197,7 @@ class CommandManager { 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); #ifndef __WXMAC__ /// @todo Maybe make this a configuration option instead? if (kind == wxITEM_NORMAL) @@ -229,7 +232,7 @@ class CommandManager { /// @param name MRU type /// @param parent Menu to append the new MRU menu to 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")); } @@ -243,7 +246,7 @@ class CommandManager { void OnMenuClick(wxCommandEvent &evt) { // This also gets clicks on unrelated things such as the toolbar, so // the window ID ranges really need to be unique - size_t id = static_cast(evt.GetId() - MENU_ID_BASE); + size_t id = static_cast(evt.GetId() - id_base); if (id < items.size() && context) cmd::call(items[id], context); @@ -276,13 +279,13 @@ class CommandManager { /// Wrapper for wxMenu to add a command manager struct CommandMenu final : public wxMenu { 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 struct CommandMenuBar final : public wxMenuBar { 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 @@ -498,7 +501,7 @@ class AutomationMenu final : public wxMenu { } 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__ auto bind_events = [&](CommandMenuBar *menu) { window->Bind(wxEVT_ACTIVATE, [=](wxActivateEvent&) { menu->cm.SetContext(c); }); @@ -514,7 +517,7 @@ namespace menu { } #endif - auto menu = agi::make_unique(c); + auto menu = agi::make_unique(id_base, c); for (auto const& item : get_menu(name)) { std::string submenu, disp; read_entry(item, "submenu", &submenu); @@ -547,8 +550,8 @@ namespace menu { menu.release(); } - std::unique_ptr GetMenu(std::string const& name, agi::Context *c) { - auto menu = agi::make_unique(c); + std::unique_ptr GetMenu(std::string const& name, int id_base, agi::Context *c) { + auto menu = agi::make_unique(id_base, c); build_menu(name, c, &menu->cm, menu.get()); menu->Bind(wxEVT_MENU_OPEN, &CommandManager::OnMenuOpen, &menu->cm); menu->Bind(wxEVT_MENU, &CommandManager::OnMenuClick, &menu->cm); diff --git a/src/subs_edit_ctrl.cpp b/src/subs_edit_ctrl.cpp index ee5389a489..30cd3e10c4 100644 --- a/src/subs_edit_ctrl.cpp +++ b/src/subs_edit_ctrl.cpp @@ -61,8 +61,9 @@ #define LANGS_MAX 1000 /// Event ids +// Check menu.h for id range allocation before editing this enum enum { - EDIT_MENU_SPLIT_PRESERVE = 1400, + EDIT_MENU_SPLIT_PRESERVE = (wxID_HIGHEST + 1) + 4000, EDIT_MENU_SPLIT_ESTIMATE, EDIT_MENU_SPLIT_VIDEO, EDIT_MENU_CUT, @@ -73,9 +74,9 @@ enum { EDIT_MENU_REMOVE_FROM_DICT, EDIT_MENU_SUGGESTION, EDIT_MENU_SUGGESTIONS, - EDIT_MENU_THESAURUS = 1450, + EDIT_MENU_THESAURUS = (wxID_HIGHEST + 1) + 5000, EDIT_MENU_THESAURUS_SUGS, - EDIT_MENU_DIC_LANGUAGE = 1600, + EDIT_MENU_DIC_LANGUAGE = (wxID_HIGHEST + 1) + 6000, EDIT_MENU_DIC_LANGS, EDIT_MENU_THES_LANGUAGE = EDIT_MENU_DIC_LANGUAGE + LANGS_MAX, EDIT_MENU_THES_LANGS diff --git a/src/timeedit_ctrl.cpp b/src/timeedit_ctrl.cpp index 4ccb454d37..f9aff32089 100644 --- a/src/timeedit_ctrl.cpp +++ b/src/timeedit_ctrl.cpp @@ -48,8 +48,9 @@ #define TimeEditWindowStyle +// Check menu.h for id range allocation before editing this enum enum { - Time_Edit_Copy = 1320, + Time_Edit_Copy = (wxID_HIGHEST + 1) + 3000, Time_Edit_Paste }; diff --git a/src/video_display.cpp b/src/video_display.cpp index 8636fba475..8c6a0a4421 100644 --- a/src/video_display.cpp +++ b/src/video_display.cpp @@ -416,7 +416,7 @@ void VideoDisplay::OnMouseWheel(wxMouseEvent& event) { } 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); menu::OpenPopupMenu(context_menu.get(), this); }