diff --git a/qui/clipboard.py b/qui/clipboard.py
index 0118bf94..fa4904ef 100644
--- a/qui/clipboard.py
+++ b/qui/clipboard.py
@@ -26,6 +26,11 @@
via Qubes RPC """
# pylint: disable=invalid-name,wrong-import-position
+# Must be imported before creating threads
+from .tray.gross_gtk3_bug_workaround import (
+ DisgustingX11FullscreenWindowHack,
+) # isort:skip
+
import asyncio
import contextlib
import json
@@ -285,6 +290,7 @@ def __init__(self, wm, qapp, dispatcher, **properties):
self.set_application_id("org.qubes.qui.clipboard")
self.register() # register Gtk Application
+ self.disgusting_hack = DisgustingX11FullscreenWindowHack.get()
self.qapp = qapp
self.vm = self.qapp.domains[self.qapp.local_name]
self.dispatcher = dispatcher
@@ -373,6 +379,7 @@ def setup_ui(self, *_args, **_kwargs):
)
self.menu = Gtk.Menu()
+ self.disgusting_hack.show_for_widget(self.menu)
title_label = Gtk.Label(xalign=0)
title_label.set_markup(_("Current clipboard"))
diff --git a/qui/devices/device_widget.py b/qui/devices/device_widget.py
index 942ae298..4bd3716c 100644
--- a/qui/devices/device_widget.py
+++ b/qui/devices/device_widget.py
@@ -17,6 +17,12 @@
#
# You should have received a copy of the GNU Lesser General Public License along
# with this program; if not, see .
+
+# Must be imported before creating threads
+from ..tray.gross_gtk3_bug_workaround import (
+ DisgustingX11FullscreenWindowHack,
+) # isort:skip
+
from typing import Set, List, Dict
import asyncio
import sys
@@ -73,6 +79,7 @@ class DevicesTray(Gtk.Application):
"""Tray application for handling devices."""
def __init__(self, app_name, qapp, dispatcher):
super().__init__()
+ self.disgusting_hack = DisgustingX11FullscreenWindowHack.get()
self.name: str = app_name
# maps: port to connected device (e.g., sys-usb:sda -> block device)
@@ -300,6 +307,7 @@ def load_css(widget) -> str:
def show_menu(self, _unused, _event):
"""Show menu at mouse pointer."""
tray_menu = Gtk.Menu()
+ self.disgusting_hack.show_for_widget(tray_menu)
theme = self.load_css(tray_menu)
tray_menu.set_reserve_toggle_size(False)
diff --git a/qui/tray/disk_space.py b/qui/tray/disk_space.py
index 8d8c8094..a2020639 100644
--- a/qui/tray/disk_space.py
+++ b/qui/tray/disk_space.py
@@ -1,4 +1,10 @@
# pylint: disable=wrong-import-position,import-error
+
+# Must be imported before creating threads
+from .gross_gtk3_bug_workaround import (
+ DisgustingX11FullscreenWindowHack,
+) # isort:skip
+
import sys
import subprocess
from typing import List
@@ -338,6 +344,7 @@ class DiskSpace(Gtk.Application):
def __init__(self, **properties):
super().__init__(**properties)
+ self.disgusting_hack = DisgustingX11FullscreenWindowHack.get()
self.pool_warned = False
self.vms_warned = set()
@@ -423,6 +430,7 @@ def make_menu(self, _unused, _event):
vm_data = VMUsageData(self.qubes_app)
menu = Gtk.Menu()
+ self.disgusting_hack.show_for_widget(menu)
menu.append(self.make_top_box(pool_data))
diff --git a/qui/tray/domains.py b/qui/tray/domains.py
index 1eae1310..e7251dc5 100644
--- a/qui/tray/domains.py
+++ b/qui/tray/domains.py
@@ -2,6 +2,12 @@
# -*- coding: utf-8 -*-
# pylint: disable=wrong-import-position,import-error,superfluous-parens
''' A menu listing domains '''
+
+# Must be imported before creating threads
+from .gross_gtk3_bug_workaround import (
+ DisgustingX11FullscreenWindowHack,
+) # isort:skip
+
import abc
import asyncio
import subprocess
@@ -537,6 +543,8 @@ def __init__(self, app_name, qapp, dispatcher, stats_dispatcher):
_('Qubes Domains\nView and manage running domains.'))
self.tray_menu = Gtk.Menu()
+ self.disgusting_hack = DisgustingX11FullscreenWindowHack.get()
+ self.disgusting_hack.show_for_widget(self.tray_menu)
self.icon_cache = IconCache()
diff --git a/qui/tray/gross_gtk3_bug_workaround.py b/qui/tray/gross_gtk3_bug_workaround.py
new file mode 100644
index 00000000..2a6a7d2b
--- /dev/null
+++ b/qui/tray/gross_gtk3_bug_workaround.py
@@ -0,0 +1,148 @@
+#!/usr/bin/python3 --
+import os
+import sys
+from typing import Optional
+
+# Modifying the environment while multiple threads
+# are running leads to use-after-free in glibc, so
+# ensure that only one thread is running.
+if len(os.listdir("/proc/self/task")) != 1:
+ raise RuntimeError("threads already running")
+# If gi.override.Gdk has been imported, the GDK
+# backend has already been set and it is too late
+# to override it.
+if "gi.override.Gdk" in sys.modules:
+ raise RuntimeError("must import this module before loading GDK")
+# Only the X11 backend is supported
+os.environ["GDK_BACKEND"] = "x11"
+
+import gi
+
+gi.require_version("Gdk", "3.0")
+gi.require_version("Gtk", "3.0")
+from gi.repository import Gtk, Gdk
+
+
+is_xwayland = "WAYLAND_DISPLAY" in os.environ
+
+
+class DisgustingX11FullscreenWindowHack:
+ """
+ No-op implementation of the hack, for use on stock X11.
+ """
+
+ __slots__ = ()
+
+ @classmethod
+ def get(cls, /) -> "DisgustingX11FullscreenWindowHack":
+ if is_xwayland:
+ return DisgustingX11FullscreenWindowHackXWayland()
+ return cls()
+
+ def clear_widget(self, /) -> None:
+ pass
+
+ def show_for_widget(self, _widget: Gtk.Widget, /) -> None:
+ pass
+
+
+class DisgustingX11FullscreenWindowHackXWayland(
+ DisgustingX11FullscreenWindowHack
+):
+ """
+ GTK3 menus have a horrible bug under Xwayland: if the user clicks on a
+ native Wayland surface, the menu is not dismissed. This class works around
+ the problem by using evil X11 hacks, such as a fullscreen override-redirect
+ window that is made transparent.
+ """
+
+ _window: Gtk.Window
+ _widget: Optional[Gtk.Widget]
+ _unmap_signal_id: int
+ _map_signal_id: int
+ __slots__ = ("_window", "_widget", "_unmap_signal_id", "_map_signal_id")
+
+ def __init__(self) -> None:
+ self._widget = None
+ # Get the default GDK screen.
+ screen = Gdk.Screen.get_default()
+ # This is deprecated, but it gets the total width and height
+ # of all screens, which is what we want. It will go away in
+ # GTK4, but this code will never be ported to GTK4.
+ width = screen.get_width()
+ height = screen.get_height()
+ # Create a window that will fill the screen.
+ window = self._window = Gtk.Window()
+ # Move that window to the top left.
+ # pylint: disable=no-member
+ window.move(0, 0)
+ # Make the window fill the whole screen.
+ # pylint: disable=no-member
+ window.resize(width, height)
+ # Request that the window not be decorated by the window manager.
+ window.set_decorated(False)
+ # Connect a signal so that the window and menu can be
+ # unmapped (no longer shown on screen) once clicked.
+ window.connect("button-press-event", self.on_button_press)
+ # When the window is created, mark it as override-redirect
+ # (invisible to the window manager) and transparent.
+ window.connect("realize", self._on_realize)
+ self._unmap_signal_id = self._map_signal_id = 0
+
+ def clear_widget(self, /) -> None:
+ """
+ Clears the connected widget. Automatically called by
+ show_for_widget().
+ """
+ widget = self._widget
+ map_signal_id = self._map_signal_id
+ unmap_signal_id = self._unmap_signal_id
+
+ # Double-disconnect is C-level undefined behavior, so ensure
+ # it cannot happen. It is better to leak memory if an exception
+ # is thrown here. GObject.disconnect_by_func() is buggy
+ # (https://gitlab.gnome.org/GNOME/pygobject/-/issues/106),
+ # so avoid it.
+ if widget is not None:
+ if map_signal_id != 0:
+ # Clear the signal ID to avoid double-disconnect
+ # if this method is interrupted and then called again.
+ self._map_signal_id = 0
+ widget.disconnect(map_signal_id)
+ if unmap_signal_id != 0:
+ # Clear the signal ID to avoid double-disconnect
+ # if this method is interrupted and then called again.
+ self._unmap_signal_id = 0
+ widget.disconnect(unmap_signal_id)
+ self._widget = None
+
+ def show_for_widget(self, widget: Gtk.Widget, /) -> None:
+ # Clear any existing connections.
+ self.clear_widget()
+ # Store the new widget.
+ self._widget = widget
+ # Connect map and unmap signals.
+ self._unmap_signal_id = widget.connect("unmap", self._hide)
+ self._map_signal_id = widget.connect("map", self._show)
+
+ @staticmethod
+ def _on_realize(window: Gtk.Window, /) -> None:
+ window.set_opacity(0)
+ window.get_window().set_override_redirect(True)
+
+ def _show(self, widget: Gtk.Widget, /) -> None:
+ assert widget is self._widget, "signal not properly disconnected"
+ # pylint: disable=no-member
+ self._window.show_all()
+
+ def _hide(self, widget: Gtk.Widget, /) -> None:
+ assert widget is self._widget, "signal not properly disconnected"
+ self._window.hide()
+
+ # pylint: disable=line-too-long
+ def on_button_press(
+ self, window: Gtk.Window, _event: Gdk.EventButton, /
+ ) -> None:
+ # Hide the window and the widget.
+ window.hide()
+ self._widget.hide()
diff --git a/qui/tray/updates.py b/qui/tray/updates.py
index 8533dea3..0a1c6b5f 100644
--- a/qui/tray/updates.py
+++ b/qui/tray/updates.py
@@ -3,6 +3,12 @@
# pylint: disable=wrong-import-position,import-error
''' A widget that monitors update availability and notifies the user
about new updates to templates and standalone VMs'''
+
+# Must be imported before creating threads
+from .gross_gtk3_bug_workaround import (
+ DisgustingX11FullscreenWindowHack,
+) # isort:skip
+
import asyncio
import sys
import subprocess
@@ -59,6 +65,7 @@ def __init__(self, app_name, qapp, dispatcher):
super().__init__()
self.name = app_name
+ self.disgusting_hack = DisgustingX11FullscreenWindowHack.get()
self.dispatcher = dispatcher
self.qapp = qapp
@@ -76,6 +83,7 @@ def __init__(self, app_name, qapp, dispatcher):
self.obsolete_vms = set()
self.tray_menu = Gtk.Menu()
+ self.disgusting_hack.show_for_widget(self.tray_menu)
def run(self): # pylint: disable=arguments-differ
self.check_vms_needing_update()
@@ -107,6 +115,8 @@ def setup_menu(self):
def show_menu(self, _unused, _event):
self.tray_menu = Gtk.Menu()
+ # TODO: disconnect
+ self.disgusting_hack.show_for_widget(self.tray_menu)
self.setup_menu()
diff --git a/rpm_spec/qubes-desktop-linux-manager.spec.in b/rpm_spec/qubes-desktop-linux-manager.spec.in
index 781f09b8..c0c48aa4 100644
--- a/rpm_spec/qubes-desktop-linux-manager.spec.in
+++ b/rpm_spec/qubes-desktop-linux-manager.spec.in
@@ -137,7 +137,6 @@ gtk-update-icon-cache %{_datadir}/icons/Adwaita &>/dev/null || :
%{python3_sitelib}/qui/devices/actionable_widgets.py
%{python3_sitelib}/qui/devices/backend.py
%{python3_sitelib}/qui/devices/device_widget.py
-%{python3_sitelib}/qui/devices/device_widget.py
%{python3_sitelib}/qui/qubes-devices-dark.css
%{python3_sitelib}/qui/qubes-devices-light.css
%{python3_sitelib}/qui/devices/AttachConfirmationWindow.glade
@@ -155,6 +154,7 @@ gtk-update-icon-cache %{_datadir}/icons/Adwaita &>/dev/null || :
%{python3_sitelib}/qui/tray/domains.py
%{python3_sitelib}/qui/tray/disk_space.py
%{python3_sitelib}/qui/tray/updates.py
+%{python3_sitelib}/qui/tray/gross_gtk3_bug_workaround.py
%dir %{python3_sitelib}/qubes_config
%dir %{python3_sitelib}/qubes_config/__pycache__