自从和 GNOME 开发者接触过之后,我决定放弃断断续续学了一段时间的 GTK 而转向 Qt 了。看了两三天的 PyQt4 tutorial,恰好遇到一需要界面的脚本,本来我会搞成 Web 的,但既然学了 Qt 嘛,当然得练练。
起因是这样子的。一天和群里的人聊到天气之类的,然后有人扔了张卫星云图出来。多年未见的云图啊,再次见到感觉好亲切,虽然我的地理和气象知识已经忘记了好多了。然后得到了这个网页。本来呢,他们想搞成在以前的天气预报里那样逐幅图像显示的滚动播放效果的,但政府网站嘛,出点问题很正常,比如我第一次播放的时候图片基本上动不了(可能是因为没有预加载以及网络不畅造成的),第二次的时候图像时不时闪一下。而且这样可控性太差,想看的时段根本看不清就跳走了,不想看的时段却一直慢慢地跳。所以我拿 Python 写了个通过 slider 滑块来控制显示的图像简易脚本。写完后才发现使用 sxiv 然后按住N/P键是一样的效果……
代码如下,需要 PyQt4 或者 PySide。对于后者,Arch 用户可以从 Arch Linux CN 源安装而不必通过 AUR 自行编译(C++ 编译很费时的)。
#!/usr/bin/env python3
# vim:fileencoding=utf-8
# TODO: 并发下载
# TODO: 下载进度显示
# TODO: 允许加载已经下载但网页上没有的云图
# TODO: 网络作为可选
import os
import sys
import re
import gzip
import urllib.request
pic_dir = '.'
try:
from PySide import QtGui, QtCore
except ImportError:
from PyQt4 import QtGui, QtCore
def getPage():
request = urllib.request.Request('http://www.weather.com.cn/static/product_video_v1.php?class=JC_YT_DL_WXZXCSYT')
request.add_header('Accept-Encoding', 'gzip')
res = urllib.request.urlopen(request)
return gzip.decompress(res.read()).decode('utf-8')
def getPics(page):
urlre = re.compile(r'\bhttp://i.weather.com.cn/i/product/pic/m/sevp_nsmc_wxcl_asc_e99_achn_lno_py_\d{17}.jpg\b')
return sorted({x.replace('/m/', '/l/') for x in urlre.findall(page)})
def download(pics):
ret = []
for p in pics:
file = os.path.split(p)[1]
file = os.path.join(pic_dir, file)
ret.append(file)
if os.path.exists(file):
continue
data = urllib.request.urlopen(p).read()
open(file, 'wb').write(data)
return ret
class YuntuShow(QtGui.QWidget):
def __init__(self, pics):
super().__init__()
self.pics = pics
self.initUI()
def initUI(self):
pic = QtGui.QPixmap(self.pics[-1])
self.pic = piclabel = QtGui.QLabel(self)
piclabel.setPixmap(pic)
slider = QtGui.QSlider(QtCore.Qt.Horizontal, self)
slider.setTickPosition(QtGui.QSlider.TicksBelow)
m = len(self.pics) - 1
slider.setMaximum(m)
slider.setSliderPosition(m)
slider.valueChanged[int].connect(self.changePic)
vbox = QtGui.QVBoxLayout()
vbox.addWidget(piclabel)
vbox.addWidget(slider)
self.setLayout(vbox)
self.resize(960 + 10, 720 + 50)
self.setWindowTitle('YuntuShow')
self.show()
def keyPressEvent(self, e):
if e.key() == QtCore.Qt.Key_Q:
self.close()
def changePic(self, value):
pic = QtGui.QPixmap(self.pics[value])
self.pic.setPixmap(pic)
def main():
urls = getPics(getPage())
pics = download(urls)
app = QtGui.QApplication(sys.argv)
yt = YuntuShow(pics)
sys.exit(app.exec_())
if __name__ == '__main__':
main()
短吧?而且我也只看了三天的教程用的时候查了下文档而已哦~比起 GTK 来快速多了。不过这其中有个很重要的原因是我已经通过 GTK 了解了基本的 GUI 编程了。
因为最开始会从网络下载数据和图像,所以要等一会儿窗口才会出现。