VB.net 2010 视频教程 VB.net 2010 视频教程 python基础视频教程
SQL Server 2008 视频教程 c#入门经典教程 Visual Basic从门到精通视频教程
当前位置:
首页 > Python基础教程 >
  • PYQT5 系列(一)——参考自《弗兰克万岁》(2)

复制代码

PYQT5(二十九)多屏幕适配问题

        
self.move(int(QApplication.desktop().screenGeometry(0).width())-180, 90, )
#  screenGeometry(0)就是主屏幕
#  1的话是分屏1,以此类推

PYQT5(三十)去掉窗口标题栏,去掉任务栏显示,窗口置顶的方法

复制代码
this->setWindowFlags
(Qt::FramelessWindowHint | Qt::Tool | Qt::WindowStaysOnTopHint); 
// 去掉标题栏,去掉任务栏显示,窗口置顶

翻译成pyqt---------------------
window.setWindowFlags
(Qt.Qt.FramelessWindowHint|Qt.Qt.WindowStaysOnTopHint|Qt.Qt.Tool)
复制代码

PYQT5(三十一)保存窗口位置下次使用

复制代码
from PyQt5.Qt import QMainWindow,QDialog
import json
window=QMainWindow()
def save_location(self):
        with open('window_location.txt','w') as f:
            data={'x':self.window.x(),'y':self.window.y()}
            f.write(json.dumps(data))
    def load_laction(self):
        with open('window_location.txt', 'r') as f:
            txt=f.read()
            print(txt)
            j=json.loads(txt)
            return j
复制代码

PYQT5(三十二)QMainWindow 重写CloseEvent

    def closeEvent(self, e):
        e.ignore()
        self.hide()
        self.save_location()

PYQT5(三十三)输入对话框

    QID = QInputDialog()
    QID.setGeometry(int((screen.width() - size.width()) / 2),
                    int((screen.height() - size.height())) / 2, 200, 200)
    text, ok = QID.getText(self.window, '输入', '标题')

PYQT5(三十四)日期控件

    self.dateEdit.setDate(QDate.currentDate())
    self.dateEdit.setCalendarPopup(True)

三十五 没啥用,作者的近况

 

PYQT5(三十六)获取屏幕分辨率大小

 

        
        from PyQt5.QtWidgets import QApplication, QWidget
        self.desktop = QApplication.desktop()
        self.screenRect = self.desktop.screenGeometry()
        self.height = self.screenRect.height()
        self.width = self.screenRect.width()

PYQT5(三十七)表格控件调整列宽

复制代码
列宽自动分配
--------------------------------------
self.tableWidget.horizontalHeader().setSectionResizeMode(QHeaderView.Stretch)
--------------------------------------
self.tableWidget.verticalHeader().setSectionResizeMode(QHeaderView.Stretch)
--------------------------------------
这个是行宽的
此种模式下,我们没法手动调整列宽。

手动调整
--------------------------------------
self.tableWidget.horizontalHeader().setSectionResizeMode(QHeaderView.Interactive)

固定值
--------------------------------------
self.tableWidget.horizontalHeader().setSectionResizeMode(QHeaderView.Fixed)

用户无法调整该部分的大小。该部分只能使用resizeSection()以编程方式调整大小。节大小默认为defaultSectionSize。

随内容分配列宽
--------------------------------------
self.tableWidget.horizontalHeader().setSectionResizeMode(QHeaderView.Stretch)

self.tableWidget.horizontalHeader().setSectionResizeMode(0, QHeaderView.ResizeToContents)

我们把它和列宽自动分配结合在一起看,效果立马显现。

上面两种方式结合
--------------------------------------
self.tableWidget.horizontalHeader().setSectionResizeMode(QHeaderView.Stretch)

self.tableWidget.horizontalHeader().setSectionResizeMode(0, QHeaderView.Interactive)


列宽是自动分配的,但是第一列我们可以手动调整宽度,而表格整体的列宽仍是自动分配的。第二、三列我们没有办法调整列宽,这个是自动分配的。

