《快速掌握PyQt5》第十章 定时器QTimer和进度条QProgressBar

《快速掌握PyQt5》第十章 定时器QTimer和进度条QProgressBar

《快速掌握PyQt5》 专栏已整理成书出版,书名为 《PyQt编程快速上手》 ,详情请见该 链接 。感谢大家一直以来的支持!祝大家PyQt用得越来越顺!

当我们要让程序定期去执行某函数的时候,QTimer就派上用场了,比如一个游戏程序,它通常会定期去调用一个函数来进行更新操作。而进度条可以用来显示某项任务的进度,从而让用户界面更加友好。

我们通常将将QTimer和QProgressBar一起搭配使用,所以本章就一起介绍了。

10.1 QTimer

以下这个程序中,按钮被点击后,QLabel显示的数字会不断增加:

import sys
from PyQt5.QtCore import QTimer, Qt
from PyQt5.QtWidgets import QApplication, QWidget, QPushButton, QLabel, QVBoxLayout
class Demo(QWidget):
    def __init__(self):
        super(Demo, self).__init__()
        self.label = QLabel('0', self)                          # 1
        self.label.setAlignment(Qt.AlignCenter)                 
        self.step = 0                                           # 2
        self.timer = QTimer(self)                               # 3
        self.timer.timeout.connect(self.update_func)
        self.ss_button = QPushButton('Start', self)             # 4
        self.ss_button.clicked.connect(self.start_stop_func)
        self.v_layout = QVBoxLayout()
        self.v_layout.addWidget(self.label)
        self.v_layout.addWidget(self.ss_button)
        self.setLayout(self.v_layout)
    def start_stop_func(self):                      
        if not self.timer.isActive():
            self.ss_button.setText('Stop')
            self.timer.start(100)
        else:
            self.ss_button.setText('Start')
            self.timer.stop()
    def update_func(self):
        self.step += 1
        self.label.setText(str(self.step))
if __name__ == '__main__':
    app = QApplication(sys.argv)
    demo = Demo()
    demo.show()
    sys.exit(app.exec_())

1. 首先实例化一个QLabel,并将文本设为0。setAlignment(Qt.AlignCenter)可以让QLabel控件在窗口中居中显示,而之前我们是通过addStretch(int)方法来让一个控件在布局中居中的,显然通过setAlignment(Qt.AlignCenter)方法更加方便:

self.h_layout.addStretch(1)
self.h_layout.addWidget(self.label)
self.h_layout.addStretch(1)

2. step变量用于计数,QLabel控件显示的就是这里的step,程序会通过QTimer来不断增加step的值;

3. 其次实例化一个QTimer,并将timeout信号连接到自定义的槽函数update_func()上:

def update_func(self):
    self.step += 1
    self.label.setText(str(self.step))

每次调用该槽函数就会将step值加1,并且用QLabel显示当前值;

4. 最后我们实例化一个QPushButton按钮来控制定时器的启动的停止,连接的自定义的槽函数如下:

def start_stop_func(self):                      
    if not self.timer.isActive():
        self.ss_button.setText('Stop')
        self.timer.start(100)
    else:
        self.ss_button.setText('Start')
        self.timer.stop()

在槽函数中通过isActive()方法来判断定时器是否处于激活状态,若没有激活,则将按钮文字变成Stop并通过start(100)方法来启动定时器,100表示100毫秒,也就是说每过0.1秒,定时器就会触发timeout信号,并执行update_func()槽函数;若已经处于激活状态,则将按钮文字变回Start并通过stop()方法停止定时器。


有些小伙伴可能想在触发timeout信号后只调用一次update_func(),那么我们可以通过setSingleShot(True)方法来设置。


运行截图如下:

​10.2 QProgressBar

这里我们将10.1章节中的QLabel用QProgressBar来代替:

import sys
from PyQt5.QtCore import Qt, QTimer
from PyQt5.QtWidgets import QApplication, QWidget, QProgressBar, QPushButton, QHBoxLayout, QVBoxLayout
class Demo(QWidget):
    def __init__(self):
        super(Demo, self).__init__()
        self.progressbar = QProgressBar(self)                   # 1
        # self.progressbar.setOrientation(Qt.Vertical)          
        self.progressbar.setMinimum(0)                          # 2
        self.progressbar.setMaximum(100)
        # self.progressbar.setRange(0, 100)
        self.step = 0                                           # 3
        self.timer = QTimer(self)                               # 4
        self.timer.timeout.connect(self.update_func)
        self.ss_button = QPushButton('Start', self)             # 5
        self.ss_button.clicked.connect(self.start_stop_func)
        self.reset_button = QPushButton('Reset', self)          # 6
        self.reset_button.clicked.connect(self.reset_func)
        self.h_layout = QHBoxLayout()
        self.v_layout = QVBoxLayout()
        self.h_layout.addWidget(self.ss_button)
        self.h_layout.addWidget(self.reset_button)
        self.v_layout.addWidget(self.progressbar)
        self.v_layout.addLayout(self.h_layout)
        self.setLayout(self.v_layout)
    def start_stop_func(self):
        if self.ss_button.text() == 'Start':
            self.ss_button.setText('Stop')
            self.timer.start(100)
        else:
            self.ss_button.setText('Start')
            self.timer.stop()
    def update_func(self):
        self.step += 1
        self.progressbar.setValue(self.step)
        if self.step >= 100:
            self.ss_button.setText('Start')
            self.timer.stop()
            self.step = 0
    def reset_func(self):
        self.progressbar.reset()
        self.ss_button.setText('Start')
        self.timer.stop()
        self.step = 0
if __name__ == '__main__':
    app = QApplication(sys.argv)
    demo = Demo()
    demo.show()
    sys.exit(app.exec_())

1. 实例化一个QProgressBar,默认是水平的,但是我们可以通过setOrientation(Qt.Vertical)方法来让进度条垂直显示;

2. 通过setMinimum()和setMaximum()方法来设置范围,也可以单单用setRange()方法来实现,这里我们将范围设为0-100;

3. 这里的step变量用于计数,之后QProgressBar会将值设为step;

4. 实例化一个QTimer,并将timeout信号连接到update_func()槽函数上:

def update_func(self):
    self.step += 1
    self.progressbar.setValue(self.step)
    if self.step >= 100:
        self.ss_button.setText('Start')
        self.timer.stop()
        self.step = 0

每次触发timeout都会调用该槽函数,在这里我们将step值加1,并将progressbar的值设为step,当step值达到pregress的最大值时(也就是说进度条达到100%),将按钮文本重新设为Start,停止定时器并将step值重设为0;

5. 实例化一个QPushButton按钮来控制QTimer的启动与停止,这里将它的clicked信号和start_stop_func()槽函数连接起来:

def start_stop_func(self):
    if self.ss_button.text() == 'Start':
        self.ss_button.setText('Stop')
        self.timer.start(100)
    else: