python 在线预览文件_用Python PyQt写一个在线预览图片的GUI

在爬完网上一篇帖子,并得到其中的所有图片链接后,写一个GUI来实现在线预览是一个很自然的想法, 相当于实现一个python版的图片浏览器, 通过这个练习,可以让我们更熟悉PyQt这个库。

这里我用的是PyQt4。

以下是我的写的程序,可以实现以下几个功能。

预览图片:

82ee5ad90684

实现鼠标左键单击即可下翻至下一张图片,鼠标右键单击则是返回前一张图片。

利用本地Cache来解决反复读取的问题。 比如说对同样的两张图片我们想来回比较,我们没必要每次都重新下载。我们可以将图片保存到本地,当以后调用同一张图片时,直接从本地读取缓存来加速。如果我们不想让外界看到缓存的图片,则可以对其进行加密 (尚未实现)。

对动图gif的支持。普通的 jpg 和 Png 格式,利用QLable 即可以显示,但是对于gif,我们则必须利用QMovie来让其动起来。

由于我刚接触这个库,仍然在学习,所以写的不妥或者冗余的地方,希望大家指出。

# coding=utf-8

import sys

import pycurl

import os

import time

from StringIO import StringIO

import re

from PyQt4 import QtGui,QtCore

from PyQt4.QtGui import *

from PyQt4.QtCore import *

# class definition

class Pic_Label(QtGui.QLabel):

def __init__(self):

super(Pic_Label,self).__init__()

self.setFrameStyle(QtGui.QFrame.StyledPanel)

self.cache_map={}

def paintEvent(self, event):

if self.extention !="gif":

size = self.size()

painter = QtGui.QPainter(self)

point = QtCore.QPoint(0,0)

scaledPix = self.pixmap.scaled(size, Qt.KeepAspectRatio, transformMode = Qt.SmoothTransformation)

# start painting the label from left upper corner

point.setX((size.width() - scaledPix.width())/2)

point.setY((size.height() - scaledPix.height())/2)

#print point.x(), ' ', point.y()

painter.drawPixmap(point, scaledPix)

else:

QLabel.paintEvent(self, event)

def mouseReleaseEvent(self,ev):

#self.emit(SIGNAL('clicked()'))

if ev.button() == Qt.RightButton:

self.emit(SIGNAL("RightClick"))

else:

self.emit(SIGNAL("LeftClick"))

def set_image(self,pic_url,index):

if (index in self.cache_map) == False:

self.cache_map[index]=False

self.pixmap = QtGui.QPixmap()

self.retrieve_from_url_cache(pic_url,index)

def retrieve_from_url_cache(self,pic_url,index):

try:

self.extention=re.search(r"\.(\w+)$", pic_url).group(1)

except:

self.extention="jpg"

cache_pic_name="Pic_"+str(index)+"."+self.extention

cache_pic_path=os.getcwd()+"\Cache_Pic\\"+cache_pic_name

if self.cache_map[index]==True:

if self.extention =="gif":

movie = QtGui.QMovie(cache_pic_path)

self.setMovie(movie)

movie.start()

else:

#print "Cached!" + cache_pic_path

if self.pixmap.load(cache_pic_path) == False:

#print "use jpg to try again"

if self.pixmap.load(cache_pic_path)== False:

#last resort, try again

self.retrieve_from_url(pic_url,index,cache_pic_path)

self.setPixmap(self.pixmap) # udpate immediately

else:

if self.extention =="gif":

data=self.retrieve_from_url(pic_url,index,cache_pic_path)

self.pixmap.loadFromData(data)

f = open(cache_pic_path, 'wb')

f.write(data)

f.close()

movie = QtGui.QMovie(cache_pic_path)

self.setMovie(movie)

movie.start()

else:

data=self.retrieve_from_url(pic_url,index,cache_pic_path)

self.pixmap.loadFromData(data)

self.pixmap.save(cache_pic_path)

