PyQt5桌面应用开发(17):类结构+QWebEngineView

news2025/1/19 16:57:21

本文目录

  • PyQt5桌面应用系列
  • PyQt5学习
  • PyQt5类结构和帮助速查
  • 实现与解释
  • 最终界面和完整源代码
    • 界面
    • 完整的代码
  • 总结

PyQt5桌面应用系列

  • PyQt5桌面应用开发(1):需求分析
  • PyQt5桌面应用开发(2):事件循环
  • PyQt5桌面应用开发(3):并行设计
  • PyQt5桌面应用开发(4):界面设计
  • PyQt5桌面应用开发(5):对话框
  • PyQt5桌面应用开发(6):文件对话框
  • PyQt5桌面应用开发(7):文本编辑+语法高亮与行号
  • PyQt5桌面应用开发(8):从QInputDialog转进到函数参数传递
  • PyQt5桌面应用开发(9):经典布局QMainWindow
  • PyQt5桌面应用开发(10):界面布局基本支持
  • PyQt5桌面应用开发(11):摸鱼也要讲基本法,两个字,16
  • PyQt5桌面应用开发(12):QFile与线程安全
  • PyQt5桌面应用开发(13):QGraphicsView框架
  • PyQt5桌面应用开发(14):数据库+ModelView+QCharts
  • PyQt5桌面应用开发(15):界面动画
  • PyQt5桌面应用开发(16):定制化控件-QPainter绘图
  • PyQt5桌面应用开发(17):类结构+QWebEngineView

PyQt5学习

学习PyQt5有一小段时间了,看了大概四五本书的样子。

  • PyQt编程快速上手,人民邮电出版社,任路顺,2023-04
  • Qt for Python PySide6 GUI界面开发详解与实例,清华大学出版社,李增刚,2022-08
  • PyQt从入门到精通,清华大学出版社,明日科技,2021-06
  • Python Qt GUI与数据可视化编程,人民邮电出版社,王维波,2019-09
  • PyQt5快速开发与实战,电子工业出版社,王硕,2017-10

这几本书的结构简单整理如下,建议只看看目录或者电子版快速过一遍就行。

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

国内还有一本书,微信读书上没有电子版。其实这本书看起来挺吸引人的。

  • Qt5/PyQt5实战指南:手把手教你掌握100个精彩案例

国外的PyQt5的书比较多,扫射了若干本,感觉比国内的书写得还要差。有一本Cookbook,都出到第三版,其内容非常简易,干货特别少,举的例子也不是很恰当。反而是国内的那几本书,没那么不堪。

要学习PyQt5,目的肯定是用PyQt5来做项目、做应用。其实最好的学习资料我感觉是qt.io
,如果对C++有一定的基础,很容易把这里面的内容结合到PyQt5中的stub(也就是*.pyi文件中描述的类接口),对具体开发的支持帮助很大。我基本上都是要用什么就看什么,然后去找Qt5
C++的文档来看。所以我写的这一系列教程,非常少罗列方法、信号和槽,因为这些信息在qt.io上非常齐全,介绍也很清晰。

这个系列基本上是从应用侧来观察PyQt5,落脚点始终在功能和应用上。国内的这几本我看过的书,相对来说,信息量也都是比较小,罗列接口的比较多,但是又不能作为完整的参考手册来使用,非常尴尬。其中那本从《PyQt5快速开发与实战》,我还不幸买了纸质版,很厚一本,挺烦人的。

PyQt5类结构和帮助速查

说起qt.io,有个小毛病,按类名查询慢吞吞的,主要是别的东西太多。那就自己撸一个。

开发需求,两个报表:

  • 显示PyQt5的类结构
  • 显示对应类的帮助

交互设计:

  • 显示一个类的树视图
  • 点击类名,显示对应的帮助

实现与解释

文件头很简单,导入包,这里专门用了*的方式导入,把所有类名都放到当前的空间中,目的是为了变了反射。

import re
import sys

