PyQT5学习之旅 1 如何自定义控件,入门做一个上位电脑串口调试软件,全部开源。(附带源码)

在这里插入图片描述

一、前言

    在物联网行业摸爬滚了四年,从移动开发、嵌入式开发、云平台开发到微信小程序公众号开发,似乎已经有了整体的流程的深度认识,但这半年似乎又没有新的输出,还在啃老本,我大胆又学了一门语言Python,对于我这种有面向对象编程基础的人来说,上手非常简单,但按照我一贯的学习方法:

    运筹帷幄之中,决胜千里之外。

    先要规划自己要实现什么功能的软件,再去针对性地翻资料,难免怕坑难免等原因,可以投资买视频学,但是我也从来不指望任何人免费能带我,能有稍些的指点我便心满意足,别指望他人能给予你什么,而是你要以一物换一物去等价交换,终究会明白,别人的屋檐再大,都不如自己有把伞。

    所以,大家一个点赞/转发我的开源文章,我非常感恩。每一个不曾学习的日子,都是对生命的辜负,但行好事,莫问前程;


    所以,我也开源我上月春节假期学习到的PyQt5,并且基于PyQt5打造的一款跨平台的电脑串口调试助手(支持windowMacOSLinux),代码和安装包在文章末尾开源地址;

    集成了市面上大多串口工具的功能,包括:

  • 自定义波特率
  • 自动换行
  • RTS 控
  • DTR 控
  • Hex 发送和接收

二、开发的必备工具

  • Python 3.8 环境
  • PyCharm 2020.1.1 x64
  • QT Designer 5.2

     对于如何选择一款开发工具,我也是挑了蛮久的,还是选择熟悉强大的IDEA系列软件,再配合QT Designer使用起来,非常Nice。下面分享下,我的快捷键,能使你的开发效率加倍。

2.1 PyCharm 如何集成 QT Designer

UI代码转可视化

  • 终端: Python38-32\Lib\site-packages\pyqt5_tools\Qt\bin\designer.exe
    在这里插入图片描述

可视转化UI代码

  • 终端: Python38-32\Lib\site-packages\pyqt5_tools\Qt\bin\designer.exe
  • 参数: -m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py
    在这里插入图片描述

打包成 exe 软件:

  • 终端: D:\Python\QT5\Demo\Scripts\pyinstaller.exe
  • 参数: -F -w $FileName$
    在这里插入图片描述

2.2、引进自定义控件

     注意头文件引入路径,以及其父类;

在这里插入图片描述

移除此控件为自定义控件

在这里插入图片描述

三、代码结构说明

3.1 串口驱动应用层封装

  • 包括串口异常中断处理;
  • 提供上层应用接口有配置、发送和接收;
  • 子线程和信号槽应用实现无缝读取串口通讯接收到的数据给上位机;
class UartRecieveThread(QThread):

    def __init__(self, run):
        super(UartRecieveThread, self).__init__()
        self.runfun = run

    def run(self):
        self.runfun()


class UartSerial(QObject):
    # 定义一个信号变量,1个参数
    signalRecieve = pyqtSignal(object)

    CODE_RECIEVE, CODE_DISCONNECT = 0, 1

    def __init__(self):
        super(UartSerial, self).__init__()
        self.mThread = UartRecieveThread(self.data_receive)  # 创建线程
        self.mSerial = serial.Serial()
        self.data_num_received = 0

    def init(self, _port="/dev/ttyUSB0", _baudrate=115200, _bytesize=8, _stopbits="N", _parity=serial.PARITY_NONE):
        self.mSerial.port = _port
        self.mSerial.baudrate = _baudrate
        self.mSerial.bytesize = _bytesize
        self.mSerial.stopbits = _stopbits
        self.mSerial.parity = _parity
        self.data_num_received = 0

    def is_port_open(self, _port_name, _baudrate):
        try:
            self.mSerial.port = _port_name
            self.mSerial.baudrate = _baudrate
            return not self.mSerial.isOpen()
        except Exception:
            return False

    # 关闭串口
    def port_close(self, _port_name, _baudrate):
        self.mSerial.port = _port_name
        self.mSerial.baudrate = _baudrate
        try:
            self.mSerial.close()
        except:
            return False
        self.mThread.quit()
        return True

    # 串口检测
    def get_all_port(self):
        # 检测所有存在的串口,将信息存储在字典中
        self.port_list_name = []
        port_list = list(serial.tools.list_ports.comports())
        i = 0

        if len(port_list) <= 0:
            return []
        else:
            for port in port_list:
                i = i + 1
                self.port_list_name.append(port[0])

        return self.port_list_name

    # 打开串口
    def try_port_open(self, _port, _baudrate=115200):
        self.mSerial.port = _port
        self.mSerial.baudrate = _baudrate
        try:
            self.mSerial.open()
        except:
            return False
        if not self.mThread.isRunning():
            self.mThread.start()
        return True

    def set_rts(self, IsTrue):
        self.mSerial.setRTS(IsTrue)

    def set_dts(self, IsTrue):
        self.mSerial.setDTR(IsTrue)

    def get_rts(self):
        return self.mSerial.rts

    def get_dts(self):
        return self.mSerial.dtr

    def set_bytesize(self, _bytesize):
        self.mSerial.bytesize = _bytesize

    def set_parity(self, _parity):
        self.mSerial.parity = _parity

    def set_stopbits(self, _stopbits):
        self.mSerial.stopbits = _stopbits

    # 接收数据
    def data_receive(self):
        while True:
            data = {}
            try:
                num = self.mSerial.inWaiting()
            except:
                self.mSerial.close()
                data['code'] = self.CODE_DISCONNECT
                data['data'] = 0
                data['length'] = 0
                self.signalRecieve.emit(data)
                return None
            if num > 0:
                if self.mSerial.isOpen():
                    buff = self.mSerial.read(num)
                    data['code'] = self.CODE_RECIEVE
                    data['data'] = buff
                    data['length'] = len(buff)
                    self.signalRecieve.emit(data)
            time.sleep(0.1)

    # 设置回调函数
    def setCallBack(self, funtion):
        self.signalRecieve.connect(funtion)

    # send data
    def send_data(self, buff="", isHexSend=False, _port="", _baudrate=115200):
        if buff != "":
            # 非空字符串
            if isHexSend:
                # hex发送
                buff = buff.strip()
                send_list = []
                while buff != '':
                    try:
                        num = int(buff[0:2], 16)
                    except ValueError:
                        return None
                    buff = buff[2:].strip()
                    send_list.append(num)
                buff = bytes(send_list)
            num = self.mSerial.write(buff)