self.setPixmap(self.pixmap) # udpate immediately

def retrieve_from_url(self,pic_url,index,file_path):

c = pycurl.Curl()

c.setopt(pycurl.PROXY, 'http://192.168.87.15:8080')

c.setopt(pycurl.PROXYUSERPWD, 'LL66269:')

c.setopt(pycurl.PROXYAUTH, pycurl.HTTPAUTH_NTLM)

buffer = StringIO()

c.setopt(pycurl.URL, pic_url)

c.setopt(c.WRITEDATA, buffer)

c.perform()

c.close()

data = buffer.getvalue()

self.cache_map[index]=True

return data

def setMovie(self,movie):

QLabel.setMovie(self, movie)

s=movie.currentImage().size()

self._movieWidth = s.width()

self._movieHeight = s.height()

class Example(QtGui.QWidget):

def __init__(self,thread_url_list):

super(Example, self).__init__()

self.url_list=thread_url_list

self.current_pic_index=0

cwd = os.getcwd()

#print cwd

directory=cwd+"\Cache_Pic"

#print directory

if not os.path.exists(directory):

os.makedirs(directory)

self.initUI()

# making subfolderss to cache pictures

def initUI(self):

layout = QtGui.QGridLayout()

self.label = Pic_Label()

self.label.set_image(self.url_list[0],0)

#self.label = QLabel()

#movie = QtGui.QMovie("Cache_Pic/Pic_0.gif")

#self.label.setMovie(movie)

#movie.start()

layout.addWidget(self.label)

layout.setRowStretch(0,1)

layout.setColumnStretch(0,1)

#self.connect(self.label,SIGNAL('clicked()'),self.fun_next)

self.connect(self.label,SIGNAL("LeftClick"),self.fun_next)

self.connect(self.label,SIGNAL("RightClick"),self.fun_prev)

#b1=QtGui.QPushButton("next")

#b2=QtGui.QPushButton("prev")

#b1.clicked.connect(self.fun_next)

#b2.clicked.connect(self.fun_prev)

#layout.addWidget(b1)

#layout.addWidget(b2)

self.setLayout(layout)

self.setGeometry(300, 300, 500, 500)

self.setWindowTitle('Picture Viewer')

self.show()

# Connect button to image updating

def fun_next(self):

if self.current_pic_index < len(self.url_list)-1:

self.current_pic_index=self.current_pic_index+1

else:

self.current_pic_index=0

self.label.set_image(self.url_list[self.current_pic_index],self.current_pic_index)

sys.stdout.write('\r')

sys.stdout.write("[ %d ] out of (%d)" % (self.current_pic_index+1,len(self.url_list)))

sys.stdout.flush()

def fun_prev(self):

if self.current_pic_index > 0:

self.current_pic_index=self.current_pic_index-1

else:

self.current_pic_index=len(self.url_list)-1

self.label.set_image(self.url_list[self.current_pic_index],self.current_pic_index)

sys.stdout.write('\r')

sys.stdout.write("[ %d ] out of (%d)" % (self.current_pic_index+1,len(self.url_list)))

sys.stdout.flush()

def view_image():

url_list=['https://i.imgur.com/waprhO3.gif','http://static.cnbetacdn.com/article/2017/0831/7f11d5ec94fa123.png','http://static.cnbetacdn.com/article/2017/0831/1b6595175fb5486.jpg']

viewer_app = QtGui.QApplication(sys.argv)

ex = Example(url_list)

sys.exit(viewer_app.exec_())

view_image()

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/533630.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

python怎样安装模块_python中如何安装模块

下面介绍几种安装Python模块的几种方式方法1&#xff1a;easy_install 方式先下载ez_setup.py,运行python ez_setup 进行easy_install工具的安装&#xff0c;之后就可以使用easy_install进行安装package了。本文安装的是Python 2.7.13版本&#xff0c;已经自带了easy_install。…

java rt_java中rt包中源码了解