# noinspection PyUnresolvedReferences
from PyQt5.QtCore import *
# noinspection PyUnresolvedReferences
from PyQt5.QtGui import *
from PyQt5.QtWebEngineWidgets import QWebEngineView
# noinspection PyUnresolvedReferences
from PyQt5.QtWidgets import *

这里就用一个魔术变量__name__和sys.modules把类名转换成类本身。从这里也可以看到,Python是如何找到一个类的。

def str_to_class(name):
    return getattr(sys.modules[__name__], name)

这个函数,就是找出pyi文件中的顶级类,也就是那些没有父类的类。在PyQt5的各个结构定义中,把父类设定为是sip.wrapper
或者PyQt5.sipsimplewrapper的哪些类。具体的路径,根据每个人PyQt5的安装位置不同。这里我们只针对"QtCore", “QtGui”, "
QtWidgets"
三个包。打开pyi文件,对每行进行模式匹配。这里有一点点Python的字符串匹配的内容"^class ([^\(\)]*)\(sip.wrapper\):$"
,这里就是匹配那些行开头是class,行结尾是:,中间把类名抓出来。方法最后,调用上面的函数,把字符串转为类。

def root_object_names(pyqt5_root="venv/Lib/site-packages/PyQt5-stubs",
                      class_fingerprint="^class ([^\(\)]*)\(sip.wrapper\):$"):
    files = [f"{pyqt5_root}/{fn}.pyi" for fn in ["QtCore", "QtGui", "QtWidgets"]]
    names = []
    for f in files:
        with open(f) as fid:
            for line in fid:
                ret = re.match(class_fingerprint, line)
                if ret is None:
                    continue
                captures = ret.groups()
                if len(captures) > 0:
                    names.append(captures[0])

    return [str_to_class(n) for n in sorted(names)]

此外还要定义一个方法,把类和子类构造成QTreeWidget中显示的形式。这个函数返回的是一个QTreeWidgetItem
,通过这个类的构造方法,自动把父子关系给建立起来了。这个函数的要点是两个。

  • __subclasses__魔术方法,得到子类;
  • 递归调用,构造父类-子类的完整的树
def walk_to_QTreeWdigetItem(self: object, parent: QTreeWidgetItem = None):
    sc = self.__subclasses__()
    item = QTreeWidgetItem(parent)
    item.setText(0, self.__name__)
    item.setText(1, f"{len(self.__dict__)}")
    item.setExpanded(True)
    for c in sc:
        walk_to_QTreeWdigetItem(c, item)
    return item

实现这个玩意的痛点就是要求能够搜索得比较快,那么这里我们定义一个包含搜索框和QTreeWidget的QWidget。构造函数中第一部分是构造界面,安排布局,上面一个搜索框,下面一个树控件。

第二部分就是树控件里面填充进PyQt5的类。

第三部分就是处理两个信号。

  • 树控件的点击
  • 搜索框的文字变化

可以看到,这个类还定义了一个信号,选择了一个项,这个信号,由树控件的点击事件触发,信号的参数就是类名。

class TreeWithSearch(QWidget):
    selectItem = pyqtSignal(str)

    def __init__(self, parent=None, classes=None):
        super(TreeWithSearch, self).__init__(parent=parent)
        self.layout = QVBoxLayout(self)
        self.searchBox = QLineEdit()
        self.classes = QTreeWidget()
        self.layout.addWidget(self.searchBox)
        self.layout.addWidget(self.classes)
        self.setLayout(self.layout)

        self.searchBox.setPlaceholderText("type class name")

        if classes is None:
            classes = root_object_names()
            classes.extend(
                root_object_names("venv/Lib/site-packages/PyQt5",
                                  "^class ([^\(\)]*)\(PyQt5.sipsimplewrapper\):$")
            )

            classes.sort(key=lambda iClass: iClass.__name__)

        for i, c in enumerate(classes):
            root = walk_to_QTreeWdigetItem(c)
            self.classes.addTopLevelItem(root)

        self.classes.setHeaderLabels(["name", "funcs"])
        self.classes.setColumnHidden(1, True)
        self.classes.setHeaderHidden(True)

        self.classes.clicked.connect(
            lambda index:
            self.selectItem.emit(self.classes.itemFromIndex(index).text(0))
        )

        self.searchBox.textChanged.connect(self._textChanged)

