如何遍历当前所有打开的 Gtk.Window 实例

发布时间 - 2026-01-03 00:00:00    点击率:

在 python 2.7 + gtk 环境下(如 gwyddion 插件开发),若需对所有已创建但尚未显示完全的 `gtk.window` 子类实例批量截图,可借助 gtk 自身的 `gtk.window.list_toplevels()` 方法获取窗口列表,并配合 `gdk.window.draw_to_image()` 实现可靠截图。

GTK 提供了原生、轻量且无需额外依赖的机制来枚举所有顶层窗口:Gtk.Window.list_toplevels()。该方法返回一个 list,包含当前进程中所有已创建(show() 已调用)的 Gtk.Window 及其子类实例——这正是你在 Gwyddion 中动态弹出的结果窗口所需的目标集合。

import gtk
import gdk

def capture_all_windows(prefix="screenshot"):
    """对所有已显示的 Gtk.Window 实例截图并保存"""
    windows = gtk.Window.list_toplevels()

    # 注意:list_toplevels() 返回顺序与窗口创建/显示顺序基本一致(LIFO 栈式,后创建者靠前)
    # 若需按创建时间排序,建议在窗口实例化时打上时间戳并自行管理
    for i, win in enumerate(windows):
        # 确保窗口已映射(即真正可见),避免白屏
        if not win.get_mapped():
            win.show_now()  # 强制立即映射,触发绘制
            while gtk.events_pending():
                gtk.main_iteration()

        # 获取底层 Gdk.Window 并截图
        gdk_win = win.get_window()
        if gdk_win is None:
            continue

        x, y, width, height = gdk_win.get_geometry()
        try:
            # 截取整个窗口客户区(不含边框)
            pixbuf = gdk.pixbuf_get_from_drawable(
                None, gdk_win, gdk.colormap_get_system(),
                0, 0, 0, 0, width, height
            )
            filename = "%s_%03d.png" % (prefix, i + 1)
            pixbuf.save(filename, "png")
            print("Saved: %s" % filename)
        except Exception as e:
            print("Failed to capture %s: %s" % (win.get_title() or "unnamed", str(e)))

# 调用示例(在你的分析循环结束后执行)
# capture_all_windows("afm_channel_2")

⚠️ 关键注意事项

  • list_toplevels() 返回的是 Python 对象引用,不是 C 层指针,因此无需手动 g_object_ref/unref —— 这是 C API 文档中针对原生 GObject 内存管理的说明,在 PyGTK(Python 2.7 绑定)中由 Python 垃圾回收自动处理;
  • 窗口“未绘制”问题本质是 GTK 的事件循环未及时刷新:Gwyddion 不调用 gtk.main() 是因其自身主循环接管了 GTK 事件调度,但你仍可通过 gtk.events_pending() + gtk.main_iteration() 主动泵送事件,确保窗口完成映射与首次绘制;
  • list_toplevels() 的遍历顺序为 最近创建/显示的窗口排在前面(内部使用链表头插法),因此索引 i 大致反映窗口弹出时序;如需严格顺序,请在创建每个窗口时将其追加到全局 list 并记录时间戳;
  • 若截图仍为空白,请检查窗口是否设置了 set_app_paintable(False) 或启用了透明背景,此时应改用 gtk.gdk.get_default_root_window() 全屏捕获并裁剪坐标区域。

综上,不依赖 wnck 等外部库,仅用 PyGTK 原生接口即可稳健解决多窗口批量截图问题,兼顾 Gwyddion 的受限运行环境与 Python 2.7 兼容性要求。


# python  # go  # windows  # app  #   # ai  # win 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: Windows10电脑怎么设置虚拟光驱_Win10右键装载ISO镜像文件  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  Bootstrap CSS布局之列表  原生JS获取元素集合的子元素宽度实例  零服务器AI建站解决方案:快速部署与云端平台低成本实践  Laravel如何自定义错误页面(404, 500)?(代码示例)  Laravel如何记录自定义日志?(Log频道配置)  PHP 500报错的快速解决方法  Laravel如何从数据库删除数据_Laravel destroy和delete方法区别  Laravel如何实现多表关联模型定义_Laravel多对多关系及中间表数据存取【方法】  香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南  弹幕视频网站制作教程下载,弹幕视频网站是什么意思?  Android使用GridView实现日历的简单功能  制作电商网页,电商供应链怎么做?  nginx修改上传文件大小限制的方法  浅谈redis在项目中的应用  Python进程池调度策略_任务分发说明【指导】  微信小程序 require机制详解及实例代码  关于BootStrap modal 在IOS9中不能弹出的解决方法(IOS 9 bootstrap modal ios 9 noticework)  Firefox Developer Edition开发者版本入口  Laravel如何配置Horizon来管理队列?(安装和使用)  Laravel如何配置和使用队列处理异步任务_Laravel队列驱动与任务分发实例  合肥制作网站的公司有哪些,合肥聚美网络科技有限公司介绍?  Win11怎么更改系统语言为中文_Windows11安装语言包并设为显示语言  手机软键盘弹出时影响布局的解决方法  Laravel事件和监听器如何实现_Laravel Events & Listeners解耦应用的实战教程  Android自定义控件实现温度旋转按钮效果  悟空浏览器如何设置小说背景色_悟空浏览器背景色设置【方法】  如何选择可靠的免备案建站服务器?  Laravel的.env文件有什么用_Laravel环境变量配置与管理详解  如何在 Go 中优雅地映射具有动态字段的 JSON 对象到结构体  Laravel storage目录权限问题_Laravel文件写入权限设置  长沙做网站要多少钱,长沙国安网络怎么样?  Win11怎么设置虚拟桌面 Win11新建多桌面切换操作【技巧】  Win10如何卸载预装Edge扩展_Win10卸载Edge扩展教程【方法】  简单实现Android文件上传  Windows Hello人脸识别突然无法使用  Laravel Fortify是什么,和Jetstream有什么关系  教学论文网站制作软件有哪些,写论文用什么软件 ?  如何用IIS7快速搭建并优化网站站点?  Laravel如何创建和注册中间件_Laravel中间件编写与应用流程  香港服务器WordPress建站指南:SEO优化与高效部署策略  微博html5版本怎么弄发超话_超话进入入口及发帖格式要求【教程】  Laravel如何优雅地处理服务层_在Laravel中使用Service层和Repository层  北京专业网站制作设计师招聘,北京白云观官方网站?  图片制作网站免费软件,有没有免费的网站或软件可以将图片批量转为A4大小的pdf?  Laravel安装步骤详细教程_Laravel环境搭建指南  如何用花生壳三步快速搭建专属网站?  ai格式如何转html_将AI设计稿转换为HTML页面流程【页面】  Linux虚拟化技术教程_KVMQEMU虚拟机安装与调优