72a089262e
* chore: bump chromium in DEPS to a7249f73ae05d456c04487ef1693325f719556dd * chore: bump chromium in DEPS to 202466fa40b58f0bb9c9a76a037d1c50154c099e * chore: bump chromium in DEPS to 2dd1b25c8d794b50fb0dd911e0c4e909ff39f145 * Update patches * update patches * Revert "[printing] Mojofy PrintHostMsg_CheckForCancel" https://chromium-review.googlesource.com/c/chromium/src/+/2226002 * chore: bump chromium in DEPS to 8c1542e7dd36854fdf4abd1a8021eeb65a6a2e2b * chore: bump chromium in DEPS to 078bc6d796334fb403acd8975b99d1c8ecd028e8 * chore: bump chromium in DEPS to d96e9f16ae852ec9dbd15bf17df3d440402413bb * update patches * chore: update patches * Use ExtensionSystem::is_ready() instead of ExtensionService::is_ready() https://chromium-review.googlesource.com/c/chromium/src/+/2207499 * Remove WebImeTextSpan https://chromium-review.googlesource.com/c/chromium/src/+/2225240 * Remove PDFAnnotations flag altogether. https://chromium-review.googlesource.com/c/chromium/src/+/2229317 * Rework find-from-selection so it's synchronous -- fixes flaky tests https://chromium-review.googlesource.com/c/chromium/src/+/2181570 * fixup! Revert "[printing] Mojofy PrintHostMsg_CheckForCancel" * chore: bump chromium in DEPS to a8a280835830c65145ed8573a9a09f36d3920418 * update sysroots * update patches * update patches * Take RFH as a parameter for DidUpdateFavicon/ManifestURL https://chromium-review.googlesource.com/c/chromium/src/+/2224745 * chore: bump chromium in DEPS to b6149cb5a5e32caf8eab67b97ef3072b72521ca8 * Update patches * Rename net::cookie_util::StripStatuses to StripAccessResults https://chromium-review.googlesource.com/c/chromium/src/+/2212697 * use net::CookieAccessResultList instead of net::CookieAccessResultList * fix mas_no_private_api patch https://chromium-review.googlesource.com/c/chromium/src/+/2230281 * chore: bump chromium in DEPS to a27feee1643d952e48f77c92d8c03aedea14b720 * update patches * fix: add new navigation state REUSE_SITE_INSTANCE To fix the new set of state checks added in https://chromium-review.googlesource.com/c/chromium/src/+/2215141 * chore: bump chromium in DEPS to ff4559a4c13d20888202474e4ab9917dbdad8a9a * update patches * Cleanup usages of old mojo types and remove unused code https://chromium-review.googlesource.com/c/chromium/src/+/2235699 * chore: bump chromium in DEPS to 05279845f76eb22900a8b0d1a11d4fd339a8e53b * chore: bump chromium in DEPS to 821558279767cffec90e3b5b947865f90089fed3 * chore: bump chromium in DEPS to 1aef04e6486be337d3dd820b2d64d6320a1b9c13 * chore: bump chromium in DEPS to dc86386e8fdd796a0f7577e91e42a7f8b7e9bc78 * chore: bump chromium in DEPS to 64f2360794f14643764092ba3e58e2ed8f9fee12 * chore: update patches * refactor: MessageLoop, you are terminated \o/ Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2246173 * refactor: plumb DownloadSchedule to DownloadItem Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2242202 * chore: fix variable typo in IPC * chore: s/BindPipeAndPassReceiver/BindNewPipeAndPassReceiver * chore: update patches * chore: XEvent becomes x11::Event Refs: https://chromium-review.googlesource.com/c/chromium/src/+/2240355 * fixup! refactor: MessageLoop, you are terminated \o/ * fixup! chore: XEvent becomes x11::Event * build: update v8 headers * chore: fix windows build * chore: disable SameSite-by-default changes https://chromium-review.googlesource.com/c/chromium/src/+/2231445 * update printing.patch * chore: bump chromium DEPS to 9ae03ef8f7d4f6ac663f725bcfe70311987652f3 * Convert WidgetHostMsg_SelectionBoundsChanged/TextInputStateChanged https://chromium-review.googlesource.com/c/chromium/src/+/2243531 * chore: update v8 patches * [XProto] Replace usages of XID and ::Window with x11::Window https://chromium-review.googlesource.com/c/chromium/src/+/2249389 * Update VideoFrameMetadata to use base::Optionals https://chromium-review.googlesource.com/c/chromium/src/+/2231706 https://chromium-review.googlesource.com/c/chromium/src/+/2238361 * --disable-dev-shm-usage for gpu process crash * [v8] Allow for 4GB TypedArrays https://chromium-review.googlesource.com/c/v8/v8/+/2249668 * update lib_src_switch_buffer_kmaxlength_to_size_t.patch * disable app.getGPUInfo spec on linux * update patches Co-authored-by: John Kleinschmidt <jkleinsc@github.com> Co-authored-by: Electron Bot <anonymous@electronjs.org> Co-authored-by: deepak1556 <hop2deep@gmail.com> Co-authored-by: Shelley Vohr <shelley.vohr@gmail.com> Co-authored-by: Samuel Attard <samuel.r.attard@gmail.com>
337 lines
12 KiB
C++
337 lines
12 KiB
C++
// Copyright (c) 2014 GitHub, Inc.
|
|
// Use of this source code is governed by the MIT license that can be
|
|
// found in the LICENSE file.
|
|
|
|
#include "shell/browser/ui/views/global_menu_bar_x11.h"
|
|
|
|
#include <dlfcn.h>
|
|
#include <glib-object.h>
|
|
|
|
#include "base/logging.h"
|
|
#include "base/strings/stringprintf.h"
|
|
#include "base/strings/utf_string_conversions.h"
|
|
#include "chrome/browser/ui/views/frame/global_menu_bar_registrar_x11.h"
|
|
#include "shell/browser/native_window_views.h"
|
|
#include "shell/browser/ui/electron_menu_model.h"
|
|
#include "ui/aura/window.h"
|
|
#include "ui/aura/window_tree_host.h"
|
|
#include "ui/base/accelerators/menu_label_accelerator_util_linux.h"
|
|
#include "ui/events/keycodes/keyboard_code_conversion_x.h"
|
|
#include "ui/gfx/x/x11.h"
|
|
|
|
// libdbusmenu-glib types
|
|
typedef struct _DbusmenuMenuitem DbusmenuMenuitem;
|
|
typedef DbusmenuMenuitem* (*dbusmenu_menuitem_new_func)();
|
|
typedef DbusmenuMenuitem* (*dbusmenu_menuitem_new_with_id_func)(int id);
|
|
|
|
typedef int (*dbusmenu_menuitem_get_id_func)(DbusmenuMenuitem* item);
|
|
typedef GList* (*dbusmenu_menuitem_get_children_func)(DbusmenuMenuitem* item);
|
|
typedef DbusmenuMenuitem* (*dbusmenu_menuitem_child_append_func)(
|
|
DbusmenuMenuitem* parent,
|
|
DbusmenuMenuitem* child);
|
|
typedef DbusmenuMenuitem* (*dbusmenu_menuitem_property_set_func)(
|
|
DbusmenuMenuitem* item,
|
|
const char* property,
|
|
const char* value);
|
|
typedef DbusmenuMenuitem* (*dbusmenu_menuitem_property_set_variant_func)(
|
|
DbusmenuMenuitem* item,
|
|
const char* property,
|
|
GVariant* value);
|
|
typedef DbusmenuMenuitem* (*dbusmenu_menuitem_property_set_bool_func)(
|
|
DbusmenuMenuitem* item,
|
|
const char* property,
|
|
bool value);
|
|
typedef DbusmenuMenuitem* (*dbusmenu_menuitem_property_set_int_func)(
|
|
DbusmenuMenuitem* item,
|
|
const char* property,
|
|
int value);
|
|
|
|
typedef struct _DbusmenuServer DbusmenuServer;
|
|
typedef DbusmenuServer* (*dbusmenu_server_new_func)(const char* object);
|
|
typedef void (*dbusmenu_server_set_root_func)(DbusmenuServer* self,
|
|
DbusmenuMenuitem* root);
|
|
|
|
namespace electron {
|
|
|
|
namespace {
|
|
|
|
// Retrieved functions from libdbusmenu-glib.
|
|
|
|
// DbusmenuMenuItem methods:
|
|
dbusmenu_menuitem_new_func menuitem_new = nullptr;
|
|
dbusmenu_menuitem_new_with_id_func menuitem_new_with_id = nullptr;
|
|
dbusmenu_menuitem_get_id_func menuitem_get_id = nullptr;
|
|
dbusmenu_menuitem_get_children_func menuitem_get_children = nullptr;
|
|
dbusmenu_menuitem_get_children_func menuitem_take_children = nullptr;
|
|
dbusmenu_menuitem_child_append_func menuitem_child_append = nullptr;
|
|
dbusmenu_menuitem_property_set_func menuitem_property_set = nullptr;
|
|
dbusmenu_menuitem_property_set_variant_func menuitem_property_set_variant =
|
|
nullptr;
|
|
dbusmenu_menuitem_property_set_bool_func menuitem_property_set_bool = nullptr;
|
|
dbusmenu_menuitem_property_set_int_func menuitem_property_set_int = nullptr;
|
|
|
|
// DbusmenuServer methods:
|
|
dbusmenu_server_new_func server_new = nullptr;
|
|
dbusmenu_server_set_root_func server_set_root = nullptr;
|
|
|
|
// Properties that we set on menu items:
|
|
const char kPropertyEnabled[] = "enabled";
|
|
const char kPropertyLabel[] = "label";
|
|
const char kPropertyShortcut[] = "shortcut";
|
|
const char kPropertyType[] = "type";
|
|
const char kPropertyToggleType[] = "toggle-type";
|
|
const char kPropertyToggleState[] = "toggle-state";
|
|
const char kPropertyVisible[] = "visible";
|
|
const char kPropertyChildrenDisplay[] = "children-display";
|
|
|
|
const char kToggleCheck[] = "checkmark";
|
|
const char kToggleRadio[] = "radio";
|
|
const char kTypeSeparator[] = "separator";
|
|
const char kDisplaySubmenu[] = "submenu";
|
|
|
|
void EnsureMethodsLoaded() {
|
|
static bool attempted_load = false;
|
|
if (attempted_load)
|
|
return;
|
|
attempted_load = true;
|
|
|
|
void* dbusmenu_lib = dlopen("libdbusmenu-glib.so", RTLD_LAZY);
|
|
if (!dbusmenu_lib)
|
|
dbusmenu_lib = dlopen("libdbusmenu-glib.so.4", RTLD_LAZY);
|
|
if (!dbusmenu_lib)
|
|
return;
|
|
|
|
// DbusmenuMenuItem methods.
|
|
menuitem_new = reinterpret_cast<dbusmenu_menuitem_new_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_new"));
|
|
menuitem_new_with_id = reinterpret_cast<dbusmenu_menuitem_new_with_id_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_new_with_id"));
|
|
menuitem_get_id = reinterpret_cast<dbusmenu_menuitem_get_id_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_get_id"));
|
|
menuitem_get_children = reinterpret_cast<dbusmenu_menuitem_get_children_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_get_children"));
|
|
menuitem_take_children =
|
|
reinterpret_cast<dbusmenu_menuitem_get_children_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_take_children"));
|
|
menuitem_child_append = reinterpret_cast<dbusmenu_menuitem_child_append_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_child_append"));
|
|
menuitem_property_set = reinterpret_cast<dbusmenu_menuitem_property_set_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_property_set"));
|
|
menuitem_property_set_variant =
|
|
reinterpret_cast<dbusmenu_menuitem_property_set_variant_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_property_set_variant"));
|
|
menuitem_property_set_bool =
|
|
reinterpret_cast<dbusmenu_menuitem_property_set_bool_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_property_set_bool"));
|
|
menuitem_property_set_int =
|
|
reinterpret_cast<dbusmenu_menuitem_property_set_int_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_menuitem_property_set_int"));
|
|
|
|
// DbusmenuServer methods.
|
|
server_new = reinterpret_cast<dbusmenu_server_new_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_server_new"));
|
|
server_set_root = reinterpret_cast<dbusmenu_server_set_root_func>(
|
|
dlsym(dbusmenu_lib, "dbusmenu_server_set_root"));
|
|
}
|
|
|
|
ElectronMenuModel* ModelForMenuItem(DbusmenuMenuitem* item) {
|
|
return reinterpret_cast<ElectronMenuModel*>(
|
|
g_object_get_data(G_OBJECT(item), "model"));
|
|
}
|
|
|
|
bool GetMenuItemID(DbusmenuMenuitem* item, int* id) {
|
|
gpointer id_ptr = g_object_get_data(G_OBJECT(item), "menu-id");
|
|
if (id_ptr != nullptr) {
|
|
*id = GPOINTER_TO_INT(id_ptr) - 1;
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
void SetMenuItemID(DbusmenuMenuitem* item, int id) {
|
|
DCHECK_GE(id, 0);
|
|
|
|
// Add 1 to the menu_id to avoid setting zero (null) to "menu-id".
|
|
g_object_set_data(G_OBJECT(item), "menu-id", GINT_TO_POINTER(id + 1));
|
|
}
|
|
|
|
std::string GetMenuModelStatus(ElectronMenuModel* model) {
|
|
std::string ret;
|
|
for (int i = 0; i < model->GetItemCount(); ++i) {
|
|
int status = model->GetTypeAt(i) | (model->IsVisibleAt(i) << 3) |
|
|
(model->IsEnabledAt(i) << 4) |
|
|
(model->IsItemCheckedAt(i) << 5);
|
|
ret += base::StringPrintf(
|
|
"%s-%X\n", base::UTF16ToUTF8(model->GetLabelAt(i)).c_str(), status);
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
} // namespace
|
|
|
|
GlobalMenuBarX11::GlobalMenuBarX11(NativeWindowViews* window)
|
|
: window_(window),
|
|
xwindow_(window_->GetNativeWindow()->GetHost()->GetAcceleratedWidget()) {
|
|
EnsureMethodsLoaded();
|
|
if (server_new)
|
|
InitServer(xwindow_);
|
|
|
|
GlobalMenuBarRegistrarX11::GetInstance()->OnWindowMapped(xwindow_);
|
|
}
|
|
|
|
GlobalMenuBarX11::~GlobalMenuBarX11() {
|
|
if (IsServerStarted())
|
|
g_object_unref(server_);
|
|
|
|
GlobalMenuBarRegistrarX11::GetInstance()->OnWindowUnmapped(xwindow_);
|
|
}
|
|
|
|
// static
|
|
std::string GlobalMenuBarX11::GetPathForWindow(x11::Window window) {
|
|
return base::StringPrintf("/com/canonical/menu/%X", window);
|
|
}
|
|
|
|
void GlobalMenuBarX11::SetMenu(ElectronMenuModel* menu_model) {
|
|
if (!IsServerStarted())
|
|
return;
|
|
|
|
DbusmenuMenuitem* root_item = menuitem_new();
|
|
menuitem_property_set(root_item, kPropertyLabel, "Root");
|
|
menuitem_property_set_bool(root_item, kPropertyVisible, true);
|
|
if (menu_model != nullptr) {
|
|
BuildMenuFromModel(menu_model, root_item);
|
|
}
|
|
|
|
server_set_root(server_, root_item);
|
|
g_object_unref(root_item);
|
|
}
|
|
|
|
bool GlobalMenuBarX11::IsServerStarted() const {
|
|
return server_;
|
|
}
|
|
|
|
void GlobalMenuBarX11::InitServer(x11::Window window) {
|
|
std::string path = GetPathForWindow(window);
|
|
server_ = server_new(path.c_str());
|
|
}
|
|
|
|
void GlobalMenuBarX11::OnWindowMapped() {
|
|
GlobalMenuBarRegistrarX11::GetInstance()->OnWindowMapped(xwindow_);
|
|
}
|
|
|
|
void GlobalMenuBarX11::OnWindowUnmapped() {
|
|
GlobalMenuBarRegistrarX11::GetInstance()->OnWindowUnmapped(xwindow_);
|
|
}
|
|
|
|
void GlobalMenuBarX11::BuildMenuFromModel(ElectronMenuModel* model,
|
|
DbusmenuMenuitem* parent) {
|
|
for (int i = 0; i < model->GetItemCount(); ++i) {
|
|
DbusmenuMenuitem* item = menuitem_new();
|
|
menuitem_property_set_bool(item, kPropertyVisible, model->IsVisibleAt(i));
|
|
|
|
ElectronMenuModel::ItemType type = model->GetTypeAt(i);
|
|
if (type == ElectronMenuModel::TYPE_SEPARATOR) {
|
|
menuitem_property_set(item, kPropertyType, kTypeSeparator);
|
|
} else {
|
|
std::string label = ui::ConvertAcceleratorsFromWindowsStyle(
|
|
base::UTF16ToUTF8(model->GetLabelAt(i)));
|
|
menuitem_property_set(item, kPropertyLabel, label.c_str());
|
|
menuitem_property_set_bool(item, kPropertyEnabled, model->IsEnabledAt(i));
|
|
|
|
g_object_set_data(G_OBJECT(item), "model", model);
|
|
SetMenuItemID(item, i);
|
|
|
|
if (type == ElectronMenuModel::TYPE_SUBMENU) {
|
|
menuitem_property_set(item, kPropertyChildrenDisplay, kDisplaySubmenu);
|
|
g_signal_connect(item, "about-to-show", G_CALLBACK(OnSubMenuShowThunk),
|
|
this);
|
|
} else {
|
|
ui::Accelerator accelerator;
|
|
if (model->GetAcceleratorAtWithParams(i, true, &accelerator))
|
|
RegisterAccelerator(item, accelerator);
|
|
|
|
g_signal_connect(item, "item-activated",
|
|
G_CALLBACK(OnItemActivatedThunk), this);
|
|
|
|
if (type == ElectronMenuModel::TYPE_CHECK ||
|
|
type == ElectronMenuModel::TYPE_RADIO) {
|
|
menuitem_property_set(item, kPropertyToggleType,
|
|
type == ElectronMenuModel::TYPE_CHECK
|
|
? kToggleCheck
|
|
: kToggleRadio);
|
|
menuitem_property_set_int(item, kPropertyToggleState,
|
|
model->IsItemCheckedAt(i));
|
|
}
|
|
}
|
|
}
|
|
|
|
menuitem_child_append(parent, item);
|
|
g_object_unref(item);
|
|
}
|
|
}
|
|
|
|
void GlobalMenuBarX11::RegisterAccelerator(DbusmenuMenuitem* item,
|
|
const ui::Accelerator& accelerator) {
|
|
// A translation of libdbusmenu-gtk's menuitem_property_set_shortcut()
|
|
// translated from GDK types to ui::Accelerator types.
|
|
GVariantBuilder builder;
|
|
g_variant_builder_init(&builder, G_VARIANT_TYPE_ARRAY);
|
|
|
|
if (accelerator.IsCtrlDown())
|
|
g_variant_builder_add(&builder, "s", "Control");
|
|
if (accelerator.IsAltDown())
|
|
g_variant_builder_add(&builder, "s", "Alt");
|
|
if (accelerator.IsShiftDown())
|
|
g_variant_builder_add(&builder, "s", "Shift");
|
|
|
|
char* name =
|
|
XKeysymToString(XKeysymForWindowsKeyCode(accelerator.key_code(), false));
|
|
if (!name) {
|
|
NOTIMPLEMENTED();
|
|
return;
|
|
}
|
|
g_variant_builder_add(&builder, "s", name);
|
|
|
|
GVariant* inside_array = g_variant_builder_end(&builder);
|
|
g_variant_builder_init(&builder, G_VARIANT_TYPE_ARRAY);
|
|
g_variant_builder_add_value(&builder, inside_array);
|
|
GVariant* outside_array = g_variant_builder_end(&builder);
|
|
|
|
menuitem_property_set_variant(item, kPropertyShortcut, outside_array);
|
|
}
|
|
|
|
void GlobalMenuBarX11::OnItemActivated(DbusmenuMenuitem* item,
|
|
unsigned int timestamp) {
|
|
int id;
|
|
ElectronMenuModel* model = ModelForMenuItem(item);
|
|
if (model && GetMenuItemID(item, &id))
|
|
model->ActivatedAt(id, 0);
|
|
}
|
|
|
|
void GlobalMenuBarX11::OnSubMenuShow(DbusmenuMenuitem* item) {
|
|
int id;
|
|
ElectronMenuModel* model = ModelForMenuItem(item);
|
|
if (!model || !GetMenuItemID(item, &id))
|
|
return;
|
|
|
|
// Do not update menu if the submenu has not been changed.
|
|
std::string status = GetMenuModelStatus(model);
|
|
char* old = static_cast<char*>(g_object_get_data(G_OBJECT(item), "status"));
|
|
if (old && status == old)
|
|
return;
|
|
|
|
// Save the new status.
|
|
g_object_set_data_full(G_OBJECT(item), "status", g_strdup(status.c_str()),
|
|
g_free);
|
|
|
|
// Clear children.
|
|
GList* children = menuitem_take_children(item);
|
|
g_list_foreach(children, reinterpret_cast<GFunc>(g_object_unref), nullptr);
|
|
g_list_free(children);
|
|
|
|
// Build children.
|
|
BuildMenuFromModel(model->GetSubmenuModelAt(id), item);
|
|
}
|
|
|
|
} // namespace electron
|