接下来就是有意思的一个点,搜索框的文字变化。这段代码是一个参数为字符串的槽函数。槽函数中,如果搜索框为空白,那么就遍历树控件中有子节点的节点,把他们全部设为折叠。

第二部分代码处理搜索。针对这个搜索字符串,我们遍历所有的节点,如果节点对应的类名包含搜索字符串,我们把这个节点的所有夫节点设为可见、展开。不包含的节点,设为隐藏。

    @pyqtSlot(str)


def _textChanged(self, name: str):
    # collapse parent nodes
    if name.isspace():
        iterator = QTreeWidgetItemIterator(self.classes, QTreeWidgetItemIterator.HasChildren)
        while (item := iterator.value()) is not None:
            item: QTreeWidgetItem
            item.setExpanded(False)
            iterator += 1
        return

    iterator = QTreeWidgetItemIterator(self.classes, QTreeWidgetItemIterator.All)
    while (item := iterator.value()) is not None:
        item: QTreeWidgetItem
        class_name: str = item.text(0).lower()
        is_show = name.strip().lower() in class_name
        item.setHidden(not is_show)
        # toggle to show and expand all its ancestors
        if is_show:
            p = item
            while p := p.parent():
                p.setHidden(False)
                p.setExpanded(True)
        iterator += 1

这里的关键知识点就是QTreeWidgetItemIterator,这个遍历器的访问方法就是it.value()函数和it += 1。如果漏掉了这个加一,那这个循环就成了无限循环。

最后就是整一个浏览视图。

class QtHelpView(QWebEngineView):

    def __init__(self, parent=None):
        super(QtHelpView, self).__init__(parent)
        self.base_url = "https://doc.qt.io/qt-5"
        self.load(QUrl(self.base_url))

    @pyqtSlot(str)
    def show_class(self, name):
        self.load(QUrl(f"{self.base_url}/{name.lower()}.html"))

主函数为一个要做的就是把界面搭起来,并且把TreeWithSearch的选择一项的信号与QtHelpView中的槽函数连起来。

if __name__ == '__main__':
    app = QApplication([])

    main_window = QMainWindow()

    tree = TreeWithSearch(main_window)

    dock = QDockWidget()
    dock.setWidget(tree)

    main_window.addDockWidget(Qt.LeftDockWidgetArea, dock)

    # pip install PyQtWebEngine
    view = QtHelpView(main_window)
    tree.selectItem.connect(view.show_class)

    main_window.setCentralWidget(view)

    main_window.resize(1440, 900)
    main_window.show()
    sys.exit(app.exec_())

最终界面和完整源代码

界面

最终实现的界面如下。通过这个app,可以浏览PyQt5主要的类结构,比如QObject和QPaintDevice这两个最主要的基类。点击一个类,就可以在右边看到对应的帮助(前提得上网)。

在这里插入图片描述

完整的代码

import re
import sys

# noinspection PyUnresolvedReferences
from PyQt5.QtCore import *
# noinspection PyUnresolvedReferences
from PyQt5.QtGui import *
from PyQt5.QtWebEngineWidgets import QWebEngineView
# noinspection PyUnresolvedReferences
from PyQt5.QtWidgets import *


def str_to_class(name):
    return getattr(sys.modules[__name__], name)


def root_object_names(pyqt5_root="venv/Lib/site-packages/PyQt5-stubs",
                      class_fingerprint="^class ([^\(\)]*)\(sip.wrapper\):$"):
    files = [f"{pyqt5_root}/{fn}.pyi" for fn in ["QtCore", "QtGui", "QtWidgets"]]
    names = []
    for f in files:
        with open(f) as fid:
            for line in fid:
                ret = re.match(class_fingerprint, line)
                if ret is None:
                    continue
                captures = ret.groups()
                if len(captures) > 0:
                    names.append(captures[0])

    return [str_to_class(n) for n in sorted(names)]