javap –verbose class名 查看class文件的具体内容javap -c class名继续看io类接口 java.io.Closeable功能&#xff1a;关闭流和相应的资源java.io.console功能&#xff1a;使用字节控制台&#xff0c;与当前的java virtual machine 相关java.io.DataInput功能&#xff1a;从二…

google 确定某点海拔高_一份“高投资回报率”的用户体验度量方法指南

本文核心就是介绍体验度量方法&#xff0c;以及如何在商业项目中如何发起一个具有高ROI(投资回报率)的用户体验量化流程。 下面文章将分为解读高投资回报和拆解体验度量、实际案例讲解三部分。一、解读高投资回报率高ROI(投资回报率)来定义体验度量流程的原因&#xff1f;3-5年…

md5 java代码_JAVA简单实现MD5注册登录加密实例代码

开发环境&#xff1a;jdk1.7&#xff0c;eclipse框架&#xff1a;springmvc&#xff0c;mybatis工具&#xff1a;maven以下代码复制即可实现MD5加密创建一个mave项目&#xff0c;加web。不懂得可以搜索一下就有了。注册用户的JSP页面代码如下。pageEncoding"utf-8"%&…

一维卷积神经网络_序列特征的处理方法之二:基于卷积神经网络方法

前言上一篇文章介绍了基本的基于注意力机制方法对序列特征的处理&#xff0c;这篇主要介绍一下基本的基于卷积神经网络方法对序列特征的处理&#xff0c;也就是TextCNN方法。序列特征的介绍&#xff0c;背景以及应用可以参考上一篇的详细介绍&#xff0c;这里简单回顾一下定义&…

java socket 阻塞模式_(四) 如何将socket设置为非阻塞模式

1. windows平台上无论利用socket()函数还是WSASocket()函数创建的socket都是阻塞模式的&#xff1a;SOCKET WSAAPI socket( _In_ int af, _In_ int type, _In_ int protocol ); SOCKET WSASocket( _In_ int af, _In_ int t…

python中的pygame模块使用方法_Pygame的基本使用

Pygame有很多模块&#xff0c;每个模块又有很多方法&#xff0c;在此不能够逐一讲解&#xff0c;所以&#xff0c;我们通过一个实例来学习Pygame&#xff0c;然后再分解代码&#xff0c;讲解代码中的模块。例&#xff1a;制作一个跳跃的小球游戏。创建一个游戏窗口&#xff0c;…

java mongodb 插入数据_mongoDB 插入数据 用java实现

