在线时间 1630 小时 最后登录 2024-1-29 注册时间 2017-5-16 听众数 82 收听数 1 能力 120 分 体力 563431 点 威望 12 点 阅读权限 255 积分 174252 相册 1 日志 0 记录 0 帖子 5313 主题 5273 精华 3 分享 0 好友 163
TA的每日心情 开心 2021-8-11 17:59
签到天数: 17 天
[LV.4]偶尔看看III
网络挑战赛参赛者
网络挑战赛参赛者
自我介绍 本人女,毕业于内蒙古科技大学,担任文职专业,毕业专业英语。
群组 : 2018美赛大象算法课程
群组 : 2018美赛护航培训课程
群组 : 2019年 数学中国站长建
群组 : 2019年数据分析师课程
群组 : 2018年大象老师国赛优
超全!Python图形界面框架PyQt5使用指南!
N/ s5 u( b- @. b1 I 9 v2 \0 n' f, ], w
使用Python开发图形界面的软件其实并不多,相对于GUI界面,可能Web方式的应用更受人欢迎。但对于像我一样对其他编程语言比如C#或WPF并不熟悉的人来说,未必不是一个好的工具。
# p9 L2 g; w1 \( S ; @5 k$ r5 S+ { T8 [" }
常见GUI框架! u3 t/ v7 X3 z7 s
3 o. c$ ?8 ]0 `1 y) M+ w" g) g. Z( G PyQt5:Qt是一个跨平台的 C++图形用户界面库。QT一度被诺基亚拥,后出售给芬兰的软件公司Digia Oyj。PyQt5是基于Digia公司Qt5的Python接口,由一组Python模块构成。PyQt5本身拥有超过620个类和6000函数及方法。在可以运行于多个平台,包括:Unix, Windows, and Mac OS。
$ M2 R, ^- i8 Z* J( v' ? * v3 ~# Y) @ B# H
Pyside6:Pyside是QT公司官方提供的Python包,上一版本为Pyside2,对应的是QT5,最新版命名规则进行了调整,更改为Pyside6,对应的是QT6版本。由于官方出品的比较看好,缺点是发布比较晚,网上的资料没有PyQt5多。1 l5 o. \0 r2 G, B' [5 M
* y, \* w! Y. E8 N+ a/ J6 m* S4 H
Tkinter:Python内置的GUI框架,使用TCL实现,Python中内嵌了TCL解释器,使用它的时候不用安装额外的扩展包,直接import,跨平台。不足之处在于UI布局全靠代码实现,只有15种常用部件,显示效果简陋。+ A7 n D7 f, U R/ V! S
% O* R- ]* Y; q PySimpleGUI:PySimpleGUI 是 Tkinter 一层包装。使用 PySimpleGUI 实现自定义 GUI 所需的代码量要比使用 Tkinter 直接编写相同的 GUI 要少得多。
! N3 l& Z8 o/ X ' Z# D: ^$ |9 x/ s
WxPython:wxPython是Python语言对流行的wxWidgets跨平台GUI工具库的绑定。用得比较广泛,跨平台,C++编写,文档少,用户可能就需要根据编程内容对不同平台中的GUI代码做一些调整。遇到问题不好解决,代码布局控件,不直观。
" w1 {3 t9 w+ F$ E, Q, U! n) X" Q + R5 ?. H; m6 m1 Q
Wax:基于wxPython ,为克服wxPython的问题而制作的一个包。
, o% k) U+ Q6 h9 C8 d: l
; b* s* D2 D) i0 W9 f Kivy:主要针对多点触控程序,智能手机平板等,也可以在没有触屏功能的系统上,全平台支持(Windows, Linux, Mac OS X, Android and iOS.)使用Python和cython编写,中文支持差,需要自己下载中文库并且制定路径。, L0 d6 Q% q. G- ~) @0 P
" m) F" V: l& }3 l- M* c7 U BeeWare:Write once. Deploy everywhere.需要与Kivy配合使用。
8 T# A. M |1 J' L$ v + M' k+ s3 ~% n4 \+ E
Toga:一个使用Python开发原生APP的GUI工具包。Toga由一个具有共享接口的基础组件库组成,以简化与平台无关的GUI开发。Toga适用于Mac OS、Windows、Linux(GTK)以及Android和iOS等移动平台。! m6 t9 Q0 F3 K9 D W
- h6 \2 F/ B. W6 Y$ ?6 K* {2 |* [
Eel:一个轻量的 Python 库,用于制作简单的类似于 Electron(但是比它更轻量) 的离线 HTML/JS GUI 应用程序,并具有对 Python 功能(capabilities)和库的完全访问权限。
Z& W9 y1 ~1 d4 i5 ^
' o# Q0 O6 t G L- A4 T Flexx:一个纯 Python 工具包,用来创建图形化界面应用程序。其使用 Web 技术进行界面的渲染。你可以用 Flexx 来创建桌面应用,同时也可以导出一个应用到独立的 HTML 文档。因为使用纯 Python 开发,所以 Flexx 是跨平台的。只需要有 Python 和浏览器就可以运行。
9 F" I2 M1 Q; J
( y) G; u, i/ l9 D% T' n/ x pywebview是围绕 webview 组件的轻量型跨平台包装器(wrapper),它允许在其自己的本机 GUI 窗口中显示 HTML 内容。它使您可以在桌面应用程序中使用 Web 技术,同时尽最大可能隐藏使用浏览器构建GUI的事实。% R" ?7 U: ?5 H/ a) z# y
0 x7 e7 V4 ?- k4 p+ G$ ~% @ enaml:一种能够让你用最小的努力就可以实现高质量GUI界面的的Python框架,也是一种独特的编程语言。enaml将声明性语言与基于约束的布局系统结合在一起,使用户可以轻松地定义灵活布局的UI。enaml应用程序可以在任何支持Python和Qt的平台上运行。9 G0 `2 Z0 n2 M$ S
, D& C& m$ d- ~ 个人想法:太多学不完,先学PyQt5,原因是资料多,学有余力再学pyside6,最后看下PySimpleGUI,看能否解决一些简单问题。
2 `( u( A; t# h% O% _; p/ m" p
: B! m. k C) h6 c& {* d1 |8 C! M PyQt5简介
' l0 f8 ^/ }3 _, Y2 J7 E; W: O- u 0 i6 v; B) U" P H7 o; x0 T
PyQt是Qt框架的Python语言实现,由Riverbank Computing开发,是最强大的GUI库之一。PyQt提供了一个设计良好的窗口控件集合,每一个PyQt控件都对应一个Qt控件,因此PyQt的API接口与Qt的API接口很接近,但PyQt不再使用QMake系统和Q_OBJECT宏。 L7 x+ x( D1 c
2 x: D$ \* r1 s0 A
PyQt5提供GPL版和商业版证书,自由开发者可以使用免费的GPL许可,如果需要将PyQt用于商业应用,则必须购买商业许可。# |- X' Z3 p( z& ~
% q' J* N' Z) Q# h& }/ u PyQt5特性如下:
8 f' o6 C7 R9 m2 z3 f0 s + _4 G" T8 ~1 v4 s4 z1 h5 \
基于高性能的Qt的GUI控件集。/ c& A7 `7 p8 l* X( {
" q/ K P8 y# s: l: L 能够跨平台运行在Linux、Window和Mac OS系统上。, @& j7 D& Y6 C" m3 i
# }3 n; V/ `3 O/ r' B/ r
使用信号槽机制进行通信。2 `5 B: N; m2 ^- ]6 [' d
3 f* B( z- u$ H1 c0 ^2 m! A 对Qt库进行完全封装。
' _3 A4 k. c h! E& {1 k8 ^
6 I! C2 l- L( H6 y 可以使用成熟的IDE进行界面设计,并自动生成可执行的Python代码。
7 Q0 |" H5 X. h l7 }7 z, ~
; r+ Q, Y5 x; c% f1 s7 V" x; B 提供一整套种类齐全的窗口控件。
3 n5 U, }# V9 ~$ ~, D' M$ l; q! s ; q! l9 }! [9 f
PyQt5是由一系列Python模块组成,有超过620个类,6000个函数和方法,主要模块如下:
2 W. @( K! S% {% N$ {+ R
* j, D C L, p- n QtCore:包含了核心的非 GUI 的功能。主要和时间、文件与文件夹、各种数据、流、URLs、mime 类文件、进程与线程一起使用。0 H( ~5 r4 v0 q+ T7 t
0 O D/ |; |$ K t/ P2 n: L& b
QtGui:包含了窗口系统、事件处理、2D 图像、基本绘画、字体和文字类。$ ^. k; y; V. p% t) M# o3 L% L
* k2 d* ~7 t. c. M4 m+ e% Y QtWidgets:包含了一系列创建桌面应用的 UI 元素。
, E- h) x8 a/ F. R% F6 X 1 d$ f- C, @* _; ~/ E, Y
QtMultimedia:包含了处理多媒体的内容和调用摄像头 API 的类。
; ?# U, ~) F& `4 Z 4 \; M# I7 K+ e
QtBluetooth:包含了查找和连接蓝牙的类。1 R; K4 a) R/ Q0 s8 U1 ?
; T4 ^ W, n- P) ~
QtNetwork:包含了网络编程的类,这些工具能让 TCP/IP 和 UDP 开发变得更加方便和可靠。
0 M* I$ z4 v ~9 s3 S & y; {$ C4 ~7 U9 u: H0 f O
QtPositioning:包含了定位的类,可以使用卫星、WiFi 甚至文本。
3 r4 j* o W. l8 M + A& Q4 _( Q4 a# c2 X$ `5 w
Enginio:包含了通过客户端进入和管理 Qt Cloud 的类。* _ A; u# h: b, b- I' |
& k7 o9 e# l' E1 Y
QtWebSockets:包含了 WebSocket 协议的类。: I, z" `; F7 V3 r4 h
% J. D5 \- j2 L. C6 \' Q! ^6 V
QtWebKit:包含了一个基 WebKit2 的 web 浏览器。; ` D$ C' R) y
' Q) [! r# Y- k6 L4 ` QtWebKitWidgets:包含了基于 QtWidgets 的 WebKit1 的类。
, e) n9 w0 o) d! @4 ? + b8 N& f0 F+ r; t
QtXml:包含了处理 xml 的类,提供了 SAX 和 DOM API 的工具。+ M' y7 B; ?- u' _) v) g6 [! {" j
; }. _* l. O& c8 b
QtSvg:提供了显示 SVG 内容的类,Scalable Vector Graphics (SVG) 是一种是一种基于可扩展标记语言 (XML),用于描述二维矢量图形的图形格式(这句话来自于维基百科)。
z- _1 [2 c8 T, n- h / z7 s, Z& K1 U0 B& I: |- ^# w
QtSql:提供了处理数据库的工具。
* E- B* C# |1 u* I6 M9 [% E ' O# r4 r% C- _0 }! [2 z
QtTest:提供了测试 PyQt5 应用的工具。
3 ~( k) L2 v9 c K
5 ?# h* `: o5 i1 a# a+ G PyQt5的安装- F2 _) Y) s; G9 B) w/ c: `
8 z! O; ?- q( g, g4 A* w) V
由于后期要使用fbs进行打包,fbs对Python 3.7以后的版本可能存在兼容问题,所以我选择了Python 3.6.8进行了整个环境的搭建。主要内容为:Python + PyCharm + PyQt5
! j& A' M, H+ L _8 P L0 j. y4 z+ Q7 ^
安装PyQt5
0 g: y1 ?! R' S& r) Y; t' | / G( H( I0 i6 x! W3 ?- [
pip install pyqt5. H' \3 f/ d I$ X* |' D
: ^5 M: r* X& m! z) K pip install pyqt5-tools
" Q+ k8 q7 D' R g# O6 [6 ] 其中pyqt5-tools为Qt Designer拖拽式的界面设计工具。安装过程中可能会报如下错误:
* W# a8 a! F; V0 Y- {, o , H; i# P$ {2 J$ R! t- ]' Y6 g, c
qt5-tools 5.15.2.1.2 has requirement click~=7.0, but you'll have click 8.0.1 which is incompatible.( |8 S& i8 w1 Q! x
解决方案:. `0 w2 O% {8 \1 O5 r% Q w7 l/ s( @
T- q7 }7 R% `. D$ S7 w, o, F
pip install click~=7.0
" F& l' f/ A, p0 Z5 I( S* q Qt Designer的配置
# u2 A5 l' a* c/ ^% E- n D: ^& e# r" ^0 g) f
Qt Designer 是通过拖拽的方式放置控件,并实时查看控件效果进行快速UI设计。( Z: h) S7 A1 T! l& ~$ L
9 X9 V" Z) ]# J 0 d5 Y6 h3 O% @$ D) @
0 L% b8 z! r; j) k5 F! U 整个画面的构成:7 a! w! `; \) a j) H1 H
B! y$ U& }# y2 R9 M
左侧的“Widget Box”就是各种可以自由拖动的组件2 V+ R; o; ]; ]3 G5 ^4 A& k2 V, J
5 k. d3 f- y7 o4 R/ n, f& l$ P 中间的“MainWindow – untitled”窗体就是画布0 W* m! B2 I9 I$ C
2 O0 J9 l: x8 r0 o+ | 右上方的”Object Inspector”可以查看当前ui的结构
5 I" D' s6 K$ W$ n7 B- g' Q( B' F0 @
% K& S. e6 B3 }' E* r2 _- R 右侧中部的”Property Editor”可以设置当前选中组件的属性7 k5 n2 C# N% q @% w# }
8 t' n7 L2 ^" Q0 ]# U' l& ]( e
右下方的”Resource Browser”可以添加各种素材,比如图片,背景等等
V9 i" S. X1 m& p" k% Z + f' V- V) C: h! m& U/ z. g) c; V3 e6 Z
最终生成.ui文件(实质上是XML格式的文件),可直接使用,也可以通过pyuic5工具转换成.py文件。: j2 }/ z/ V6 l9 I9 m t; M6 S _2 t$ t
; Y: x2 D" t3 G
QtDisigner配置8 o0 W1 n( \7 z! V
$ j( }% J& ~; a9 C* z. ]7 F" L
在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:
; s2 _5 U0 d7 u. l' }6 W7 E
* s& e$ ^# ?& T Name: QtDisigner
; ^0 W" L2 r% s& i" w! ~ : r5 ]. \: }1 y. `
Program : D:\Program Files\Python36\Lib\site-packages\qt5_applications\Qt\bin\designer.exe # 请根据实际修改! x- u' m& L6 T, Z/ g# V! N
2 U$ c, N9 }3 [* T4 E+ o Working directory: $FileDir$
$ |8 _. t, B3 z6 y# Y; e3 G& s PyUIC配置) F, n# w+ ^8 d9 K# X
4 E* ?4 n: a* [4 \8 _* i
PyUIC主要是把Qt Designer生成的.ui文件换成.py文件。
8 o2 F0 u+ ~$ D: p 6 @# m, l( G3 \/ O" v
在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:4 N" O5 V9 p! ^; c! }: \: B$ j. @
7 N; F: z+ L; C+ M4 F
Name: PyUIC
0 N& u! l- u% s" Q. Y f
# i! t& L* _) ~/ ^ Program : D:\Program Files\Python36\python.exe # 当前Python目录,请根据实际修改) C- b+ n- F( S! c4 G# S5 p! S, @
2 \7 Y( J% A4 t3 o6 g7 O Arguments: -m PyQt5.uic.pyuic $FileName$ -o $FileNameWithoutExtension$.py! \3 r; X Q( {' w0 {' Z0 f
$ k( x: W0 t n4 R n( I Working directory: $FileDir$* }2 W. G: P7 z4 k" P6 R
PyRCC配置
* r* S+ r. ~. |5 ^; z9 T
/ x1 k" A# d& Y# r3 v9 ~$ f PyRCC主要是把编写的.qrc资源文件换成.py文件。在Pycharm中,依次打开 File – Settings – Tools – External Tools,点击 + Create Tool,配置如下:1 z. l$ v0 b. q7 B9 p s
& v: ^5 _7 N" @/ U$ R$ m
Name: PyRCC" O9 _, W0 M6 z z6 ?5 X
. ~: E, g: P+ j9 R) D% |4 c0 X! _ Program: D:\Program Files\Python36\pyrcc5.exe # 当前rcc工具目录,请根据实际修改$ \9 y! d7 N$ H$ n: n# D
- q, v1 O7 d% ~
Arguments: $FileName$ -o $FileNameWithoutExtension$_rc.py
/ E; x4 l& E$ [2 t+ e0 U. x
4 G" o- O4 j; c a Working directory: $FileDir$
7 W6 v7 P" b+ x! Z PyQt5使用示例4 Z* U4 M/ Z) Z/ x9 L8 g: H7 A
5 J) [# b- I b* L2 r: r8 W
创建一个空白的界面:0 e1 q5 f* P$ f: [9 o# H8 G
# R+ ]9 v& ]+ P4 R( A s- ~9 X
import sys& A4 A7 ]- W7 F& M# O; T$ T' s
( @% y/ F4 g3 {9 h2 ]
from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel
0 a+ _- I& a0 d0 S6 i 3 N- B" [0 }+ c& s2 A" K' K3 `& Q
app = QApplication(sys.argv)7 l" ]7 O! X& t+ L; Y
1 T; X* n/ {% G$ z" W" H$ L
win = QMainWindow()
" x1 V) O5 c2 z2 ^7 r $ f; d* p' A) w( q8 B, _. t# k
win.setGeometry(400, 400, 400, 300)1 \- p% a8 Z- ^& A8 Q1 f0 P Y
* x1 S7 `% D. S5 \+ t! f+ L- _ win.setWindowTitle("Pyqt5 Tutorial")
) p! i+ r& m( a2 b0 s- e4 D4 l$ W , O5 a) D# O) M: R8 `( U/ x/ [
win.show()
7 C; P7 N3 Z7 Z/ m# o
8 i; D( q' c: y6 E+ c8 O( S sys.exit(app.exec_())
* f1 V, V$ S8 R- b3 F) L) p( D& e , K" D b6 O$ m2 w3 j0 ]
" }& x" }: r, [# S2 V9 X
4 {% [$ q6 ^6 ], H R" @8 U1 ~ 其中:
9 D2 W/ C: ]6 o; F' i
: i; O$ [* F; B5 J Qapplication():每个GUI都必须包含一个Qapplication,argv表示获取命令行参数,如果不用获取,则可以使用[]代替。/ J, r2 ]" ^; b8 u. x* C6 h
" y3 K- }) }; t) ? QMainWindow():类似一个容器(窗口)用来包含按钮、文本、输入框等widgets。arg标识可以获取命令行执行时的参数。/ s2 P4 l) O( u0 h* q& ^3 R
, J u8 l9 v# C* }9 |; z! l0 w SetGeometry是用来定义 QMainWindow() 窗口的尺寸, 语法:setGeometry(x, y, width, height ),其中x,y为屏幕上的坐标点。3 ^1 g; p- J! r4 |* E6 ]+ \
! f. a# n! _ b1 _4 P+ s show():用来显示窗口
+ X! z9 V2 G4 h/ E
# F% q j1 T( j c6 _/ Y4 [ exit(app.exec_()):设置窗口一直运行指导使用关闭按钮进行关闭$ {1 W; F* L. }' Z7 @- J
0 } t$ K! Y* Y! B5 B PyQt5支持的常见Widgets有:* `8 ^6 Z. b! ?7 @# W/ o7 y
) ]7 F" A) D) m) b1 |0 K* \! L 5 y1 E# Z6 `+ J! b6 ]. k. N; u6 A
* B* ]" k# a+ @2 A6 d, F3 J
从上到下,从左到右依次为:Qlabel、QcomboBox、QcheckBox、QradioButton、QpushButton、QtableWidget、QlineEdit、Qslider、QProgressBar/ L) J8 q! ~* W+ H Z: k- P& {
! p* s2 Y6 \% [
对于使用Pyqt5设置文本内容,我们使用Qlabel:
" ~ W; V# y; v, }6 u% Z 8 B R# |/ F) F* ]
import sys
x7 n# e; E# y8 l' } 6 k% y0 I* E6 o& {
from PyQt5.QtWidgets import QApplication, QMainWindow, QLabel$ s, | K2 w1 T% Q1 x/ k! I p
+ i! e% V: o3 [$ r app = QApplication(sys.argv)
3 i+ o1 X! J% }* u9 b ' I* F# ?3 E! l
win = QMainWindow()2 D e- C! Y) V3 l6 z! i
) A) o- W8 ~9 k2 N* t3 e L win.setGeometry(400, 400, 400, 300)
1 ^; r0 R7 p, V- S3 T
8 d* Z; S/ i9 B# P3 x win.setWindowTitle("Pyqt5 Tutorial")
: B6 C4 r. f+ @/ ?9 u
+ H+ T4 q& r. Y7 U6 @ \# Label Text2 B2 S" \) N- F$ u
7 L9 z: Q4 _" e! k1 b0 R' d label = QLabel(win)8 W3 R( k3 H8 F1 Z% V
( t4 L9 M5 Z* ?9 ^3 k( Q label.resize(200, 100), z5 i9 |* G7 T; _
- ^2 `4 I! J/ Y6 Y) ~- u
label.setText("Hi this is Pyqt5")( X, v: ^: Y% H8 E( Z0 H) j% s1 v
( N% z" k2 t$ a, _6 }* E- M3 t0 a label.move(100, 100) K- N; Y b: c5 v, n! \
% R: J* P+ b: k1 m8 p win.show()
, ^& T, Q6 _% Q7 x/ D3 W3 F 9 D. _! K4 M. V- S
sys.exit(app.exec_())! ^& W( p/ ?% M9 T" U- b
( a* S* g. X: \3 H
4 D8 q3 |3 ]6 X p& O* a# V
' [8 M+ A7 s, A: o5 \ 按钮与事件:; h$ e+ A% q9 ^' k6 m8 f! O
, H+ |7 j: k, |3 U. X' b$ r( T
import sys0 V# ?0 N; o" _# ?$ t- @; L: M7 r
6 z* E! q3 N% S, J from PyQt5.QtWidgets import QApplication, QMainWindow, QPushButton+ O' R- F9 L ^$ b
) O) o; O* M' O" Y, i2 } def click():
3 O) _) V! F7 }- a1 u
* T' N* L, w8 ~6 `- x) s/ s. G print("Hy Button is clicked!")6 r& {7 M3 u3 D2 r
$ E" U) E1 c! B
app = QApplication(sys.argv)
" X/ k# o4 C$ l- ~
. [4 i* P3 ^. y6 x }! c; t5 C, V win = QMainWindow()
; ]8 x0 F6 c! C, L n) @6 J
, r& t8 S5 f+ {! g/ p ^ win.setGeometry(400, 400, 400, 300)
+ x+ t5 u3 O5 C3 w7 a% F! s
& j+ d- y. |0 {& V win.setWindowTitle("Pyqt5 Tutorial") W( B X* u) u2 a4 n
) `' }5 W u" Q2 W1 s) x: p \# Button+ r+ W+ x* u9 l2 [! O% \! f7 m
3 ~& F0 @2 J; P% f7 H button = QPushButton(win)
/ o5 D1 V8 v) X1 E ) D3 ?; s3 x8 H& h
button.resize(200, 100)
, W2 H3 a' d i8 }& W$ k
0 b$ Z5 T2 c5 U0 B9 n% h button.setText("Hi! Click Me")+ F% i1 |' x0 ]' [! V% f9 h- j! |7 r
+ k- h2 ]5 y' P! i
button.move(100, 100)
m) Y5 j; |0 s' s2 s
6 i/ v4 }6 Y6 ^0 M# ]- A button.clicked.connect(click)
* g7 V4 w. Z6 ]7 {; H . k7 \6 J0 u& m! l }$ C. U
win.show()) ~5 p$ p3 E, y2 e6 a L
8 H( E0 }8 B4 x, U: R6 p$ O
sys.exit(app.exec_())
! l4 p8 N- b+ A + ]* m# P: \* ^* q4 v) T5 L1 i
: Q# j, c t5 N/ ~( p6 B
2 r4 Y2 a% e: b5 i S+ u: ~* a! M button.clicked.connect() 在按钮点击后执行特定的事件。
* _0 g( t( R( k' t; L 6 Q- H3 A$ j C5 {" z1 z' N& {
PyQt5实战
* @% G' D; X% z3 E ( F$ V* D" c# L/ X M# s
实战项目:简易的天气查询软件 Q* `% x0 Q8 h _4 i
2 p" G# {4 ~- H+ s 1、使用Qt Designer设计一个界面; y+ }7 f$ ~4 O" K( ]; K1 g
9 S" ^' P7 C9 h3 }9 U- F* r
+ B8 ?7 H" j8 r
! p& ?8 Q: E3 R+ q) }- w! X 用到的控件有Button, GroupBox, Label,ComboBox,TextEdit,同时定义了两个按钮queryBtn及clearBtn,分别用来查询及清空天气数据。我们需要绑定槽函数,方法如下:: ]5 J; X0 ~2 b
9 p; g. N D4 b8 v) Y6 m' T
在Qt Designer右下角选择 信号/槽编辑器,点击+号新增
$ ~0 ` U& J- v' I / N3 J/ o# u/ x+ P# U, E$ c
分别选择queryBtn及clearBtn,选择信号 clicked(), 接收者 Dialog 及槽 accept(),(槽函数这里不知道如何定义,后期在代码里再进行修改)- K+ P5 C% ?/ h6 J# M
) g7 e! ]6 u L% V6 \- b 以上完成后保存为Weather.ui文件。0 c1 J5 [0 O0 h
! y* M! d! P( i# g. g 2、转换.ui文件为.py文件
# r, s4 v8 n0 L2 C
2 \- R5 v% E) w; } PyQt5支持直接使用.ui文件:
% z' M. z! O, h5 D: i6 r 4 C+ e% t" ]3 V. i
import sys8 A: Y* B2 b/ l4 }$ ~3 x
- p- a8 K7 U4 b) b$ H! S
from PyQt5 import QtWidgets, uic
, a! m1 z( z/ K# i W
+ i' j3 [5 s1 F8 |( w5 D7 G app = QtWidgets.QApplication(sys.argv)1 Q8 X) f% z1 X) j+ |- G' [
% k9 z. y* p" \" W window = uic.loadUi("mainwindow.ui")
- M. `; N5 X. I8 q + r* W% Q ^* V. P9 P2 B
window.show()
* _$ |; ]% ~ t8 L5 Q 0 t$ Y; T* M, u# @
app.exec()
! O& \% ^# h$ o& L6 M9 V 但是为了更好的自定义及修改上面的槽函数,可以使用External Tools – PyUIC,即可生成Weather.py,实际运行命令如下:3 j( ^+ Y9 {) O7 d" m
" s0 \+ e' _! u9 J D:\Program Files\Python36\python.exe -m PyQt5.uic.pyuic Weather.ui -o Weather.py/ K4 W$ v, p3 g6 z+ u8 U' f$ p/ Y7 c
其中,我们需要把两个按钮绑定的槽函数: g" n! D$ X) X. |# e
( p1 }# V$ \( e( D1 E! I: \# m \# self.queryBtn.clicked.connect(Dialog.accept)9 ^, J2 n, v6 u+ i6 G
0 h7 b2 ]5 g) w+ n- ^9 f0 A X \# self.clearBtn.clicked.connect(Dialog.accept)' Z; T6 h' q& b0 f$ `* k
) [& z( J8 d" W9 ]* E( {; B \# 修改为:
- m. E7 `6 \) S* G" n
; C; ]1 P7 X+ W9 x0 y self.queryBtn.clicked.connect(Dialog.queryWeather)+ M8 Z( i5 q9 N* _! t
6 P5 D0 s) Z; A! t
self.clearBtn.clicked.connect(Dialog.clearText)
. F6 I# b6 r- f4 f2 F 最终的Weather.py内容如下:9 d" {4 s1 a+ F
& U3 [" ^1 @ Z5 X
\# -*- coding: utf-8 -*-( Q5 r4 w9 w* d* W5 \
$ T; f6 K. U8 r9 R* c" i \# Form implementation generated from reading ui file 'Weather.ui'# t3 ?% c) a c2 q
$ W% h$ O+ k1 B' ?5 E \#* c0 ^! P6 J4 Z! [0 x
9 s6 X$ D7 Z& n8 l
\# Created by: PyQt5 UI code generator 5.15.4
8 C, _7 L; y5 I- k8 e# J# `; W; s + T+ K4 Z$ q% j i7 O' C' W
\#
2 l5 Z$ e9 B9 B: n 8 }. q3 I3 v$ G6 r# I6 \' v
\# WARNING: Any manual changes made to this file will be lost when pyuic5 is
( `) p; L: Z. `5 a- i$ v3 E 0 Z7 {, L2 Q) n" W8 L
\# run again. Do not edit this file unless you know what you are doing.
( g0 |5 b6 ]6 m! L h ) f' P5 i4 l$ F) U2 S9 e
from PyQt5 import QtCore, QtGui, QtWidgets
4 D5 f3 p9 T; B8 _5 ? 8 K6 K3 Y" @" r
class Ui_Dialog(object):$ w) A; a; w B) M
0 Y! J5 f% t3 ?6 v' h) H( Z
def setupUi(self, Dialog):; Q' c8 |6 p, {5 P% {" ?
9 {2 _% P: q% ~" p Dialog.setObjectName("Dialog")
! m; r( T) d/ g
3 ^3 x2 Y0 w+ e t% ^ Dialog.resize(600, 600)6 k. R! v0 C W. M
) G) G2 x# n l3 D5 z
self.groupBox = QtWidgets.QGroupBox(Dialog)
4 L4 ]% N, s5 h4 o 7 N" I/ e; N A8 b N
self.groupBox.setGeometry(QtCore.QRect(30, 20, 551, 511))
[" d0 C4 S6 d* B 1 s1 ]+ t4 R4 C' l6 j
self.groupBox.setObjectName("groupBox")' A) w0 t: @- x! s$ I2 M8 i' X
- k) ?9 k4 F& ?$ m9 ?) B) h7 W; ` self.label_2 = QtWidgets.QLabel(self.groupBox)7 T+ y1 J6 `$ [! U( _: @' J
3 ]. }0 R4 y5 X) e+ @1 Z* K1 d
self.label_2.setGeometry(QtCore.QRect(20, 30, 31, 16))0 B+ ?# ? G: j- H9 p6 S; T1 ^) T
3 i( |: F- K$ q& o2 n& U6 d* k
self.label_2.setObjectName("label_2")
# v* t6 D, m) V6 u1 ^/ r
) u/ Y4 _% W/ G$ B0 P) H3 ^* {. z self.comboBox = QtWidgets.QComboBox(self.groupBox)7 Y2 _$ x+ x1 t7 \: t
4 v, A) Y+ y: T. F$ ^
self.comboBox.setGeometry(QtCore.QRect(70, 30, 87, 22))
" U' g2 C1 v1 j/ C4 q( v- n ; b9 {' ]; ?5 M1 W# Q
self.comboBox.setObjectName("comboBox")% r* o% [9 h) g- g0 Q
1 h1 A3 G3 @- p1 T' T6 b. e self.comboBox.addItem("")
' G4 h) D7 m& e
4 F4 u N, ~* G self.comboBox.addItem("")0 | F( t9 H8 D9 I+ \ u
]# _1 c) Y& q# I# M$ E. c
self.comboBox.addItem("")
5 H, [ w% |+ \6 ]3 ? y( s! c9 q* ?$ y, A# C
self.textEdit = QtWidgets.QTextEdit(self.groupBox)
8 K8 F8 |4 v# W% V3 u5 j/ k
. L( S$ w' Z9 R: x$ d, S$ ?# v self.textEdit.setGeometry(QtCore.QRect(20, 70, 491, 411))4 y/ J: P' t: g6 [ g' P3 s8 w8 ^
1 G! I. d& p& x* m; r self.textEdit.setObjectName("textEdit")
0 X" O) H7 \/ \$ o: I0 p+ X 9 q1 c# E4 A3 t- O( D
self.queryBtn = QtWidgets.QPushButton(Dialog)
- s! X F; W+ P: d! Q6 @ + t4 ?5 A" [3 t
self.queryBtn.setGeometry(QtCore.QRect(490, 560, 93, 28))
% f9 G4 M8 p0 {4 A
/ [8 g: ^* ~# G3 Q2 `& k self.queryBtn.setObjectName("queryBtn")# t5 z h( [$ |! y- q7 t
. R9 b8 n: D3 o self.clearBtn = QtWidgets.QPushButton(Dialog)
6 r. G: N2 B$ z+ K: }. { * s7 G3 X+ C% m6 i% Y* h- q
self.clearBtn.setGeometry(QtCore.QRect(30, 560, 93, 28))
9 I$ R& o5 U! }% H# T' H6 n$ k 9 Y+ l8 k/ v, q: ~ d9 O' L' K! d( r
self.clearBtn.setObjectName("clearBtn")
7 J/ R2 A: K$ M- a' N
$ o% f. O3 T- U% b' U; d0 U/ S self.retranslateUi(Dialog)
/ ]/ ?0 Y2 j1 `; G1 k
# k- s* _0 n3 V& ?8 p self.clearBtn.clicked.connect(Dialog.clearText)
/ E6 t4 s, q* D! \
& o8 R7 M& y# H: z; \$ a self.queryBtn.clicked.connect(Dialog.queryWeather)
5 i$ R0 @1 K: o& F4 a) H, ]2 Y
9 q, U! }4 _8 c QtCore.QMetaObject.connectSlotsByName(Dialog)
4 {! l1 { \6 q- y+ i7 ]* F ' z* I0 J7 w k
def retranslateUi(self, Dialog):/ y6 ?! n$ X1 x/ `) T/ w
+ V- \2 s0 G3 G) |! i. F3 j' U
_translate = QtCore.QCoreApplication.translate2 Y3 U$ y1 |+ X o( x4 e+ x" l
0 _( Q$ E. ?3 F9 k
Dialog.setWindowTitle(_translate("Dialog", "Dialog"))
5 ]& k% l Z9 K+ E * z0 c5 k+ W1 a; x# E
self.groupBox.setTitle(_translate("Dialog", "城市天气预报"))
+ s) n- x. I; s4 c, b5 U/ |
* \% ^1 Q+ I2 Y6 G A self.label_2.setText(_translate("Dialog", "城市"))
; C# J" ~1 w0 ]: A% e8 h
- m$ R! S' e8 d5 X% z self.comboBox.setItemText(0, _translate("Dialog", "北京"))- H; s" s$ l: V5 G' l% a# A
9 z8 ?7 n" X- x: Y
self.comboBox.setItemText(1, _translate("Dialog", "苏州"))! D* G0 B$ \3 b2 Z( n; R+ P+ U
8 j+ Y+ }+ ^% Y( C
self.comboBox.setItemText(2, _translate("Dialog", "上海"))
- i' X7 y) j1 h- o( Y9 U! Q ' ?3 \1 n0 x, {. Q) |) i/ h( c, w
self.queryBtn.setText(_translate("Dialog", "查询"))
# I) i4 E2 O4 E3 e8 f
) m" n, F' [( _: J4 B7 p% L- P self.clearBtn.setText(_translate("Dialog", "清空"))3 \" g1 z% P2 z' m8 D% _
4 K: W5 @* c, H9 A( R4 m" U
3、调用MainDialog
: o r, u: Z2 u8 d9 e 6 H$ W/ L2 g' N
在MainDialog中调用界面类Ui_Dialog,然后在其中中添加查询天气的业务逻辑代码,这样就做到了界面显示和业务逻辑的分离。新增demo.py文件, 在MainDialog类中定义了两个槽函数queryWeather()和clearText(),以便在界面文件Weather.ui中定义的两个按钮(queryBtn 和clearBtn) 触发clicked 信号与这两个槽函数进行绑定。
4 u) R) Y$ T7 E# q7 I 0 m7 O4 P6 E6 }6 m4 |7 p
完整代码如下:
" x1 l1 }, L- n( Y: Z 1 h& m. k0 o# U( I7 ^' p
import sys
+ r- x+ Q; Q8 x8 \, y1 E
% b' |0 i. U, S6 K, I import Weather8 t: V% m, B2 Z
1 [/ `& g9 j" I: a: U, p from PyQt5.QtWidgets import QApplication, QDialog* m# w T& S3 R" I# N; Q: g! x
' K7 W$ L! H( f- q2 i5 Z* h X import requests
- O$ G5 h8 K1 u) n5 o4 D
8 `9 d, u, J$ N1 j) Y$ W class MainDialog(QDialog):
+ x4 l2 P9 X$ k
8 n4 e: D6 }: x4 D def __init__(self, parent=None):8 }) v' n' [& u+ j; J8 D" Y$ M& m
5 ~5 j, A+ y k2 y8 [& ~: f super(QDialog, self).__init__(parent)
( ^/ |6 W8 i+ ^! z. `# b2 K 7 L- r2 A4 ~, _ v$ ~+ V4 [# ^
self.ui = Weather.Ui_Dialog()
0 K$ w1 h( h3 L
: a, _. z* m) o. F6 ^ Y self.ui.setupUi(self)8 C6 g8 m+ a& ^) {" Z
; ]$ e+ P+ Q8 Q: J* W- t
def queryWeather(self):; |- f5 q; h, d& D" W7 [% J
1 s8 A; b2 A [
cityName = self.ui.comboBox.currentText()
3 S1 H0 ~- ? k
5 A' `& N7 p+ d6 s- a w cityCode = self.getCode(cityName): M4 u1 H( i! t( ]; u9 p, Q; Q
7 ]7 l( |' j4 n& W4 H
r = requests.get(
, D' U# N6 C5 t9 c% s$ m
, Q& n8 o4 Y" m* ~ "https://restapi.amap.com/v3/weather/weatherInfo?key=f4fd5b287b6d7d51a3c60fee24e42002&city={}".format(
) D1 P: s/ C5 m
A8 r/ |. [% \( A$ C cityCode))- \$ F/ q( ]+ `" Y
1 Z6 S2 W. S/ D) j/ S. r if r.status_code == 200:
' g$ N$ r* `$ s6 b* V9 j & ]" H; \% A# ^! n7 \) j
data = r.json()['lives'][0]
4 c! ]2 @2 K# Q
3 m* x# X% Q) c$ h ` weatherMsg = '城市:{}\n天气:{}\n温度:{}\n风向:{}\n风力:{}\n湿度:{}\n发布时间:{}\n'.format(2 I/ i5 f8 z- X
# w( ]9 C% K$ {2 l
data['city'],
: q1 s- k7 N: k; {; C2 ] * t7 [# s6 F1 n }' o( v0 [
data['weather'],
, [. l6 k# t5 l
# S# u x: ]# L; E data['temperature'], v1 l8 h* `: f' @
; h8 t: j& `. r; b4 E0 K' d u
data['winddirection'],; W2 E0 G: [" A$ g, |9 C
' e4 B( K6 H1 h/ r
data['windpower'],
1 g7 y5 I4 J! ^" x1 B " y" x# B* L& e! \4 L
data['humidity'],
" ?/ [4 Y1 G/ x- m2 i
; T x( t$ n/ P9 O6 N data['reporttime'],
9 `8 x2 J& _2 _0 P$ N : b6 d) R# b+ |% s. ^7 D* ?- U
)
! r6 F+ C6 Q" e5 P2 @0 M+ S
0 j" i6 B6 B9 m P7 ]1 O* p else:
" X+ v" ^/ y3 p3 }) D# o, {/ Q; |
) s2 w: P- z! a* U4 ] weatherMsg = '天气查询失败,请稍后再试!'
- A# O9 M; a2 }4 H% @
6 y8 M) q6 A. R self.ui.textEdit.setText(weatherMsg)4 P" Y, }1 Z3 Y7 T. J/ m4 R
, S' }) l, x$ i* }6 q! q+ y) C% a def getCode(self, cityName):
, K* I* Y/ t$ K+ o * o, I; d% F7 H" b0 @
cityDict = {"北京": "110000",
1 x% l- ~8 l' D/ l4 M+ ?" S 8 g/ h% ` w# m
"苏州": "320500",; y' V" G, P& {& u: f G
' H+ R- P1 B) m1 M/ t- Z0 ^ "上海": "310000"}' I. W+ Z8 a% X% K9 n$ C# e
* I% ~$ L! s( j! _4 P: I **return** cityDict.get(cityName, '101010100')
3 ?" {' f a1 Q8 }& h# L 5 }+ z$ N9 K! k0 l3 r* D, N( e
def clearText(self):/ A1 V0 d ]- G/ @5 L5 ~2 r6 ^
* B. w$ g k3 ]7 z
self.ui.textEdit.clear()' U7 W- ?0 {' x% `6 p; }
! |) u2 I& A: W/ Y% \) j
if __name__ == '__main__':
+ F) R9 f( E4 i) n1 a
. |) N2 [, B9 B) w myapp = QApplication(sys.argv)
6 x. I1 m/ l5 O0 F6 _ : L$ C6 q( p1 d. n# C; H4 e; c
myDlg = MainDialog()# y( `0 Y9 \* ?
) b6 t1 G& F, T j% a7 i myDlg.show()' u& l# M7 d% k7 x6 b
+ F$ U% g% N h4 @" s5 W
sys.exit(myapp.exec_())7 n; t, j' @/ w# \7 m7 z# o5 w0 j
( O4 B+ K9 h5 j2 n
运行demo.py并执行查询后的效果:6 e7 B% Z9 Q* Y1 ~
' b2 w- B+ M* P+ \+ Q2 _ H5 C; r
! [8 N. m! Q( G$ b8 z9 ]3 e * h3 k* I- D+ [' D* B
4、将代码打包成exe文件% q( E: O8 L9 Y/ H: u
! F/ p5 } O9 M$ X3 Y6 ?
将.py文件打包成可执行的exe在Python中称为freezing,常用的工具有:PyInstaller, py2exe, cx_Freeze, bbfreze, py2app等。功能对比:1 J3 m ^" T0 h; {; f" ?) U
# \6 | G c( u1 Y * k: ~* _' R* {8 ] D0 ^
. ^0 d) x7 I' R4 a% q( }$ j py2exe:软件更新已经不活跃,因此也就略过。
1 K( N: z( T8 c! I2 t/ T , d( C- }4 N: E) P g
pyinstaller:明确支持win8、win10、理论上支持win7,,支持apple Macos, linux。pyinsaller可以打包成文件夹形式内含exe入口执行文件的形式,也可以是一个单独的exe文件。. ^$ j6 J+ `6 p, r: K9 m
/ ]! l K! ~; n8 `2 I fbs:基于PyInstaller,使用起来更加方便* Q% m" n8 l9 G
9 f6 X {; Y8 v. o5 }" I 这里选择了fbs来打包。fbs的安装方法:
" L% A; L% o# Q/ L5 r' E
+ T- i: n$ E8 g" W pip install fbs* }$ n4 Q9 |3 _# r1 d) W3 B
使用方法,在命令行中输入:: n4 a8 J3 B2 ]' \% u7 d0 c0 s: b
9 |6 C0 G0 w+ T/ h& Y8 x6 }3 W
fbs startproject
$ R% Z( d* v4 h 执行完成后需要输入一些APP的名称等。完成后会生成如下目录:
+ u0 }5 C0 V7 Y# v" | " J6 @; [+ \: y+ q+ B# R/ U' @
" x; K: @" D b6 ` T# O
" e9 @ Q2 o7 ?3 u) q 将刚才编写的PyQt5的代码(demo.py和Weather.py)拖到src/main/python文件夹下,删除原有的main.py,并将demo.py修改为main.py。然后打开 main.py,在文件头部添加如下代码:
. M# V5 w! F" u! Y1 e; g" y* ?) p8 F$ B # W T$ G c) A0 Q. z$ H- N
from fbs_runtime.application_context.PyQt5 import ApplicationContext
% c8 u! n2 m4 u9 f. \ _4 _ ```: R; t/ z) F( F E% l
完成后执行:
7 w9 e( [; n9 G" b% T ```
: a* X+ n7 y$ S fbs freeze% g& q; H) r" W) e9 d7 |
```
: ?1 Z% k% `+ p# @! Q 即可实现打包。生成的exe可执行文件在\target\MyApp文件下。
* {0 E- Q/ l- A. U% ~
' W" t+ j* C% [( j4 ^' z& Z, Y ————————————————% {3 V& x& {/ x" I6 _( H$ V
版权声明:本文为CSDN博主「宋宋讲编程」的原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接及本声明。
$ q0 I1 z# c# j 原文链接:https://blog.csdn.net/qiqi1220/article/details/126289667
+ D9 x! r, t K* h6 r! B
# S6 G" W3 H& Y( _( T X t1 U 6 R; L) v4 D: c, S% T. N
- G5 O6 q1 f6 w' L6 E
zan