PyQt事件机制及其应用

一、实例前置

一个小闹钟应用

创建主窗口类

首先我们创建了一个名为AlarmClock的类,它继承自QMainWindow。这个类将包含我们的GUI组件和逻辑。

python 复制代码
from Alarm_clock import Ui_MainWindow

class AlarmClock(QMainWindow):
    def __init__(self):
        super().__init__()

        # 初始化 UI
        self.ui = Ui_MainWindow()
        self.ui.setupUi(self)

二、创建UI界面

预先设计好的UI文件Alarm_clock.ui

这里,我们使用pyside6-uic把一个预先设计好的UI文件Alarm_clock.ui转成py格式,然后通过from import导入,这个文件定义了我们的界面布局

三、QtWidgets

在 PyQt 中,QtWidgets是一个非常重要的模块,它提供了大量用于构建图形用户界面(GUI)的部件(Widgets)类。这些部件包括但不限于按钮(QPushButton)、标签(QLabel)、文本框(QLineEdit)、布局管理器(如QVBoxLayoutQHBoxLayout)等,是创建交互式 GUI 应用程序的基础构建块。

①QApplication

QApplication是整个应用程序的核心管理类。它负责管理应用程序的生命周期,包括初始化、事件循环处理、资源管理等诸多关键事务。一个 PyQt 应用程序通常需要创建一个QApplication对象作为起点来启动整个应用程序的运行。

python 复制代码
     import sys
     from PyQt6.QtWidgets import QApplication

     app = QApplication(sys.argv)
     # 在这里可以通过sys.argv来获取命令行参数,并根据参数进行应用程序的初始化设置

②QMainWindow

在 PyQt 中,QMainWindow是一个用于创建主窗口类型应用程序的关键类。它提供了一个完整的框架来构建包含菜单栏、工具栏、状态栏和中心部件等典型布局的主窗口应用程序,为复杂的图形用户界面(GUI)应用程序提供了一种标准的组织结构。

python 复制代码
     from PyQt6.QtWidgets import QMainWindow, QApplication
     import sys

     app = QApplication(sys.argv)
     main_window = QMainWindow()
     main_window.show()
     sys.exit(app.exec())

四、PyQt时间事件处理

①QTimer

在 PyQt 中,QTimer是一个用于实现定时器功能的类。它允许你在指定的时间间隔后触发一个信号,这个信号可以连接到一个槽函数,从而实现周期性的操作或者在延迟一段时间后执行某个操作。这在很多场景下非常有用,比如实现动画效果、定期更新数据显示、自动保存功能等。

例:创建用于检查闹钟的定时器

python 复制代码
self.timer = QTimer(self)
self.timer.timeout.connect(self.check_alarm)
self.alarm_time = None  # 初始化闹钟时间

例:创建滚动动画的定时器

python 复制代码
self.scroll_timer = QTimer(self)
self.scroll_timer.timeout.connect(self.scroll_footer_label)

②QTime

在 PyQt 中,QTime类主要用于表示时间。它提供了一种方便的方式来处理时间相关的操作,如获取当前时间、设置特定时间、比较时间大小、进行时间的加减运算等。这在很多需要涉及时间处理的应用场景中非常有用,例如日程安排应用、闹钟应用、时间记录工具等。

例:设置初始时间显示

python 复制代码
self.ui.timeLabel.setText(QTime.currentTime().toString("HH:mm:ss"))

例: 判断用户输入的时间格式

python 复制代码
self.alarm_time = QTime.fromString(input_time, "HH:mm:ss")
if not self.alarm_time.isValid():
    self.alarm_time = QTime.fromString(input_time, "H:m:s")
if not self.alarm_time.isValid():
    self.alarm_time = QTime.fromString(input_time, "HH:mm:ss")
if not self.alarm_time.isValid():
    self.alarm_time = QTime.fromString(input_time, "H:m:s")
if not self.alarm_time.isValid():
    self.ui.Alarm_display.setText("时间格式无效!")
        return

五、PyQt事件机制思维导图

PyQt事件机制思维导图

六、完整代码

python 复制代码
import sys
from PySide6.QtWidgets import QApplication, QMainWindow, QLabel
from PySide6.QtCore import QTimer, QTime, Qt
from PySide6.QtGui import QPainter, QLinearGradient, QColor, QFont, QMouseEvent, QIcon
from PySide6.QtMultimedia import QMediaPlayer, QAudioOutput
from Alarm_clock import Ui_MainWindow

