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.pswp.cn/news/533630.shtml
繁體地址,請注明出處:http://hk.pswp.cn/news/533630.shtml
英文地址,請注明出處:http://en.pswp.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…