def walk_to_QTreeWdigetItem(self: object, parent: QTreeWidgetItem = None):
    sc = self.__subclasses__()
    item = QTreeWidgetItem(parent)
    item.setText(0, self.__name__)
    item.setText(1, f"{len(self.__dict__)}")
    item.setExpanded(True)
    for c in sc:
        walk_to_QTreeWdigetItem(c, item)
    return item


class TreeWithSearch(QWidget):
    selectItem = pyqtSignal(str)

    def __init__(self, parent=None, classes=None):
        super(TreeWithSearch, self).__init__(parent=parent)
        self.layout = QVBoxLayout(self)

        self.searchBox = QLineEdit()
        self.searchBox.setPlaceholderText("type class name")
        self.classes = QTreeWidget()

        if classes is None:
            classes = root_object_names()
            classes.extend(
                root_object_names("venv/Lib/site-packages/PyQt5",
                                  "^class ([^\(\)]*)\(PyQt5.sipsimplewrapper\):$")
            )

            classes.sort(key=lambda iClass: iClass.__name__)

        for i, c in enumerate(classes):
            root = walk_to_QTreeWdigetItem(c)

            self.classes.addTopLevelItem(root)
        self.classes.setHeaderLabels(["name", "funcs"])
        self.classes.setColumnHidden(1, True)
        self.classes.setHeaderHidden(True)
        self.layout.addWidget(self.searchBox)
        self.layout.addWidget(self.classes)

        self.setLayout(self.layout)

        self.classes.clicked.connect(
            lambda index:
            self.selectItem.emit(self.classes.itemFromIndex(index).text(0))
        )

        self.searchBox.textChanged.connect(self._textChanged)

    @pyqtSlot(str)
    def _textChanged(self, name: str):
        # collapse parent nodes
        if name.isspace():
            iterator = QTreeWidgetItemIterator(self.classes, QTreeWidgetItemIterator.HasChildren)
            while (item := iterator.value()) is not None:
                item: QTreeWidgetItem
                item.setExpanded(False)
                iterator += 1
            return

        iterator = QTreeWidgetItemIterator(self.classes, QTreeWidgetItemIterator.All)
        while (item := iterator.value()) is not None:
            item: QTreeWidgetItem
            class_name: str = item.text(0).lower()
            is_show = name.strip().lower() in class_name
            item.setHidden(not is_show)
            # toggle to show and expand all its ancestors
            if is_show:
                p = item
                while p := p.parent():
                    p.setHidden(False)
                    p.setExpanded(True)
            iterator += 1


class QtHelpView(QWebEngineView):

    def __init__(self, parent=None):
        super(QtHelpView, self).__init__(parent)
        self.base_url = "https://doc.qt.io/qt-5"
        self.load(QUrl(self.base_url))

    @pyqtSlot(str)
    def show_class(self, name):
        self.load(QUrl(f"{self.base_url}/{name.lower()}.html"))


if __name__ == '__main__':
    app = QApplication([])

    main_window = QMainWindow()

    tree = TreeWithSearch(main_window)

    dock = QDockWidget()
    dock.setWidget(tree)

    main_window.addDockWidget(Qt.LeftDockWidgetArea, dock)

    # pip install PyQtWebEngine
    view = QtHelpView(main_window)
    tree.selectItem.connect(view.show_class)

    main_window.setCentralWidget(view)

    main_window.resize(1440, 900)
    main_window.show()
    sys.exit(app.exec_())

总结

  1. qt.io是最好的学习资源;
  2. Python的良好反射特性(魔术方法)是探索实际代码和机制的很好工具;
  3. 学一个东西拿来用好过学一百个东西。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/554437.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

