0

0

PySide6 中 QWidget 动态绘制与视频录制教程

碧海醫心

碧海醫心

发布时间:2025-10-24 13:48:14

|

406人浏览过

|

来源于php中文网

原创

PySide6 中 QWidget 动态绘制与视频录制教程

本教程详细介绍了如何在 pyside6 应用中实现 qwidget 的动态内容绘制,并同时将这些动态帧捕获并保存为视频文件。文章将指导读者正确使用 qpainter 进行界面绘制,并通过 qwidget 的 `grab()` 方法结合 `imageio` 库高效地将实时画面转换为视频帧,避免常见的绘制上下文错误,确保流畅的显示与录制。

引言:PySide6 动态绘制与视频生成的需求

在许多图形界面应用中,我们可能需要在一个 QWidget 上实时显示动态内容(例如动画、数据可视化),并同时将这些动态变化的过程录制成视频或 GIF。直接在 paintEvent 中尝试将内容绘制到 QImage 上,再将 QImage 渲染回 QWidget,往往会导致 QPainter 上下文冲突或 QWidget::render 调用错误。本文将提供一个健壮的解决方案,利用 QPainter 进行高效的界面绘制,并通过 QWidget.grab() 结合 imageio 库实现无缝的视频帧捕获与生成。

核心概念与技术

要实现上述目标,我们需要掌握以下核心技术和库:

  1. PySide6 (Qt for Python): 用于构建图形用户界面。
    • QWidget: 基础的用户界面组件。
    • QPainter: 用于在绘制设备(如 QWidget、QPixmap、QImage)上进行低级绘制。
    • QTimer: 用于定时触发更新,实现动画效果。
    • QPixmap, QImage: 用于处理图像数据。
  2. imageio: 一个强大的 Python 库,用于读取和写入各种图像和视频文件格式。
    • 需要安装 imageio 和 imageio[ffmpeg](用于支持 FFmpeg 编解码器,以便生成常见的视频格式如 AVI, MP4)。
  3. NumPy: 用于高效地处理图像数据,将 QImage 转换为 imageio 可接受的 NumPy 数组格式。

环境准备

在开始之前,请确保您的 Python 环境中已安装必要的库:

pip install PySide6 imageio numpy
pip install imageio[ffmpeg] # 确保视频编码功能可用

实现动态绘制与帧捕获

我们将创建一个自定义的 PlotWidget 类,它继承自 QWidget。这个组件将负责:

  1. 定时更新: 使用 QTimer 定期触发绘制和帧捕获。
  2. 界面绘制: 在 paintEvent 中使用 QPainter 绘制动态内容。
  3. 帧捕获与视频生成: 在定时器触发的方法中,捕获当前 QWidget 的内容,并将其追加到 imageio 视频写入器中。

1. PlotWidget 初始化

在 PlotWidget 的构造函数中,我们将设置定时器、初始化视频写入器,并定义窗口大小。

import imageio, numpy as np
from PySide6.QtWidgets import QApplication, QWidget
from PySide6.QtCore import QPoint, QRect, QTimer, Qt
from PySide6.QtGui import QPainter, QPointList, QImage

WIDTH = 720
HEIGHT = 720

class PlotWidget(QWidget):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.setWindowTitle("PySide6 动态绘制与视频录制")
        self.setFixedSize(WIDTH, HEIGHT) # 固定窗口大小

        self._timer = QTimer(self)
        self._timer.setInterval(100) # 每100毫秒触发一次,即10帧/秒
        self._timer.timeout.connect(self.frame)

        self._points = QPointList() # 示例数据,用于绘制

        self._totalFrames = 100 # 录制100帧后停止
        # 初始化 imageio 视频写入器,指定输出文件名和帧率
        self._vid_writer = imageio.get_writer('output_video.avi', fps=10) 

        self._timer.start() # 启动定时器

2. paintEvent 实现

paintEvent 负责在 QWidget 上进行绘制。关键在于,QPainter 应该直接作用于 self (即 QWidget 实例),而不是一个临时的 QImage。

PHP 网络编程技术与实例(曹衍龙)
PHP 网络编程技术与实例(曹衍龙)

PHP网络编程技术详解由浅入深,全面、系统地介绍了PHP开发技术,并提供了大量实例,供读者实战演练。另外,笔者专门为本书录制了相应的配套教学视频,以帮助读者更好地学习本书内容。这些视频和书中的实例源代码一起收录于配书光盘中。本书共分4篇。第1篇是PHP准备篇,介绍了PHP的优势、开发环境及安装;第2篇是PHP基础篇,介绍了PHP中的常量与变量、运算符与表达式、流程控制以及函数;第3篇是进阶篇,介绍

下载
    def paintEvent(self, event):
        with QPainter(self) as painter: # QPainter 直接作用于当前 QWidget
            rect = QRect(QPoint(0, 0), self.size())
            painter.fillRect(rect, Qt.white) # 填充背景
            painter.setPen(Qt.red) # 设置画笔颜色
            painter.drawPoints(self._points) # 绘制示例点

3. frame 方法:动画逻辑与帧捕获