class GradientTextLabel(QLabel):
    def __init__(self, text="", parent=None):
        super().__init__(text, parent)
        self.setFont(QFont("Arial", 40, QFont.Weight.Bold))
        self.setStyleSheet("background-color: transparent;")  # 背景透明

    def paintEvent(self, event):
        painter = QPainter(self)
        painter.setRenderHint(QPainter.Antialiasing)

        # 创建一个线性渐变(红到紫,带透明度)
        gradient = QLinearGradient(0, 0, self.width(), 0)
        gradient.setColorAt(0.0, QColor(100, 0, 200, 200))  # 紫色,透明度78%
        gradient.setColorAt(1.0, QColor(255, 0, 0, 200))  # 红色,透明度78%

        # 设置文字渐变
        pen = painter.pen()
        pen.setBrush(gradient)  # 使用渐变作为笔刷
        pen.setWidth(2)  # 设置笔宽度
        painter.setPen(pen)

        # 绘制文字
        painter.drawText(self.rect(), Qt.AlignCenter, self.text())
        painter.end()


class AlarmClock(QMainWindow):
    def __init__(self):
        super().__init__()

        # 初始化 UI
        self.ui = Ui_MainWindow()
        self.ui.setupUi(self)

        # 设置应用图标
        self.setWindowIcon(QIcon("G:\longz\人工智能作业\PythonQt\闹钟界面/蓝色风信子_爱给网_aigei_com.ico"))

        # 开始隐藏footerLabel
        self.ui.footerLabel.hide()

        # 设置时间标签为渐变文字标签
        self.gradient_label = GradientTextLabel(parent=self)
        self.gradient_label.setGeometry(self.ui.timeLabel.geometry())  # 保持位置一致
        self.ui.timeLabel.deleteLater()  # 删除原来的 timeLabel
        self.ui.timeLabel = self.gradient_label  # 替换为渐变标签

        # 设置初始时间
        self.ui.timeLabel.setText(QTime.currentTime().toString("HH:mm:ss"))

        # 固定窗口大小为 800x600
        self.setFixedSize(800, 600)

        # 启动时间更新定时器(始终更新)
        self.time_timer = QTimer(self)
        self.time_timer.timeout.connect(self.update_time_display)
        self.time_timer.start(1000)  # 每秒更新一次

        # 纯净窗口
        self.setWindowFlag(Qt.WindowType.FramelessWindowHint)

        # 定时器
        self.timer = QTimer(self)
        self.timer.timeout.connect(self.check_alarm)
        self.alarm_time = None

        # 滚动动画
        self.scroll_timer = QTimer(self)
        self.scroll_timer.timeout.connect(self.scroll_footer_label)

        self.footer_label_start_pos = self.ui.footerLabel.geometry().x()  # 起始位置
        self.footer_label_end_pos = -self.ui.footerLabel.width() + 800  # 结束位置

        # 音乐播放器
        self.media_player = QMediaPlayer(self)
        self.audio_output = QAudioOutput(self)
        self.media_player.setAudioOutput(self.audio_output)
        self.media_player.setSource("G:/longz/人工智能作业/PythonQt/闹钟界面/Scott Lloyd Shelly - Terraria Soundtrack - 01 Overworld Day.mp3")

        # 连接按钮点击事件
        self.ui.pushButton.clicked.connect(self.handle_button_click)

    # 处理按钮点击事件
    def handle_button_click(self):
        if self.ui.pushButton.text() == "启动":
            self.start_alarm_timer()
        elif self.ui.pushButton.text() == "停止":
            self.stop_alarm()

    # 停止闹钟
    def stop_alarm(self):
        # 停止定时器、滚动动画和音乐播放
        self.timer.stop()
        self.scroll_timer.stop()
        self.media_player.stop()
        self.ui.footerLabel.hide()

        # 重置按钮和状态显示
        self.ui.Alarm_display.setText("闹钟已停止")
        self.ui.pushButton.setText("启动")

    # 启动闹钟
    def start_alarm_timer(self):
        input_time = self.ui.Alarm_input.text()
        if not input_time:
            self.ui.Alarm_display.setText("请输入时间!")
            return

        # 支持多种时间格式
        self.alarm_time = QTime.fromString(input_time, "HH:mm:ss")
        if not self.alarm_time.isValid():
            self.alarm_time = QTime.fromString(input_time, "H:m:s")
        if not self.alarm_time.isValid():
            self.alarm_time = QTime.fromString(input_time, "HH:mm:ss")
        if not self.alarm_time.isValid():
            self.alarm_time = QTime.fromString(input_time, "H:m:s")
        if not self.alarm_time.isValid():
            self.ui.Alarm_display.setText("时间格式无效!")
            return

        # 格式化时间并显示
        formatted_time = self.alarm_time.toString("HH:mm:ss")
        self.ui.Alarm_input.setText(formatted_time)

        # 启动定时器
        self.timer.start(1000)
        self.ui.Alarm_display.setText("闹钟已启动")
        self.ui.pushButton.setText("停止")

    # 更新时间显示
    def update_time_display(self):
        current_time = QTime.currentTime()
        self.ui.timeLabel.setText(current_time.toString("HH:mm:ss"))

    # 检查闹钟是否到达时间
    def check_alarm(self):
        current_time = QTime.currentTime()
        if self.alarm_time and current_time >= self.alarm_time and current_time <= self.alarm_time.addSecs(600):
            self.ui.Alarm_display.setText("闹钟响了!")
            self.ui.footerLabel.show()

            if not self.scroll_timer.isActive():
                self.scroll_timer.start(30)  # 滚动动画
            if self.media_player.mediaStatus() != QMediaPlayer.MediaStatus.LoadingMedia:
                self.media_player.play()
        else:
            # 停止定时器、滚动动画和音乐播放
            self.scroll_timer.stop()
            self.media_player.stop()
            self.ui.footerLabel.hide()
            # 状态显示
            self.ui.Alarm_display.setText("闹钟已结束")

    # 滚动 footerLabel
    def scroll_footer_label(self):
        current_geometry = self.ui.footerLabel.geometry()
        x = current_geometry.x() - 5  # 每次向左移动 5 像素

        # 如果超出左边界,将其重新设置到右边界
        if x < self.footer_label_end_pos:
            x = self.footer_label_start_pos

        self.ui.footerLabel.setGeometry(x, current_geometry.y(), current_geometry.width(), current_geometry.height())

    def mousePressEvent(self, event: QMouseEvent):
        self.px = event.pos().x()
        self.py = event.pos().y()

    def mouseMoveEvent(self, event: QMouseEvent):
        if event.buttons() == Qt.MouseButton.LeftButton:
            self.move(int(event.globalPosition().x()) - self.px, int(event.globalPosition().y()) - self.py)