自定义列宽

--------------------------------------
1. self.tableWidget.setColumnWidth(0, 40)
2. self.tableWidget.setColumnWidth(1, 200)
3. self.tableWidget.setColumnWidth(2, 200)
复制代码

PYQT(三十八)PDF阅读器

复制代码
https://blog.csdn.net/weixin_43773093/article/details/88783757

随便复制了一下,以防走丢。
UI 设计
首先使用 Qt Designer 设计出图形界面:

新建一个 MainWindow 主界面,然后设置一个 toolbar,并在 toolbar 中添加三个 action,并为每个 action 设置好相应图标。

也可以直接 compile 我制作好的 PyReader.ui 文件,或者导入 Ui_PyReader.py 文件。

依赖要求
Python3
PyQt5
PyMuPDF

主要任务
我们使用 PyMuPDF 来解析 PDF ,来获取 PDF 文本信息。

安装
我们只须在 cmd 中输入:pip install PyMuPDF,即可安装 PyMuPDF。

导入

导入 PyMuPDF
import fitz
在本节中,我们只需了解以下几个基本操作:

fitz.open() 函数用来读取 PDF 文件内容,doc.loadPage() 函数用来获取具体某一页的信息。特别的 ,我们使用loadPage(0) 来获取封面信息。

# 读取 PDF
doc = fitz.open(fname)
# 获取第 n 页内容
page = doc.loadPage(n)
本节主要的内容就是把封面渲染到主界面中,并完成添加与删除封面的任务。

显示表格
我们采用 QtWidgets.QTableWidget 表格控件来显示封面。

首先让我们设置表格样式与功能:

其中,我们设置了单元格的纵横比为 4 : 3,以及其他的一些静态属性,并将 self.table 与右键菜单绑定,支持点击单元格调用 self.generateMenu 函数。
----------------------------------------------------
def _setTableStyle(self):
    # 开启水平与垂直滚轴
    self.table.setVerticalScrollBarPolicy(Qt.ScrollBarAlwaysOn)
    self.table.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOn)
    # 设置 5 行 8 列 的表格
    self.table.setColumnCount(8)
    self.table.setRowCount(5)
    # 设置标准宽度
    self.width = self.screen.width() // 8
    # 设置单元格的宽度
    for i in range(8):
        self.table.setColumnWidth(i, self.width)
    # 设置单元格的高度
    # 设置纵横比为 4 : 3
    for i in range(5):
        self.table.setRowHeight(i, self.width * 4 // 3)
    # 隐藏标题栏
    self.table.verticalHeader().setVisible(False)
    self.table.horizontalHeader().setVisible(False)
    # 禁止编辑
    self.table.setEditTriggers(QAbstractItemView.NoEditTriggers)
    # 不显示网格线
    self.table.setShowGrid(False)
    # 将单元格绑定右键菜单
    # 点击单元格,调用 self.generateMenu 函数
    self.table.setContextMenuPolicy(Qt.CustomContextMenu)
    self.table.customContextMenuRequested.connect(self.generateMenu)
----------------------------------------------------
加封面
首先让我们来看如何生成 TableWidget 可显示的 图像类文件。

我们通过 doc.loadPage(0) 获取页面对象,并传递给 render_pdf_page() 函数,设置缩放比为 1 : 1。首先构建 QImage 对象,在通过 convertFromImage 函数将 QImage 对象转化为可显示对象。
----------------------------------------------------
# 显示 PDF 封面
# page_data 为 page 对象
def render_pdf_page(page_data, for_cover=False):
    # 图像缩放比例
    zoom_matrix = fitz.Matrix(4, 4)
    if for_cover:
        zoom_matrix = fitz.Matrix(1, 1)
    
    # 获取封面对应的 Pixmap 对象
    # alpha 设置背景为白色
    pagePixmap = page_data.getPixmap(
        matrix = zoom_matrix, 
        alpha=False) 
    # 获取 image 格式
    imageFormat = QtGui.QImage.Format_RGB888 
    # 生成 QImage 对象
    pageQImage = QtGui.QImage(
        pagePixmap.samples,
        pagePixmap.width, 
        pagePixmap.height, 
        pagePixmap.stride,
        imageFormat)

    # 生成 pixmap 对象
    pixmap = QtGui.QPixmap()
    pixmap.convertFromImage(pageQImage)
    return pixmap
----------------------------------------------------
接着,我们就要想单元格中添加封面图片:

我们使用工具栏中的 + 号来添加 PDF 封面。
----------------------------------------------------
self.addbar.triggered.connect(self.open),当点击 + 号时,就会调用 self.open 函数。
----------------------------------------------------
我们通过 getOpenFileName() 函数 来获取文件地址,self 后面的三个参数分别是窗口名称,文件默认路径以及支持的文件类型,这个函数返回文件的地址。

filter_book() 函数用来确保不会重复显示同一本书的封面。
----------------------------------------------------
def getfile(self):
    # 打开单个文件
    fname, _ = QFileDialog.getOpenFileName(self, 'Open files', './', '(*.pdf)')
    return fname

def open(self):
    # 打开文件
    fname = self.getfile()
    if self.filter_book(fname):
        self.setIcon(fname)
                
# 获取无重复图书的地址
def filter_book(self, fname):
    if not fname:
        return False
    if fname not in self.booklist:
        self.booklist.append(fname)
        return True
    return False      
----------------------------------------------------               
然后,我们就要将 PDF 封面渲染到主界面上:

label.setScaledContents(True) 使得图片可以充满 label。self.table.setCellWidget(self.x, self.y, label) 用来设置标签的行与列。最后确保每八个元素换行,换行后将列数清零。
----------------------------------------------------
def setIcon(self, fname):
    # 打开 PDF
    doc = fitz.open(fname)
    # 加载封面
    page = doc.loadPage(0)
    # 生成封面图像
    cover = render_pdf_page(page, True)
    label = QLabel(self)
    # 设置图片自动填充 label
    label.setScaledContents(True)
    # 设置封面图片
    label.setPixmap(QPixmap(cover))
    # 设置单元格元素为 label
    self.table.setCellWidget(self.x, self.y, label)
    # 删除 label 对象,防止后期无法即时刷新界面
    # 因为 label 的生存周期未结束
    del label
    # 设置当前行数与列数
    self.crow, self.ccol = self.x, self.y
    # 每 8 个元素换行
    if (not self.y % 7) and (self.y):
        self.x += 1
        self.y = 0
    else:
        self.y += 1
----------------------------------------------------
右键菜单
上面我们已经提到,如何将单元格与右键菜单绑定。

本次教程中,右键菜单只有两项,分别为开始阅读(暂未实现),以及删除图书。
----------------------------------------------------
def generateMenu(self, pos):
    row_num = col_num = -1
    # 获取选中的单元格的行数以及列数
    for i in self.table.selectionModel().selection().indexes():
        row_num = i.row()
        col_num = i.column()
    # 若选取的单元格中有元素,则支持右键菜单
    if (row_num < self.crow) or (row_num == self.crow and col_num <= self.ccol):
        menu = QMenu()
        # 添加选项
        item1 = menu.addAction('开始阅读')
        item2 = menu.addAction('删除图书')
        # 获取选项
        action = menu.exec_(self.table.mapToGlobal(pos))
        if action == item1:
            pass
        # 点击选项二,调用 self.delete_book 删除图书
        elif action == item2:
            self.delete_book(row_num, col_num)
----------------------------------------------------
接下来,让我们看如何删除图书:

首先维护一个 self.booklist ,里面储存无重复 PDF 文件地址。首先获取图书在 booklist 中的索引,在 booklist 中删除该元素。接着清空选中单元格之后(包含选中单元格)的所有单元格的内容。最后将 booklist 中 index 之后的图书地址重新显示到 table 上。简单地说,就是删除选中单元格,并将之后单元格向前挪一位。

----------------------------------------------------
# 删除图书
def delete_book(self, row, col):
    # 获取图书在列表中的位置
    index = row * 8 + col
    self.x = row
    self.y = col
    if index >= 0:
        self.booklist.pop(index)

    i, j = row, col
    while 1:
        # 移除 i 行 j 列单元格的元素
        self.table.removeCellWidget(i, j)
        # 一直删到最后一个有元素的单元格
        if i == self.crow and j == self.ccol:
            break
        if (not j % 7) and j:
            i += 1
            j = 0
        else:
            j += 1

    # 如果 booklist 为空,设置当前单元格为 -1
    if not self.booklist:
        self.crow = -1
        self.ccol = -1
    # 删除图书后,重新按顺序显示封面图片
    for fname in self.booklist[index:]:
        self.setIcon(fname)
----------------------------------------------------
复制代码

PyQt5(三十九)QTableWidget(表单控件)自适应窗口大小、栏位大小调整及布局

https://blog.csdn.net/yl_best/article/details/84070231

PYQT5(四十)下载并安装配置QT-DESIGNER

https://www.jianshu.com/p/4df033879a3f

 

 

(1)找不着

PYQT5(2)MyScrollWidget拖放文件功能的实现 

复制代码
class MyScrollWidget(QWidget):
    def __init__(self):
        super(QWidget, self).__init__()
        # self.resize(800, 600)
        self.move(0, 0)
        # self.setMouseTracking(False)
        self.last_time_move = 0
        self.last_time_move_x = 0
    def eventFilter(self,source, event):
        try:
            if event.type() == QEvent.MouseMove:
                print(event.pos().y(),event.pos().x())
                if self.last_time_move == 0:
                    self.last_time_move = event.pos().y()
                    self.last_time_move_x=event.pos().x()
                distance = (self.last_time_move - event.pos().y())/100
                distance_x = (self.last_time_move - event.pos().x())/100
                self.scroll(distance_x,0)
                self.last_time_move = event.pos().y()
                self.last_time_move_x=event.pos().x()
            elif event.type() == QEvent.MouseButtonRelease:
                self.last_time_move = 0
                self.last_time_move_x = 0
            return QWidget.eventFilter(self, source, event)
        except Exception as e:
            print(e)
#by the way 没能搞定y轴上的滚动,不过发现用滚轮也可以,就先放一放了,实现功能再回来优化
复制代码

PYQT5(3)多线程QProgressBar卡死的问题

复制代码
def init_progress(parent):
    m = wake_progress()
    parent.progress = m
    m.run()
    if not m.window.isVisible():
        m.window.show()

#激活方法

class wake_progress(QThread):
#线程类
    def __init__(self):
        super().__init__()
        self.window = QMainWindow()
        self.p=''
    def run(self):
        self.p =file_upload_statu_bar(self.window)

class file_upload_statu_bar(progress_bar):
#原始类
    def __init__(self,Form):
        print('progress bar')
        self.window=Form
        super().__init__()
        self.setupUi(Form)
        self.window.show()
    def change(self,name):
        self.label.setText(name)
复制代码

出现进度条的线程卡死的情况

最后通过参考自知诸狭的文章
其中一段:
对于执行很耗时的程序来说,由于PyQt需要等待程序执行完毕才能进行下一步,这个过程表现在界面上就是卡顿;而如果在执行这个耗时程序时不断地运行QApplication.processEvents(),那么就可以实现一边执行耗时程序,一边刷新页面的功能,会给人一种相对更流畅的感觉,QApplication.processEvents()的使用方法是,在主函数执行耗时操作的地方,加入QApplication.processEvents(),processEvents()函数的使用方法简单来说就是刷新页面。

于是在循环内加入

复制代码
from PyQt5.QtWidgets import QApplication
def slotAdd(self): 
  for n in range(10): 
  str_n = 'File index {0}'.format(n)
  self.listFile.addItem(str_n) 
  QApplication.processEvents()#这一句是关键
  time.sleep(1) 
复制代码

就可以成功解决问题

PYQT5 (4)之QSystemTrayIcon实现窗口最小化到托盘

复制代码
from PyQt5 import QtGui

from PyQt5.QtWidgets import QSystemTrayIcon


class MyTray(QSystemTrayIcon):
    def __init__(self):
        super().__init__()
        try:
            self.setIcon(QtGui.QIcon('icon/car.png'))
            self.activated.connect(self.iconClicked)
        except Exception as e:
            print(e)
    def bind(self,window):
        self.parent_window=window
    def test(self):
        try:
            self.parent_window.show()
        except Exception as e:
            print(e)

    def iconClicked(self,reason):
    #鼠标点击icon传递的信号会带有一个整形的值,1是表示单击右键,2是双击,3是单击左键,4是用鼠标中键点击"
        print('click')
        if reason == 2 :#2是双击
            self.test()
复制代码

加一段添加二级菜单的功能,暂未试验过

复制代码
    def showMenu(self):
        "设计托盘的菜单,这里我实现了一个二级菜单"
        self.menu = QMenu()
        self.menu1 = QMenu()
        self.showAction1 = QAction("显示消息1", self, triggered=self.showM)
        self.showAction2 = QAction("显示消息2", self,triggered=self.showM)
        self.quitAction = QAction("退出", self, triggered=self.quit)

        self.menu1.addAction(self.showAction1)
        self.menu1.addAction(self.showAction2)
        self.menu.addMenu(self.menu1, )

        self.menu.addAction(self.showAction1)
        self.menu.addAction(self.showAction2)
        self.menu.addAction(self.quitAction)
        self.menu1.setTitle("二级菜单")
        self.setContextMenu(self.menu)
复制代码

PYQT5(5)带跳转按钮的QTableWidgets

复制代码
from PyQt5 import QtWidgets, QtCore
from PyQt5.QtWidgets import QTableWidget, QTableWidgetItem


class btn_table(QTableWidget):
    def __init__(self,x,y,window,res):
        self.centralwidget = QtWidgets.QWidget(window)
        self.centralwidget.setObjectName("centralwidget")
        self.centralwidget.resize(500,500)
        super().__init__(self.centralwidget)
        self.setGeometry(QtCore.QRect(20, 10, 471, 401))
        self.setObjectName("client_table")
        self.setColumnCount(x)
        self.setRowCount(y)

        self.buttonForRow(res)

        self.setVerticalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOn)
        self.setHorizontalScrollBarPolicy(QtCore.Qt.ScrollBarAlwaysOn)
    def buttonForRow(self,res):#添加内容并在最后一行添加一个按钮
        # requir:[[value1,value2,value3,...],[value1,value2,value3,value4,...]]
        self.clear()
        for i in range(0,len(res)):
            try:
                item = self.horizontalHeaderItem(i)
                self.setHorizontalHeaderItem(i, item)
            except Exception as e:
                print(e)
                t = QtWidgets.QTableWidgetItem()
                self.setHorizontalHeaderItem(i, t)
                item = self.horizontalHeaderItem(i)

            for a in range(0,len(res[i])):
                self.setItem(i,a,QtWidgets.QTableWidgetItem(str(res[i][a])))
                print(res[i][a])
                # self.setItem(i, 1, QtWidgets.QTableWidgetItem(str(res[i][1])))
            btn=self.GenerateBtn(res[i][1])
            self.setCellWidget(i,len(res[0]),btn)
    def GenerateBtn(self,clientid):
        viewBtn = QtWidgets.QPushButton('查看')
        viewBtn.setStyleSheet(''' text-align : center;
                              background-color : DarkSeaGreen;
                              height : 30px;
                              border-style: outset;
                              color:white;
                              font : 13px; ''')
        viewBtn.clicked.connect(lambda:self.jump(clientid) )
        return viewBtn
    def jump(self,clientid):
        print(clientid)
