如何使用 Python 获取任务栏大小和位置

Chr*_*lli 5 python taskbar python3 display-resolution

我正在用Python开发一个模块,我想找到一种方法来获得显示器分辨率减去任务栏,而无需任何用户交互,并且最好以不干扰可视界面的方式。如果我可以获取任务栏的大小和位置,我可以轻松地从显示器分辨率中减去它。但是,我似乎找不到办法。

在 Windows 中,我可以使用该win32api模块并用于GetMonitorInfo获取显示器分辨率。然而,我还没有找到在 Ubuntu 或一般 Linux 中执行此操作的方法。如果您也有办法在 macOS 中执行此操作,那么也很高兴知道,但这对于这个问题来说不是必需的。

我希望支持尽可能多的桌面环境,但我意识到这样做是多么困难,因为每个桌面环境可能都有自己的解决方案。考虑到这一点,我更愿意从这些桌面环境开始:

  • 凯德
  • XFCE
  • 侏儒
  • 肉桂

如果这需要一些外部程序来获取它,我更愿意使用桌面环境或操作系统附带的东西。

Ser*_*nyy 5

我想找到一种方法来获得显示器分辨率减去任务栏,而无需任何用户交互,并且最好以不干扰视觉界面的方式。

这就是人们提出的问题,我将根据我的发现在整个答案中重点关注这一点。获取任务栏大小和位置是一个原始的想法,但正如评论中所讨论的,如果不知道大量信息,这将很难实现 - 哪个面板或扩展坞正在运行,它们是否有 _NET_WM_STRUT属性集,它们是否通过以下方式公开该信息GSettings数据库或其他方法。我确信这不是不可能,但相当困难。

NET_WORKAREA 属性

根据根窗口的规范,_NET_WORKAREA 定义如下:

_NET_WORKAREA、x、y、宽度、高度 CARDINAL/32

该属性必须由窗口管理器在计算每个桌面的工作区域时设置。包含每个桌面的几何图形。这些几何图形是相对于每个桌面上的视口指定的,并指定完全包含在视口内的区域。桌面应用程序应该使用工作区来适当地放置桌面图标。

窗口管理器应该通过当前页面减去停靠窗口和面板窗口占用的空间来计算该空间,如客户端窗口上设置的 _NET_WM_STRUT 或 _NET_WM_STRUT_PARTIAL 属性所示。

有几点需要注意:

  • 符合这些规范的窗口管理器必须设置此属性,因此在 XFCE、Budgie、Unity、GNOME、KDE ​​等桌面环境中(即任何不使用任何实验性窗口管理器的官方 Ubuntu 版本)将设置此属性; 换句话说,这将适用于支持的桌面
  • 这会计算可用的宽度和高度,还会计算相对于整个显示屏最左上角的偏移量(在双屏的情况下,最左侧屏幕的左上角将是起点)
  • 或是由停靠点和面板设置的属性_NET_WM_STRUT_NET_WM_STRUT_PARTIAL这意味着,如果面板或扩展坞没有设置它,它们基本上会表现得像一个普通窗口,只是位于所有其他窗口之上(并且可能使用户感到恼火1)。也就是说,这对我们来说已经足够好了。我已经在 Ubuntu 18.04 上的 Cinnamon、LXQt 和 XFCE 上测试了此属性。在所有情况下,坞站和面板都会正确地向窗口管理器通知其支柱区域。这意味着,_NET_WORKAREA在 99% 的情况下都适合您。

xprop可以通过根窗口上的实用程序在 shell 中查询此属性。

$ xprop -root _NET_WORKAREA
_NET_WORKAREA(CARDINAL) = 0, 32, 3120, 974, 0, 32, 3120, 974
Run Code Online (Sandbox Code Playgroud)

这是我的双显示器设置的输出,两个显示器沿顶部对齐,一个底座和顶部面板。如果您注意到只有前四个值很重要,而其他四个值重复。特别令人感兴趣的是3120974。这就是我们感兴趣的值。前两个告诉我们桌面最左边的角从哪里开始我的顶部面板占据 32 点,因此桌面最左边的角一直从左侧开始,向下偏移 32 点。与底部的偏移通常对于窗口来说并不重要 - 它们会填充可用的工作区域。

如果我将扩展坞放在主显示器的右侧,您会注意到宽度减少了,但高度增加了:

$ xprop -root _NET_WORKAREA
_NET_WORKAREA(CARDINAL) = 0, 32, 3059, 1018, 0, 32, 3059, 1018
Run Code Online (Sandbox Code Playgroud)

身高的变化是44点。再添加 32(顶部面板窗口的大小),您将得到 1050 正好是我的桌面的几何形状(注意之前没有停靠栏时的 3120,因此使用了完整的桌面宽度):

$ xprop -root _NET_DESKTOP_GEOMETRY
_NET_DESKTOP_GEOMETRY(CARDINAL) = 3120, 1050
Run Code Online (Sandbox Code Playgroud)

Python 和xprop

上面的命令xprop随大多数桌面一起提供,并且属于x11-utils软件包,因此如果您的目标是基于主要 Debian 的系统,则大多数情况下您应该可以使用它。我要做的是使用Python的subprocess,并利用subprocess.check_output()or命令并通过模块subprocess.run()解析输出。re

>>> import re,subprocess
>>> out = subprocess.check_output(['xprop','-root','_NET_WORKAREA'])
>>> workarea_tokens = re.split('=|,',out.decode())
>>> workarea_tokens[1:5]
[' 0', ' 32', ' 3059', ' 1018']
Run Code Online (Sandbox Code Playgroud)

Python 和 Gdk

另一种选择(通常是我的首选方式)是使用与其他 GNOME 项目库密切相关的 Gdk 库。如果您正在使用 Gtk 开发桌面应用程序,那么您已经拥有必要的工具,那么为什么不使用它们呢?棘手的部分是您必须计算每个单独显示器的工作区域。

$ xprop -root _NET_WORKAREA
_NET_WORKAREA(CARDINAL) = 0, 32, 3120, 974, 0, 32, 3120, 974
Run Code Online (Sandbox Code Playgroud)

该程序的结果:

$ python3  display_area.py
1440 32 1619 1018
0 0 1440 900
# width of two monitors with dock on right adds up to _NET_WORKAREA
$ echo $((1619+1440))
3059
$ xprop -root _NET_WORKAREA
_NET_WORKAREA(CARDINAL) = 0, 32, 3059, 1018, 0, 32, 3059, 1018
Run Code Online (Sandbox Code Playgroud)

问题

现在,还记得我提到过我有双显示器吗?这样的设置开启了很多可能性,但也揭示了某些奇怪的行为。

# added xfce4-panel at the bottom of second screen, no change in height - 900
$ python3 display_area.py ; xprop -root _NET_WORKAREA
1440 32 1619 1018
0 0 1440 900
_NET_WORKAREA(CARDINAL) = 526, 32, 2533, 1018, 526, 32, 2533, 1018
# stopped xfce4-panel process, same height - 900
$ python3 display_area.py ; xprop -root _NET_WORKAREA
1440 32 1619 1018
0 0 1440 900
_NET_WORKAREA(CARDINAL) = 0, 32, 3059, 1018, 0, 32, 3059, 1018
# added xfce4-panel to the top of second monitor
$ python3 display_area.py ; xprop -root _NET_WORKAREA
1440 49 1619 1001
0 0 1440 900
_NET_WORKAREA(CARDINAL) = 0, 49, 3059, 1001, 0, 49, 3059, 1001
Run Code Online (Sandbox Code Playgroud)

xfce4-panel请记住,我在一个屏幕上有一个面板和一个底座,并在另一个屏幕上添加了测试。Gdk 的错误结果是第二个监视器工作区域没有改变。xprop还报告了错误的数字 - 在第一个输出526点中,从左侧偏移的位置根本不正确。然而,在最后的输出中,垂直偏移增加了,这是一件好事 - 添加的额外宽度xfce4-panel应该添加到总数中,而且可能有帮助的是我的两个显示器在顶部对齐。

换句话说,如果您的应用程序计划确定没有坞站和面板的区域,这些方法最适合单显示器设置。在双显示器设置中 - 结果可能会不一致。请注意,我正在使用 Metacity 窗口管理器,错误可能归因于此。然而,我目前没有办法用双显示器设置来测试其他 WM,所以我将把这部分保留为不确定。

待办事项(有一天)

  • C 或 Python 中的 libx11 示例

  • 使用替代窗口管理器进行双显示器测试

  • 查找底座和面板位置


脚注

1功能未记录,取决于用户的耐心。