Spring 如何处理请求参数和表单数据

当我们开发 Web 应用程序时,处理请求参数和表单数据是必不可少的。Spring MVC 是一个流行的 Java Web 框架,提供了多种方式来处理请求参数和表单数据。本文将介绍 Spring MVC 中处理请求参数和表单数据的常用方式,并提供相应的代码示例。 处…

Redis数据结构-SDS

一、SDS(Simple Dynamic String,简单动态字符串) Redis没有使用C语言传统的字符串表示方式(以’\0’结尾的字符数组),而是自己实现了sds的抽象类型,Redis默认使用sds作为字符串的表示。 set ms…

新视野(2023.5.5-2023.5.12)

一、知识: 媒体账号买卖网站 A5新媒体交易:https://xmt.a5.net/ 新媒虎:https://www.xinmeihu.com/ 二、资讯: GPT最大的竞争对手Claude宣布支持100K的上下文!基本上一本中篇小说都能塞进去了,你再不需要ChatPDF这种…

【框架源码】手写Spring框架IOC容器核心流程

要是想要了解Spring IOC底层,首先我们就得先了解什么是IOC。 IOC就是控制反转,把对象创建和对象之间的调用过程交给Spring进行管理。 使用IOC目的就是之前我们创建对象的方式都是用new的方式创建,这个方式有个缺点,被创建对象的…

【halcon资料】取出区域的轮廓上所有转折点

