iT邦幫忙

2021 iThome 鐵人賽

DAY 18
0
Software Development

【今年還是不夠錢買psQQ,不如我們用PyQt自己寫一個】系列 第 18

【沒錢買ps,PyQt自己寫】Day 18 / Project 使用 QTimer,自製碼表(計時器) PyQt5 stopwatch DIY

  • 分享至 

  • xImage
  •  

看完這篇文章你會得到的成果圖

前言

這篇我們要來學一個新的東西 QTimer!
QTimer 是獨立於程式運作的計時器,
你可能會想什麼時候會用到?

其實很意外的,真正的程式反而非常經常會需要這個,功能來作為「自動更新」,
我們仔細想想,一般的程式一行接一行執行下去,但萬一有個背景任務,
是隨著時間要定期更新的,我們主程式的流程哪有「突然切換至時間功能,並更新」,這樣的設計
(如果是 step by step 的設計程式,這樣也太難。)

因此我們有 QTimer 的設計,來幫助我們定期更新一些東西,或做一些其他事情。

之後也會介紹 QThread,也同樣的是讓主程式在執行的同時,能夠同時有其他的支線任務可以獨立運行!

之前內容的重點複習 (前情提要)

我們接下來的討論,會基於讀者已經先讀過我 day5 文章 的架構下去進行程式設計
如果還不清楚我程式設計的邏輯 (UI.py、controller.py、start.py 分別在幹麻)
建議先閱讀 day5 文章後再來閱讀此文。

https://www.wongwonggoods.com/python/pyqt5-5/

此篇文章的範例程式碼 github

https://github.com/howarder3/ironman2021_PyQt5_photoshop/tree/main/day18_qtimer

UI 設計部份 (UI.py)

這次我們的重點不是放在 UI 設計,我們簡單拉一個作為結果顯示用的 Qlabel 即可。

轉換成 UI.py

一樣的編譯指令,我們加上 -x (也可不加),
我們就可以先檢視看看轉換後的視窗是不是跟我們想像的一樣。

轉換 day18.ui -> UI.py

pyuic5 -x day18.ui -o UI.py

執行看看 UI.py 畫面是否如同我們想像

一樣,這程式只有介面 (視覺上的呈現),沒有任何互動功能

  • 看看我們製作出來的介面
python UI.py

controller 設計部份 (controller.py)

QTimer 的使用方式很簡單,主要我們需要設定一個 timeout 時間,
每經過一次 timeout,我們的程式就會做一次指定的事情

from PyQt5 import QtCore 
from PyQt5.QtWidgets import QMainWindow, QFileDialog
from PyQt5.QtCore import QTimer

import time
import os

from UI import Ui_MainWindow

class MainWindow_controller(QMainWindow):
    def __init__(self):
        super().__init__() # in python3, super(Class, self).xxx = super().xxx
        self.ui = Ui_MainWindow()
        self.ui.setupUi(self)
        self.setup_control()

    def setup_control(self):
        self.timer=QTimer() # init QTimer
        self.timer.timeout.connect(self.run) # when timeout, do run one
        self.timer.start(1) # start Timer, here we set '1ms' while timeout one time
        self.time_counter = 0  # init time counter # for testing: 3599000

    def run(self):
        self.ui.label.setText(str(self.set_time_counter_format(self.time_counter))) # show time_counter (by format)
        self.time_counter += 1 # time_counter + 1


    def set_time_counter_format(self, time_counter):
        ms = time_counter % 1000
        total_sec = max(0, (time_counter - ms)//1000)
        hour = max(0, total_sec//3600)
        minute = max(0, total_sec//60 - hour * 60)
        sec = max(0, (total_sec - (hour * 3600) - (minute * 60)))
        return f"{hour}:{minute:0>2}:{sec:0>2}.{ms:0>3}"

我們先來看一下,最主要的設定 QTimer 部分

設定主程式 setup_control()

  • self.timer=QTimer():初始化 QTimer
  • self.timer.timeout.connect(self.run):當 QTimer 每經過一次 timeout,執行 run 的動作
  • self.timer.start(1):開始 QTimer,這裡的"1" 代表的是 1ms,所以我們每 1ms 就會 timeout 一次
  • self.time_counter = 0:我們初始化計數器,從 0 開始計數。

設定 run(),每 timeout 就執行一次

  • self.ui.label.setText(str(self.set_time_counter_format(self.time_counter)))):設定格式並顯示時間
  • self.time_counter += 1:計數器 +1 (因為 timeout 一次)

設定時間格式 set_time_counter_format()

  • 這邊就是我自己計算的公式了,有幾個計算上的重點
  • max(0, x):避免計算結果 < 0 導致顯示負數,我們直接給予 0 這個值
  • f"{hour}:{minute:0>2}:{sec:0>2}.{ms:0>3}": f-string 的 格式設定,我們抓住「:」冒號後的就是格式設定內容,「0」表示補0,「>」表示靠右,「2(3)」表示總共2(3)位。

執行結果

照我們 day5 的程式架構,我們執行

python start.py

於是我們就這樣完成了我們的碼表 (stopwatch)。

此外,我們可以透過更改 counter 的初始值來修正程式錯誤
例如依照我們上面的算式,我們可以設定初始值為 "3599000",我們可以同時驗證「小時」、「分鐘」、「秒」、能不能正確增加 (因為從約 0:59:50開始)

Reference


★ 本文也同步發於我的個人網站(會有內容目錄與顯示各個小節,閱讀起來更流暢):【PyQt5】Day 18 / Project 使用 QTimer,自製碼表(計時器) PyQt5 stopwatch DIY


上一篇
【沒錢買ps,PyQt自己寫】Day 17 / Project 製作標註 roi 工具, 開始導入 OpenCV 作為繪圖引擎, 在圖上畫點並顯示座標
下一篇
【沒錢買ps,PyQt自己寫】Day 19 - 使用 QProgressBar,製作進度條的功能
系列文
【今年還是不夠錢買psQQ,不如我們用PyQt自己寫一個】30
圖片
  直播研討會
圖片
{{ item.channelVendor }} {{ item.webinarstarted }} |
{{ formatDate(item.duration) }}
直播中

尚未有邦友留言

立即登入留言