复制代码

大概效果如下

 

PYQT(6)python3.4+pymssql+pyinstaller 编译exe 出错问题

复制代码
我用的是pyinstaller -F -w来生成单独的exe 文件,总是出现
缺少__mssql什么的文件

百度一下得出解决办法:

http://www.tuicool.com/articles/7RbU3i

后来.py文件加了下面代码

import pymssql
import _mssql
_mssql.__version__

使用pyinstaller打包的时候主要注意这几个包
requests
pymssql
复制代码

PYQT5(7)实现可以拖拽的窗体

复制代码
class MyWindow(QMainWindow):
    def __init__(self):
        super().__init__()

    def mousePressEvent(self, event):
        if event.button() == Qt.Qt.LeftButton:
            self.m_flag = True
            self.m_Position = event.globalPos() - self.pos()  # 获取鼠标相对窗口的位置
            event.accept()
            self.setCursor(QCursor(Qt.Qt.OpenHandCursor))  # 更改鼠标图标

    def mouseMoveEvent(self, QMouseEvent):
        if Qt.Qt.LeftButton and self.m_flag:
            self.move(QMouseEvent.globalPos() - self.m_Position)  # 更改窗口位置
            QMouseEvent.accept()

    def mouseReleaseEvent(self, QMouseEvent):
        self.m_flag = False
        self.setCursor(QCursor(Qt.Qt.ArrowCursor))
复制代码

PYQT5(8)重写Qlabel

复制代码
from PyQt5 import Qt, QtCore
from PyQt5.QtWidgets import QLabel, QFrame
class dialog_label(QLabel):
def init(self,centralwidget):
super().init(centralwidget)
# self.dialog_label.setGeometry(QtCore.QRect(30, 30, 251, 91))
self.setFrameShape(QFrame.Box)
self.setStyleSheet('border-width: 1px;border-style: solid;border-color: rgb(255, 170, 0);')
print('dialog_label')

0人点赞
PYQT5进阶日志
复制代码

PYQT5(9)QLabel 实现右键在点击位置弹出菜单,并绑定事件

复制代码
class MyLabel(QLabel):
    def __init__(self,centralwidget):
#centralwidget 窗体参数
        super().__init__(centralwidget)
        self.setContextMenuPolicy(QtCore.Qt.CustomContextMenu)
        self.customContextMenuRequested.connect(self.rightMenuShow)#开放右键策略
    def rightMenuShow(self, point):
#添加右键菜单
            self.popMenu = QMenu()
            tj=QAction(u'添加', self)
            sc=QAction(u'删除', self)
            xg = QAction(u'修改', self)
            self.popMenu.addAction(tj)
            self.popMenu.addAction(sc)
            self.popMenu.addAction(xg)
#绑定事件
            tj.triggered.connect(self.test)
            sc.triggered.connect(self.test)
            xg.triggered.connect(self.test)
            self.showContextMenu(QtGui.QCursor.pos())
    def test(self):
        print('test')
    def showContextMenu(self, pos):
#调整位置
        '''''
        右键点击时调用的函数
        '''
        # 菜单显示前,将它移动到鼠标点击的位置

        self.popMenu.move( pos)
        self.popMenu.show()
复制代码

PYQT5(10)Qlabel实现捕捉键盘按键事件

复制代码
    def keyPressEvent(self, event):
        if (event.key() == Qt.Qt.Key_P):#P
            if QApplication.keyboardModifiers() == Qt.Qt.ShiftModifier:#shirft
                print("shift + p")
                self.talk_edit.show()
                self.talk_edit.setFocus()
            else:
                print("p")
                print('okokok')
复制代码

https://blog.csdn.net/m0_37828248/article/details/79766580?utm_source=blogxgwz8

PYQT5(11)自适应文字高宽的Qtextedit


相关教程
关于我们--广告服务--免责声明--本站帮助-友情链接--版权声明--联系我们       黑ICP备07002182号