一、说明 在区域运算的时候,有时候需要用图形的顶点来描述,比如,两个图中对象需要对齐,或者仿射变换,于是特征点是需要提取的。本文给出一个提取顶点的示例。 二、算子 1.1 get_region_polygon算子 (1&a…

2023互联网Java面试真题1000道(附答案)

前言 2023 跳槽不迷茫,大家可以先收藏再看,后续跳槽都能用上的! Java程序员绝大部分工作的时间都是增删改查,很多人觉得这项工作没什么技术含量,任何一件事情都要站在不同的角度去考虑,对于大部分的java程序…

拼多多新阶段,透露出不寻常

一个企业的特质,往往由这个企业的领导人所决定。 企业文化本质上就是领导人文化,领导人的风格会决定这个企业当下的现状。一个成功的大企业,往往需要不同的领导人来接替完成其发展使命。 在创业期,企业领导人需要的是勇气、执行…

统计一个数的二进制中1的个数(三种方法)

那么好了好了&#xff0c;宝子们&#xff0c;今天给大家分享一篇经典例题的三种实现方法&#xff0c;来吧&#xff0c;开始整活&#xff01;⛳️ 一、基础法 #define _CRT_SECURE_NO_WARNINGS 1 #include <stdio.h> int number_of_one(int n) {int count 0;while(n){if…

一文读懂selenium自动化测试(基于Python)

前言 我们今天来聊聊selenium自动化测试&#xff0c;我们都知道selenium是一款web自动化测试的工具&#xff0c;它应该如何去运用呢?我们接着看下去。 ​1、Selenium简介&#xff1a; 1.1 Selenium&#xff1a; Selenium是一款主要用于Web应用程序自动化测试的工具集合。Sele…

gcc-arm-none-eabi-10.3-2021.10-src.tar.bz2 的编译环境搭建

文章目录 gcc-arm-none-eabi-10.3-2021.10-src.tar.bz2 的编译环境搭建概述实验END gcc-arm-none-eabi-10.3-2021.10-src.tar.bz2 的编译环境搭建 概述 正在迁移Smoothieware_best-for-pnp到MCUXPresso的失败实验中徘徊. 将Smoothieware_best-for-pnp升级到和MCUXPresso相同的…

LeetCode:26. 删除有序数组中的重复项

26. 删除有序数组中的重复项 1&#xff09;题目2&#xff09;代码3&#xff09;结果 1&#xff09;题目 给你一个 升序排列 的数组 nums &#xff0c;请你 原地 删除重复出现的元素&#xff0c;使每个元素 只出现一次 &#xff0c;返回删除后数组的新长度。元素的 相对顺序 应…

可以写进简历的软件测试项目(银行/金融/电商/商城......)

目录 一、引言 二、测试任务 三、测试进度 四、测试资源 五、测试策略 六、测试完成标准 七、风险和约束 八、问题严重程度描述和响应时间规范 九、测试的主要角色和职责 ​有需要实战项目的评论区留言吧&#xff01; 软件测试是使用人工或者自动的手段来运行或者测定…

赛效:如何制作gif动态图

将一张静态图转为动态图的好处是增强照片的表现力和吸引力&#xff0c;通过动态效果和音效等元素来传达更加生动和直观的信息&#xff0c;更容易吸引人们的注意力和兴趣。此外&#xff0c;动态图还可以增强视觉效果&#xff0c;使得图片更加生动和有趣&#xff0c;更容易被人们…

C++系列之类与对象(下)

&#x1f497; &#x1f497; 博客:小怡同学 &#x1f497; &#x1f497; 个人简介:编程小萌新 &#x1f497; &#x1f497; 如果博客对大家有用的话&#xff0c;请点赞关注再收藏 &#x1f31e; 插入和输出流 //流插入不能写成成员函数&#xff0c;因为Date对象默认占用了第…

浅谈人工智能

人工智能的概念和现状 人工智能&#xff08;Artificial Intelligence&#xff0c;简称AI&#xff09;是指通过计算机程序和算法来模拟人类智能&#xff0c;包括学习、推理、感知、语言理解、图像识别等方面。自20世纪50年代以来&#xff0c;人工智能领域的研究取得了巨大的进展…

ChatGPT模型大战:讯飞星火认知大模型、百度文心一言能否击败GPT-4(含个人内测体验测试邀请码获取方法,2小时申请成功,亲测有效)

目录 前言讯飞星火内测申请申请方式内测体验登录界面 百度内测申请内测对比基本问答事实性问答科普文写作小红书文案项目计划撰写古文理解模型的常识能力和反事实推理代码理解法律相关广告话术数字排序数值计算推理解题跨语言能力文生图 总结个人感受一、你有使用过这种对话式A…

MySQL无法启动服务--NET HELPMSG 3534

一、问题综述 安装完MySQL&#xff0c;使用 net start mysql 时&#xff0c;出现无法启动服务 二、解决思路 1. 查看一下错误日志 出现了问题&#xff0c;第一步先查看一下错误日志&#xff0c;使用 mysqld --console 打印错误信息&#xff0c;发现是 Failed to find valid …

H264: [ RTP传H264裸流 ] > 如何传(关注点:H264部分)

RTP传h264裸流, 如何传: 可能有几种情况: 1 一帧传一个NALU(NALU很小) 2 一帧传几个NALU(几个NALU很小)[STAP-A] 3 一帧连一个NALU都传不完(一个NALU很大)[FU-A] 如何解决这些问题?? 单一NALU模式:一帧传一个NALU [rtp帧头] [nalu header] [多媒体数据] 一帧传几个NAL…

字节8年测试经验,从功能测试到自动化测试,我整理了这一份2000字进阶学习指南

随着软件行业的不断发展&#xff0c;软件测试技术也在不断地更新&#xff0c;出现了众多的自动化功能测试工具&#xff0c;如HP的Quick Test Professional&#xff08;最新版本名为UFT&#xff09;及开源的Selenium。性能测试工具如LoadRunner、JMeter等。 所谓自动化测试&…

OpenCloudOS是哪个Linux版本?哪国的?

OpenCloudOS是哪个Linux版本&#xff1f;哪国的&#xff1f;OpenCloudOS国产的开源Linux操作系统。OpenCloudOS完全兼容CentOS 8。 OpenCloudOS是什么&#xff1f;OpenCloudOS是Linux哪个版本&#xff1f;OpenCloudOS是哪个国家的&#xff1f;OpenCloudOS是一个国产操作系统开…