frame 方法由 QTimer 定时调用。它负责更新绘制数据、触发 paintEvent、捕获当前 QWidget 的画面,并将其追加到视频文件中。

    def frame(self):
        # 示例:更新绘制数据,这里只是简单地清空并添加一个点
        self._points.clear()
        self._points.append(QPoint(np.random.randint(0, WIDTH), np.random.randint(0, HEIGHT)))

        # 如果还有帧需要录制
        if self._totalFrames > 0:
            self.update() # 触发 paintEvent,更新界面显示

            # 捕获 QWidget 的当前内容
            pixmap = self.grab() 
            # 将 QPixmap 转换为 QImage,并确保格式为 RGB888,便于 NumPy 处理
            qimg = pixmap.toImage().convertToFormat(QImage.Format_RGB888)

            # 将 QImage 的像素数据转换为 NumPy 数组
            # 注意:这里直接访问 QImage 的底层数据,效率高
            # strides 参数是关键,确保 NumPy 正确解析内存布局
            array = np.ndarray((qimg.height(), qimg.width(), 3), 
                               buffer=qimg.constBits(), 
                               strides=[qimg.bytesPerLine(), 3, 1], 
                               dtype=np.uint8)

            # 如果视频写入器未关闭,则追加帧
            if not self._vid_writer.closed:
                self._vid_writer.append_data(array)
        else:
            # 录制完成后,停止定时器并关闭视频写入器
            self._timer.stop()
            if not self._vid_writer.closed:
                self._vid_writer.close()
            print("视频录制完成!")

        self._totalFrames -= 1 # 减少剩余帧数

4. 资源清理 (closeEvent)

为了确保视频文件正确关闭,即使程序异常退出,也应在 QWidget 关闭时执行清理操作。

    def closeEvent(self, event):
        if not self._vid_writer.closed:
            self._vid_writer.close() # 关闭视频写入器
        self._timer.stop() # 停止定时器
        event.accept() # 接受关闭事件

完整示例代码

将以上部分整合,形成一个可运行的完整示例:

import imageio, numpy as np
from PySide6.QtWidgets import QApplication, QWidget
from PySide6.QtCore import QPoint, QRect, QTimer, Qt
from PySide6.QtGui import QPainter, QPointList, QImage

WIDTH = 720
HEIGHT = 720

class PlotWidget(QWidget):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.setWindowTitle("PySide6 动态绘制与视频录制")
        self.setFixedSize(WIDTH, HEIGHT)

        self._timer = QTimer(self)
        self._timer.setInterval(100) # 100ms = 10 FPS
        self._timer.timeout.connect(self.frame)

        self._points = QPointList()

        self._totalFrames = 100 # 录制100帧
        self._vid_writer = imageio.get_writer('output_video.avi', fps=10) # 输出视频文件

        self._timer.start() # 启动定时器

    def closeEvent(self, event):
        """
        在窗口关闭时,确保视频写入器和定时器被正确关闭。
        """
        if not self._vid_writer.closed:
            self._vid_writer.close()
            print("视频写入器已关闭。")
        self._timer.stop()
        event.accept()

    def frame(self):
        """
        定时器触发的方法,用于更新数据、重绘界面并捕获帧。
        """
        # 示例:更新绘制数据,这里只是简单地添加一个随机点
        self._points.clear()
        self._points.append(QPoint(np.random.randint(0, WIDTH), np.random.randint(0, HEIGHT)))

        if self._totalFrames > 0:
            self.update() # 触发 paintEvent 重新绘制界面

            # 捕获 QWidget 的当前显示内容
            pixmap = self.grab()
            # 转换为 QImage,并指定 RGB888 格式,便于后续 NumPy 处理
            qimg = pixmap.toImage().convertToFormat(QImage.Format_RGB888)

            # 将 QImage 的像素数据转换为 NumPy 数组
            # 注意 strides 参数确保正确解析 QImage 的内存布局
            array = np.ndarray((qimg.height(), qimg.width(), 3), 
                               buffer=qimg.constBits(), 
                               strides=[qimg.bytesPerLine(), 3, 1], 
                               dtype=np.uint8)

            # 将 NumPy 数组作为一帧追加到视频文件
            if not self._vid_writer.closed:
                self._vid_writer.append_data(array)
        else:
            # 录制帧数达到上限,停止定时器并关闭视频写入器
            self._timer.stop()
            if not self._vid_writer.closed:
                self._vid_writer.close()
            print(f"视频录制完成,已生成 {self._totalFrames} 帧,文件:output_video.avi")
            # 录制完成后可以考虑关闭应用程序
            QApplication.instance().quit()

        self._totalFrames -= 1

    def paintEvent(self, event):
        """
        QPainter 绘制事件,用于在 QWidget 上绘制内容。
        """
        with QPainter(self) as painter: # QPainter 直接作用于当前 QWidget
            rect = QRect(QPoint(0, 0), self.size())
            painter.fillRect(rect, Qt.white) # 填充白色背景

            painter.setPen(Qt.red) # 设置画笔颜色为红色
            painter.setBrush(Qt.NoBrush) # 不填充
            painter.drawPoints(self._points) # 绘制随机点