import java.net.UnknownHostException;import com.mongodb.BasicDBObject;import com.mongodb.DB;import com.mongodb.DBCollection;import com.mongodb.DBObject;import com.mongodb.Mongo;/** *用java 往mongoDB插入数据 * author wwd* */public class InsertData {publi…

python的特征提取实验一_Spark 2.1.0 入门:特征抽取 — TF-IDF(Python版)

这一部分我们主要介绍和特征处理相关的算法&#xff0c;大体分为以下三类&#xff1a;特征抽取&#xff1a;从原始数据中抽取特征特征转换&#xff1a;特征的维度、特征的转化、特征的修改特征选取&#xff1a;从大规模特征集中选取一个子集特征提取TF-IDF (HashingTF and IDF)…

java addlast_Java中的LinkedList addLast()方法: java.util.LinkedList.addLast() - Break易站

Java中的java.util.LinkedList.addLast()方法用于在LinkedList的末尾插入特定元素。句法&#xff1a;void addLast(Object element)参数&#xff1a;此函数接受单个参数元素&#xff0c;如上面的语法所示。此参数指定的元素将附加在列表的末尾。返回值&#xff1a;此方法不返回…

macos降级_iOS12.3 beta2更新了什么 iOS12.3测试版2新特性与升降级方法

4月0日凌晨&#xff0c;苹果发布了iOS12.3 beta2&#xff0c;作为iOS12.3第二个测试版&#xff0c;相比前一个版本&#xff0c;发布时间间隔近2周&#xff0c;这次依然是小版本更新&#xff0c;不过相对良心一些&#xff0c;主要是多了一些与国内用户相关的东西。iOS12.3 beta …

java技术教程视频_Spring开发视频教程高级篇+源码(400M)33讲

Spring开发视频教程高级篇源码(400M)33讲01_全面阐释Spring及其各项功能.rar 02_搭建与测试Spring的开发环境.rar 03_编码剖析Spring管理Bean的原理.rar 04_Spring的三种实例化Bean的方式.rar 05_配置Spring管理的bean的作用域.rar 06_Spring管理的Bean的生命周期.rar 07_编码剖…

python的常见矩阵除法_Numpy矩阵除法返回所有零

我对下面的矩阵有个除法错误。我想用行和的101向量除以1010matrix。在[[5731, 3, 20, 8, 12, 54, 46, 8, 39, 2],[ 2, 6472, 47, 24, 7, 44, 7, 11, 116, 12],[ 55, 36, 5296, 104, 84, 27, 106, 53, 183, 14],[ 50, 49, 132, 5312, 2, 253, 36, 58, 142, 97],[ 16, 28, 36, 9,…

java rc2加密_急求java RC2加密算法

下面是一段C RC2加密 要求要用java 重写 能互相加密解密QSBEncryptRc2::QSBEncryptRc2(){EncryKey "DingXin Communication Key 20080613";}//解密失败时返回失败描述AnsiString QSBEncryptRc2::GetDecryptErrMsg(){int ErrorCode;AnsiString ErrMsg;ErrorCode …

linux配置usb主从_杂集:浅谈关于Mongodb数据库主从复制

Linux下Mongodb数据库主从复制配置Mongodb的三种集群搭建的方式&#xff1a;Master-Slaver&#xff1a;主从[目前被副本集取代]。Replica Set&#xff1a;副本集。Sharding&#xff1a;切片。Mongodb单实例缺点&#xff1a;适合简易开发时使用&#xff0c;生产使用不行&#xf…

java读取邮箱附件_使用javamail获取附件内容

我正在使用javamail来自动化一些电子邮件处理。使用javamail获取附件内容我设法连接到pop3服务器并获取消息。其中一些包含附件。根据邮件标题&#xff0c;我可以“预测”我需要获取的附件的文件名。但我无法得到它的内容:(我有一个函数public byte[] searchForContent(Part pa…

高斯拟合 vc++代码_NMA2020W1 极大似然法模型拟合与bootstrap

常见的线性模型&#xff1a; 求解方式有两种&#xff0c;一种是计算均方误差&#xff08;MSE&#xff09;&#xff0c;使得均方误差最小。图1找到梯度为零的点即可。而之前一直比较模糊的最大似然法也比较清楚了。一般线性模型&#xff0c;我们假定误差项是符合高斯分布的&…

java文件快速扫描仪_Java扫描仪具有示例的NextNextShort()方法

扫描仪类hasNextShort()方法语法&#xff1a;public boolean hasNextShort();public boolean hasNextShort(int rad);hasNextShort()方法在java.util包中可用。hasNextShort()方法用于检查此扫描程序在其输入中是否具有下一个标记&#xff0c;是否可以将其作为隐式基数中的shor…

python sqlite并发处理_python sqlite大数据 处理

{"moduleinfo":{"card_count":[{"count_phone":1,"count":1}],"search_count":[{"count_phone":4,"count":4}]},"card":[{"des":"阿里技术人对外发布原创技术内容的最大平台&…

java sax xml文件解析_java解析xml文件-DOM/SAX

java解析xml文件的两种方式1&#xff1a;DOM原理&#xff1a;把整个文档加载到内存&#xff0c;转化成dom树&#xff0c;之后应用程序可以随机的访问dom树的任何数据&#xff0c;灵活 快&#xff0c;但消耗内存一个简单的xml使用java解析//builder工厂DocumentBuilderFactory f…