基于PyQT5制作英雄聯(lián)盟全皮膚下載器
這是通過(guò)博主寫的英雄聯(lián)盟下載器下載的部分的英雄皮膚,可以看一下效果。每個(gè)英雄的皮膚的會(huì)自動(dòng)根據(jù)英雄名稱創(chuàng)建相應(yīng)的文件夾存放。
實(shí)現(xiàn)思路比較簡(jiǎn)單,同樣是通過(guò)PyQt5來(lái)編寫下載頁(yè)面。最后通過(guò)request模塊來(lái)進(jìn)行皮膚的下載部分編寫。演示一下操作過(guò)程是下面這樣的,選擇好皮膚的存儲(chǔ)路徑。然后直接點(diǎn)擊開始下載就行了,并且可以在文本瀏覽器中查看下載進(jìn)度信息。
接下來(lái),介紹一下代碼塊的主要是實(shí)現(xiàn)部分。首先,介紹一下整個(gè)代碼塊都使用了哪些第三方模塊。
# 英雄聯(lián)盟皮膚下載相關(guān)依賴模塊 import requests # 網(wǎng)絡(luò)請(qǐng)求庫(kù) import re # 正則表達(dá)式匹配庫(kù) import json # JSON格式轉(zhuǎn)換庫(kù) import os # 應(yīng)用操作庫(kù) import time # 時(shí)間模塊 from random import random # 隨機(jī)數(shù)模塊 from fake_useragent import UserAgent # user_agent 生成庫(kù) import logging # 日志模塊 import sys # 系統(tǒng)操作 # pyqt5的模塊引用這里就不介紹了,最近一直在用。 from PyQt5.QtGui import * from PyQt5.QtWidgets import * from PyQt5.QtCore import *
UI界面的設(shè)計(jì)過(guò)程代碼塊以及信號(hào)與槽函數(shù)的應(yīng)用過(guò)程。
def init_ui(self): self.setWindowTitle('英雄皮膚下載器 公眾號(hào):[Python 集中營(yíng)]') self.setWindowIcon(QIcon('lol.ico')) self.resize(500,250) vbox = QVBoxLayout() self.save_dir = QLineEdit() self.save_dir.setReadOnly(True) self.save_btn = QPushButton() self.save_btn.setText('路徑') self.save_btn.clicked.connect(self.save_btn_click) self.thread_ = DownLoadThread(self) self.thread_.trigger.connect(self.update_log) self.start_btn = QPushButton() self.start_btn.setText('開始下載') self.start_btn.clicked.connect(self.start_btn_click) grid = QGridLayout() grid.addWidget(self.save_dir, 0, 0, 1, 2) grid.addWidget(self.save_btn, 0, 2, 1, 1) grid.addWidget(self.start_btn, 0, 3, 1, 1) self.result_brower = QTextBrowser() self.result_brower.setFont(QFont('宋體', 8)) self.result_brower.setReadOnly(True) self.result_brower.setPlaceholderText('英雄皮膚批量下載進(jìn)度顯示區(qū)域...') self.result_brower.ensureCursorVisible() vbox.addWidget(self.result_brower) vbox.addLayout(grid) self.setLayout(vbox) def start_btn_click(self): self.start_btn.setEnabled(False) self.thread_.start() def update_log(self, text): cursor = self.result_brower.textCursor() cursor.movePosition(QTextCursor.End) self.result_brower.append(text) self.result_brower.setTextCursor(cursor) self.result_brower.ensureCursorVisible() def save_btn_click(self): directory = QFileDialog.getExistingDirectory(self, "選取文件夾", self.cwd) self.save_dir.setText(directory)
UI界面的設(shè)計(jì)代碼,遵循的是和其他PyQt5一樣的設(shè)計(jì)范式。按照這樣的范式寫出來(lái)的UI代碼塊個(gè)人覺(jué)得看起來(lái)也比較美觀。
為了防止下載過(guò)程比較慢的情況下會(huì)導(dǎo)致UI界面的主線程直接掛掉,所以在編寫下載業(yè)務(wù)的邏輯時(shí)是需要單獨(dú)使用QThread子線程的方式來(lái)編寫的。這樣可以使得業(yè)務(wù)邏輯和主頁(yè)面邏輯分離,就不會(huì)產(chǎn)生掛掉的情況發(fā)生了。
class DownLoadThread(QThread): trigger = pyqtSignal(str) def __init__(self, parent=None): super(DownLoadThread, self).__init__(parent) # 初始化日志對(duì)象 self.logger = logging.getLogger('英雄聯(lián)盟皮膚') logging.basicConfig() self.logger.setLevel(logging.DEBUG) self.parent = parent self.working = True def __del__(self): self.working = False self.wait() def run(self): ''' 英雄聯(lián)盟皮膚下載函數(shù) :return: ''' while self.working == True: # 構(gòu)造useragent身份設(shè)備信息 headers = { "User-Agent": str(UserAgent().random), } self.logger.info('生成身份設(shè)備信息完成') self.trigger.emit('生成身份設(shè)備信息完成!') # 在LOL的官網(wǎng)經(jīng)過(guò)分析champion.js中包含了我們要用到的英雄ID編號(hào),因此,將這個(gè)JS文件下載下來(lái) champion_js_url = "https://lol.qq.com/biz/hero/champion.js" # 發(fā)送http請(qǐng)求、請(qǐng)求地址為這個(gè)JS文件的地址。就會(huì)將其下載下來(lái)了。 response = requests.get(url=champion_js_url, headers=headers) self.logger.info('champion.js 文件下載完成') self.trigger.emit('champion.js 文件下載完成!') # 獲取下載的文本信息 text = response.text self.logger.info('champion.js 文本信息:' + text) # 匹配champion對(duì)象對(duì)應(yīng)的JSON數(shù)據(jù),該JSON數(shù)據(jù)的第0個(gè)位置是包含英雄ID編號(hào)的JSON數(shù)據(jù) hero_json = re.findall(r'champion=(.*?);', text, re.S)[0] # 將JSON數(shù)據(jù)轉(zhuǎn)換為dict字典 hero_dict = json.loads(hero_json) self.logger.info('英雄ID字典信息:' + str(hero_dict)) # 逐個(gè)英雄信息遍歷 for hero_data in hero_dict["data"].items(): # 獲取英雄詳細(xì)信息的JS文件 hero_js_url = "http://lol.qq.com/biz/hero/{}.js" # 發(fā)送請(qǐng)求下載該JS文件、hero_data[0]取第0位就是英雄ID response = requests.get(url=hero_js_url.format(hero_data[0]), headers=headers) # 獲取下載文本 text = response.text self.logger.info(hero_data[0] + '.js 文本信息:' + text) self.trigger.emit(hero_data[0] + '.js 文件下載完成!') skins_dict = json.loads(re.findall("{}=(.*?);".format(hero_data[0]), text, re.S)[0]) self.logger.info('當(dāng)前英雄皮膚字典:' + str(skins_dict)) # 從字典 skins_dict 獲取皮膚列表 skins_list = skins_dict["data"]["skins"] # 獲取英雄名稱 hero_name = hero_data[1]["name"] # 在當(dāng)前目錄下面創(chuàng)建images文件夾、以英雄名稱作為文件夾名稱 os.makedirs(r"./images/{}".format(hero_name), exist_ok=True) for skin_info in skins_list: # 初始化皮膚圖片地址 skin_pic_url = "https://ossweb-img.qq.com/images/lol/web201310/skin/big{}.jpg" # 發(fā)送下載請(qǐng)求 reponse = requests.get(url=skin_pic_url.format(skin_info["id"]), headers=headers) try: self.logger.info('保存路徑' + self.parent.save_dir.text().strip()) # 保存皮膚 with open(r"" + self.parent.save_dir.text().strip() + "/{}/{}.jpg".format(hero_name, skin_info["name"]), "wb") as f: f.write(reponse.content) self.logger.info("{} 下載完成!".format(skin_info["name"])) self.trigger.emit("{} 下載完成!".format(skin_info["name"])) except: self.logger.error(skin_info["name"] + ',下載出現(xiàn)異常.跳過(guò)執(zhí)行下一個(gè)!') self.trigger.emit(skin_info["name"] + ',下載出現(xiàn)異常.跳過(guò)執(zhí)行下一個(gè)!') self.sleep(round(random(), 5))
由于下載這一塊的業(yè)務(wù)比較多,為了方便大家查看。基本上主要的代碼塊我都寫上了注釋,需要的小伙伴可以根據(jù)自己的需求進(jìn)行改造。
完整代碼
# -*- coding:utf-8 -*- # @author Python 集中營(yíng) # @date 2022/1/7 # @file test7.py # done '''英雄聯(lián)盟皮膚下載相關(guān)依賴模塊''' import requests # 網(wǎng)絡(luò)請(qǐng)求庫(kù) import re # 正則表達(dá)式匹配庫(kù) import json # JSON格式轉(zhuǎn)換庫(kù) import os # 應(yīng)用操作庫(kù) import time # 時(shí)間模塊 from random import random # 隨機(jī)數(shù)模塊 from fake_useragent import UserAgent # user_agent 生成庫(kù) import logging # 日志模塊 import sys # 系統(tǒng)操作 '''UI 界面相關(guān)依賴模塊''' # pyqt5的模塊引用這里就不介紹了,最近一直在用。 from PyQt5.QtGui import * from PyQt5.QtWidgets import * from PyQt5.QtCore import * class HeroSkin(QWidget): def __init__(self): super(HeroSkin, self).__init__() self.cwd = os.getcwd() self.init_ui() def init_ui(self): self.setWindowTitle('英雄皮膚下載器 公眾號(hào):[Python 集中營(yíng)]') self.setWindowIcon(QIcon('lol.ico')) self.resize(500,250) vbox = QVBoxLayout() self.save_dir = QLineEdit() self.save_dir.setReadOnly(True) self.save_btn = QPushButton() self.save_btn.setText('路徑') self.save_btn.clicked.connect(self.save_btn_click) self.thread_ = DownLoadThread(self) self.thread_.trigger.connect(self.update_log) self.start_btn = QPushButton() self.start_btn.setText('開始下載') self.start_btn.clicked.connect(self.start_btn_click) grid = QGridLayout() grid.addWidget(self.save_dir, 0, 0, 1, 2) grid.addWidget(self.save_btn, 0, 2, 1, 1) grid.addWidget(self.start_btn, 0, 3, 1, 1) self.result_brower = QTextBrowser() self.result_brower.setFont(QFont('宋體', 8)) self.result_brower.setReadOnly(True) self.result_brower.setPlaceholderText('英雄皮膚批量下載進(jìn)度顯示區(qū)域...') self.result_brower.ensureCursorVisible() vbox.addWidget(self.result_brower) vbox.addLayout(grid) self.setLayout(vbox) def start_btn_click(self): self.start_btn.setEnabled(False) self.thread_.start() def update_log(self, text): cursor = self.result_brower.textCursor() cursor.movePosition(QTextCursor.End) self.result_brower.append(text) self.result_brower.setTextCursor(cursor) self.result_brower.ensureCursorVisible() def save_btn_click(self): directory = QFileDialog.getExistingDirectory(self, "選取文件夾", self.cwd) self.save_dir.setText(directory) class DownLoadThread(QThread): trigger = pyqtSignal(str) def __init__(self, parent=None): super(DownLoadThread, self).__init__(parent) # 初始化日志對(duì)象 self.logger = logging.getLogger('英雄聯(lián)盟皮膚') logging.basicConfig() self.logger.setLevel(logging.DEBUG) self.parent = parent self.working = True def __del__(self): self.working = False self.wait() def run(self): ''' 英雄聯(lián)盟皮膚下載函數(shù) :return: ''' while self.working == True: # 構(gòu)造useragent身份設(shè)備信息 headers = { "User-Agent": str(UserAgent().random), } self.logger.info('生成身份設(shè)備信息完成') self.trigger.emit('生成身份設(shè)備信息完成!') # 在LOL的官網(wǎng)經(jīng)過(guò)分析champion.js中包含了我們要用到的英雄ID編號(hào),因此,將這個(gè)JS文件下載下來(lái) champion_js_url = "https://lol.qq.com/biz/hero/champion.js" # 發(fā)送http請(qǐng)求、請(qǐng)求地址為這個(gè)JS文件的地址。就會(huì)將其下載下來(lái)了。 response = requests.get(url=champion_js_url, headers=headers) self.logger.info('champion.js 文件下載完成') self.trigger.emit('champion.js 文件下載完成!') # 獲取下載的文本信息 text = response.text self.logger.info('champion.js 文本信息:' + text) # 匹配champion對(duì)象對(duì)應(yīng)的JSON數(shù)據(jù),該JSON數(shù)據(jù)的第0個(gè)位置是包含英雄ID編號(hào)的JSON數(shù)據(jù) hero_json = re.findall(r'champion=(.*?);', text, re.S)[0] # 將JSON數(shù)據(jù)轉(zhuǎn)換為dict字典 hero_dict = json.loads(hero_json) self.logger.info('英雄ID字典信息:' + str(hero_dict)) # 逐個(gè)英雄信息遍歷 for hero_data in hero_dict["data"].items(): # 獲取英雄詳細(xì)信息的JS文件 hero_js_url = "http://lol.qq.com/biz/hero/{}.js" # 發(fā)送請(qǐng)求下載該JS文件、hero_data[0]取第0位就是英雄ID response = requests.get(url=hero_js_url.format(hero_data[0]), headers=headers) # 獲取下載文本 text = response.text self.logger.info(hero_data[0] + '.js 文本信息:' + text) self.trigger.emit(hero_data[0] + '.js 文件下載完成!') skins_dict = json.loads(re.findall("{}=(.*?);".format(hero_data[0]), text, re.S)[0]) self.logger.info('當(dāng)前英雄皮膚字典:' + str(skins_dict)) # 從字典 skins_dict 獲取皮膚列表 skins_list = skins_dict["data"]["skins"] # 獲取英雄名稱 hero_name = hero_data[1]["name"] # 在當(dāng)前目錄下面創(chuàng)建images文件夾、以英雄名稱作為文件夾名稱 os.makedirs(r"./images/{}".format(hero_name), exist_ok=True) for skin_info in skins_list: # 初始化皮膚圖片地址 skin_pic_url = "https://ossweb-img.qq.com/images/lol/web201310/skin/big{}.jpg" # 發(fā)送下載請(qǐng)求 reponse = requests.get(url=skin_pic_url.format(skin_info["id"]), headers=headers) try: self.logger.info('保存路徑' + self.parent.save_dir.text().strip()) # 保存皮膚 with open(r"" + self.parent.save_dir.text().strip() + "/{}/{}.jpg".format(hero_name, skin_info["name"]), "wb") as f: f.write(reponse.content) self.logger.info("{} 下載完成!".format(skin_info["name"])) self.trigger.emit("{} 下載完成!".format(skin_info["name"])) except: self.logger.error(skin_info["name"] + ',下載出現(xiàn)異常.跳過(guò)執(zhí)行下一個(gè)!') self.trigger.emit(skin_info["name"] + ',下載出現(xiàn)異常.跳過(guò)執(zhí)行下一個(gè)!') self.sleep(round(random(), 5)) if __name__ == '__main__': app = QApplication(sys.argv) main = HeroSkin() main.show() sys.exit(app.exec_())
以上就是基于PyQT5制作英雄聯(lián)盟全皮膚下載器的詳細(xì)內(nèi)容,更多關(guān)于PyQT5皮膚下載器的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Python棧算法的實(shí)現(xiàn)與簡(jiǎn)單應(yīng)用示例
這篇文章主要介紹了Python棧算法的實(shí)現(xiàn)與簡(jiǎn)單應(yīng)用,簡(jiǎn)單講述了棧的原理并結(jié)合實(shí)例形式給出了基于棧實(shí)現(xiàn)的進(jìn)制轉(zhuǎn)換與括號(hào)匹配等相關(guān)使用技巧,需要的朋友可以參考下2017-11-11python pyinstaller打包exe報(bào)錯(cuò)的解決方法
這篇文章主要給大家介紹了關(guān)于python pyinstaller打包exe報(bào)錯(cuò)的解決方法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者使用python具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-11-11Python庫(kù)中可以操作JavaScript盤點(diǎn)解析
這篇文章主要為大家介紹了Python庫(kù)之可以操作JavaScript盤點(diǎn)解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-06-06python和C++共享內(nèi)存?zhèn)鬏攬D像的示例
這篇文章主要介紹了python和C++共享內(nèi)存?zhèn)鬏攬D像的示例,幫助大家利用python處理圖片,感興趣的朋友可以了解下2020-10-10詳談在flask中使用jsonify和json.dumps的區(qū)別
下面小編就為大家分享一篇詳談在flask中使用jsonify和json.dumps的區(qū)別,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-03-03使用tf.keras.MaxPooling1D出現(xiàn)錯(cuò)誤問(wèn)題及解決
這篇文章主要介紹了使用tf.keras.MaxPooling1D出現(xiàn)錯(cuò)誤問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12Python利用socket模塊開發(fā)簡(jiǎn)單的端口掃描工具的實(shí)現(xiàn)
這篇文章主要介紹了Python利用socket模塊開發(fā)簡(jiǎn)單的端口掃描工具的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01一次性徹底講透Python中pd.concat與pd.merge
本文主要介紹了一次性徹底講透Python中pd.concat與pd.merge,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06