if __name__ == '__main__':
    app = QApplication([])
    plot_widget = PlotWidget()
    plot_widget.show()
    app.exec()

注意事项与最佳实践

  1. QPainter 上下文: 始终确保 QPainter 在其绘制设备上是唯一的活动实例。在 paintEvent 中,QPainter(self) 是正确的用法,因为它直接在当前 QWidget 上绘制。避免在同一个 paintEvent 周期内尝试在不同设备(如 QImage 和 QWidget)之间切换 QPainter。
  2. 帧捕获时机: self.grab() 应该在 self.update() 之后调用,以确保捕获到的是最新的绘制内容。
  3. 图像格式转换: QPixmap 转换为 QImage 时,选择合适的格式(如 QImage.Format_RGB888 或 Format_ARGB32)可以简化后续与 NumPy 的集成。imageio 通常期望 RGB 格式的 NumPy 数组。
  4. NumPy 数组转换效率: 使用 qimg.constBits() 直接访问 QImage 的底层数据缓冲区,并结合 np.ndarray 的 buffer 和 strides 参数,是最高效的转换方式,避免了数据复制。
  5. 资源管理: 务必在应用关闭或视频录制完成后,调用 _vid_writer.close() 来释放文件句柄并确保视频文件完整。closeEvent 是一个理想的清理位置。
  6. 帧率控制: QTimer.setInterval() 和 imageio.get_writer(fps=...) 的帧率应保持一致,以确保视频播放速度与预期相符。
  7. 性能考虑: 对于高分辨率或高帧率的视频录制,图像转换和写入操作可能会消耗较多 CPU 资源。可以考虑在单独的线程中执行视频写入操作,以避免阻塞 UI 线程。

总结

通过本教程,我们学习了如何在 PySide6 中优雅地实现 QWidget 的动态绘制,并同时将这些动态画面录制成视频。关键在于理解 QPainter 的绘制上下文,利用 QWidget.grab() 进行界面捕获,并通过 imageio 库将捕获的图像帧高效地转换为视频。这种方法避免了常见的绘制错误,并提供了一个清晰、专业的解决方案,适用于需要实时动画显示和视频输出的 PySide6 应用。

相关专题

更多
python开发工具
python开发工具

php中文网为大家提供各种python开发工具,好的开发工具,可帮助开发者攻克编程学习中的基础障碍,理解每一行源代码在程序执行时在计算机中的过程。php中文网还为大家带来python相关课程以及相关文章等内容,供大家免费下载使用。

772

2023.06.15

python打包成可执行文件
python打包成可执行文件

本专题为大家带来python打包成可执行文件相关的文章,大家可以免费的下载体验。

661

2023.07.20

python能做什么
python能做什么

python能做的有:可用于开发基于控制台的应用程序、多媒体部分开发、用于开发基于Web的应用程序、使用python处理数据、系统编程等等。本专题为大家提供python相关的各种文章、以及下载和课程。

764

2023.07.25

format在python中的用法
format在python中的用法

Python中的format是一种字符串格式化方法,用于将变量或值插入到字符串中的占位符位置。通过format方法,我们可以动态地构建字符串,使其包含不同值。php中文网给大家带来了相关的教程以及文章,欢迎大家前来阅读学习。

679

2023.07.31

python教程
python教程

Python已成为一门网红语言,即使是在非编程开发者当中,也掀起了一股学习的热潮。本专题为大家带来python教程的相关文章,大家可以免费体验学习。

1365

2023.08.03

python环境变量的配置
python环境变量的配置

Python是一种流行的编程语言,被广泛用于软件开发、数据分析和科学计算等领域。在安装Python之后,我们需要配置环境变量,以便在任何位置都能够访问Python的可执行文件。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

570

2023.08.04

python eval
python eval

eval函数是Python中一个非常强大的函数,它可以将字符串作为Python代码进行执行,实现动态编程的效果。然而,由于其潜在的安全风险和性能问题,需要谨慎使用。php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

579

2023.08.04

scratch和python区别
scratch和python区别

scratch和python的区别:1、scratch是一种专为初学者设计的图形化编程语言,python是一种文本编程语言;2、scratch使用的是基于积木的编程语法,python采用更加传统的文本编程语法等等。本专题为大家提供scratch和python相关的文章、下载、课程内容,供大家免费下载体验。

730

2023.08.11

C++ 高级模板编程与元编程
C++ 高级模板编程与元编程

本专题深入讲解 C++ 中的高级模板编程与元编程技术,涵盖模板特化、SFINAE、模板递归、类型萃取、编译时常量与计算、C++17 的折叠表达式与变长模板参数等。通过多个实际示例,帮助开发者掌握 如何利用 C++ 模板机制编写高效、可扩展的通用代码,并提升代码的灵活性与性能。

4

2026.01.23

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
最新Python教程 从入门到精通
最新Python教程 从入门到精通

共4课时 | 14.2万人学习

Django 教程
Django 教程

共28课时 | 3.4万人学习

SciPy 教程
SciPy 教程

共10课时 | 1.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号