3.2 自定义控件

在实现某些功能时候,发现自带的控件无法满足需求时候,我们只能自定义控件,我重写了几个控件:

1 、重写 QComboBox

为了实现能自定义波特率,比如用户输入74880;

class MyQComBox(QComboBox):
    popupAboutToBeShown = pyqtSignal()  # 创建一个信号

    def showPopup(self):  # 重写showPopup函数
        super(MyQComBox, self).showPopup()
        self.popupAboutToBeShown.emit()  # 发送信号

2、重写 QWidget

为了能够全局监听键盘按键,实现一键清空接受栏;

class MineWidget(QWidget):

    # 定义一个信号变量,1个参数
    signalMineWidget = pyqtSignal(object)

    def set_connect_key_press(self, fun):
        self.signalMineWidget.connect(fun)

    # 检测键盘回车按键
    def keyPressEvent(self, event):
        self.signalMineWidget.emit(event.key())

    def mousePressEvent(self, event):
        self.signalMineWidget.emit(event.button())

3.3 接收数据并显示

这里要特别说明的是,在解码时候,特要指明 utf-8编码,以及带上 ignore表示接受解码失败异常;

def at_callback_handler(obj):
    if obj['code'] == 1:
        if ui.bt_open_off_port.text() != '打开串口':
            QMessageBox.critical(MainWindow, '错误信息', '串口异常断开!')
            # 面板可操作
            checkoutPortStatus(True)
            ui.bt_open_off_port.setText('打开串口')
            refreshPort()
    else:
        buff = (obj['data'])
        now_time = datetime.now()  # 获取当前时间
        # new_time = now_time.strftime('[%H:%M:%S]')  # 打印需要的信息,依次是年月日,时分秒,注意字母大小写
        if ui.checkBox_show_hex.checkState():
            out_s = ''
            for i in range(0, len(buff)):
                out_s = out_s + '{:02X}'.format(buff[i]) + ' '

            if ui.checkBox_show_add_ctrl.checkState():
                ui.textBrowserShow.append(out_s)
            else:
                ui.textBrowserShow.insertPlainText(out_s)
            ui.textBrowserShow.moveCursor(QTextCursor.End)
        else:
            try:
                if ui.checkBox_show_add_ctrl.checkState():
                    ui.textBrowserShow.append(buff.decode('utf-8', 'ignore'))
                else:
                    ui.textBrowserShow.insertPlainText(buff.decode('utf-8', 'ignore'))

                ui.textBrowserShow.moveCursor(QTextCursor.End)
            except:
                # 乱码显示
                pass

四、效果

在这里插入图片描述


关注微信公众号 徐宏blog 发送 210317 以获取全部代码


另外,不要把我的博客作为学习标准,我的只是笔记,难有疏忽之处,如果有,请指出来,也欢迎留言哈!

  • 玩转esp8266带你飞、加群QQ群,不喜的朋友勿喷勿加:434878850
  • 个人邮箱:xuhongv@yeah.net 24小时在线,有发必回复!
  • esp8266源代码学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp8266
  • esp32源代码学习汇总(持续更新,欢迎star):https://github.com/xuhongv/StudyInEsp32
  • 关注下面微信公众号二维码,干货多多,第一时间推送!

在这里插入图片描述

评论 7
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

半颗心脏

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值