if __name__ == "__main__":
    app = QApplication(sys.argv)
    window = AlarmClock()
    window.setWindowFlag(Qt.WindowType.FramelessWindowHint)
    window.setAttribute(Qt.WidgetAttribute.WA_TranslucentBackground)
    window.show()
    sys.exit(app.exec())
相关推荐
Wallace Zhang1 小时前
STM32F103_LL库+寄存器学习笔记06 - 梳理串口与串行发送“Hello,World“
笔记·stm32·学习
爱的叹息1 小时前
Spring 及 Spring Boot 条件化注解(15个)完整列表及示例
spring boot·python·spring
计算机毕设定制辅导-无忧学长1 小时前
从入门到精通:HTML 项目实战中的学习进度(一)
前端·学习·html
无闻墨客2 小时前
数据可视化(matplotlib)-------图表样式美化
python·信息可视化·matplotlib
www_pp_2 小时前
使用Python和OpenCV进行指纹识别与验证
开发语言·python·opencv
HKkuaidou2 小时前
外星人入侵(python设计小游戏)
python·pygame
豌豆花下猫2 小时前
Python 潮流周刊#95:像人类一样使用计算机(摘要)
后端·python·ai
测试杂货铺2 小时前
白盒测试用例的设计
自动化测试·软件测试·python·测试工具·职场和发展·单元测试·测试用例
java资料站3 小时前
linux ntpdate 时间矫正
linux·运维·python
唐古乌梁海3 小时前
【Django】教程-3-数据库相关介绍
python·django