摘要:概要應(yīng)同學(xué)邀請,演示如何使用內(nèi)嵌瀏覽器瀏覽網(wǎng)頁,并注入腳本實(shí)現(xiàn)自動(dòng)化操作。在默認(rèn)配置中植入內(nèi)容,這樣腳本會(huì)在所有打開的網(wǎng)頁中執(zhí)行,不論跳轉(zhuǎn)到哪個(gè)網(wǎng)址。腳本使用網(wǎng)址中的路徑名,判斷當(dāng)前網(wǎng)頁位置,從而決定執(zhí)行哪種操作。
概要
應(yīng)同學(xué)邀請,演示如何使用 PyQt5 內(nèi)嵌瀏覽器瀏覽網(wǎng)頁,并注入 Javascript 腳本實(shí)現(xiàn)自動(dòng)化操作。
sg 原貼地址: 如何在Python利用runJavaScript模擬鼠標(biāo)移動(dòng)頁面的某個(gè)元素
https://segmentfault.com/q/10...
下面測試的是一個(gè)廉價(jià)機(jī)票預(yù)訂網(wǎng)站(http://www.flyscoot.com/),關(guān)鍵點(diǎn)如下
使用 QWebEngineView 加載網(wǎng)頁,并顯示進(jìn)度。
在默認(rèn)配置(QWebEngineProfile)中植入 Javascript 內(nèi)容,這樣腳本會(huì)在所有打開的網(wǎng)頁中執(zhí)行,不論跳轉(zhuǎn)到哪個(gè)網(wǎng)址。
Javascript 腳本使用網(wǎng)址中的路徑名,判斷當(dāng)前網(wǎng)頁位置,從而決定執(zhí)行哪種操作。
python 代碼示例#!/usr/bin/env python3 # -*- coding: utf-8 -*- """使用 PyQt5 內(nèi)嵌瀏覽器瀏覽網(wǎng)頁,并注入 Javascript 腳本實(shí)現(xiàn)自動(dòng)化操作。""" import os import sys from datetime import datetime from PyQt5.QtWidgets import ( QWidget, QApplication, QVBoxLayout, QHBoxLayout, QDesktopWidget, QTextEdit, QLabel, QLineEdit, QPushButton, QFileDialog, QProgressBar, ) from PyQt5.QtCore import QUrl, pyqtSlot from PyQt5.QtWebEngineWidgets import QWebEngineView, QWebEngineProfile, QWebEngineScript, QWebEnginePage class Browser(QWidget): def __init__(self): super().__init__() self.init_ui() # 腳本 self.profile = QWebEngineProfile.defaultProfile() self.script = QWebEngineScript() self.prepare_script() def init_ui(self): self.webView = QWebEngineView() self.logEdit = QTextEdit() self.logEdit.setFixedHeight(100) self.addrEdit = QLineEdit() self.addrEdit.returnPressed.connect(self.load_url) self.webView.urlChanged.connect( lambda i: self.addrEdit.setText(i.toDisplayString())) self.jsEdit = QLineEdit() self.jsEdit.setText("inject.js") loadUrlBtn = QPushButton("加載") loadUrlBtn.clicked.connect(self.load_url) chooseJsBtn = QPushButton("選擇腳本文件") chooseJsBtn.clicked.connect(self.choose_js_file) # 導(dǎo)航/工具 top = QWidget() top.setFixedHeight(80) topBox = QVBoxLayout(top) topBox.setSpacing(0) topBox.setContentsMargins(5, 0, 0, 5) progBar = QProgressBar() progBox = QHBoxLayout() progBox.addWidget(progBar) topBox.addLayout(progBox) naviBox = QHBoxLayout() naviBox.addWidget(QLabel("網(wǎng)址")) naviBox.addWidget(self.addrEdit) naviBox.addWidget(loadUrlBtn) topBox.addLayout(naviBox) naviBox = QHBoxLayout() naviBox.addWidget(QLabel("注入腳本文件")) naviBox.addWidget(self.jsEdit) naviBox.addWidget(chooseJsBtn) topBox.addLayout(naviBox) self.webView.loadProgress.connect(progBar.setValue) # 主界面 layout = QVBoxLayout(self) layout.addWidget(self.webView) layout.addWidget(top) layout.addWidget(self.logEdit) self.show() self.resize(1024, 900) self.center() def center(self): qr = self.frameGeometry() cp = QDesktopWidget().availableGeometry().center() qr.moveCenter(cp) self.move(qr.topLeft()) @pyqtSlot() def load_url(self): url = self.addrEdit.text().strip() if not url.lower().startswith("http://") and not url.lower().startswith("https://"): url = "http://{}".format(url) self.load(url) @pyqtSlot() def choose_js_file(self): f, _ = QFileDialog.getOpenFileName(filter="Javascript files(*.js)") if os.path.isfile(f): self.jsEdit.setText(f) self.prepare_script() def prepare_script(self): path = self.jsEdit.text().strip() if not os.path.isfile(path): self.log("invalid js path") return self.profile.scripts().remove(self.script) with open(path, "r") as f: self.script.setSourceCode(f.read()) self.profile.scripts().insert(self.script) self.log("injected js ready") def log(self, msg, *args, **kwargs): m = msg.format(*args, **kwargs) self.logEdit.append("{} {}".format( datetime.now().strftime("%H:%M:%S"), m)) def load(self, url): self.log(f"loading {url}") self.addrEdit.setText(url) self.webView.load(QUrl(url)) if __name__ == "__main__": app = QApplication(sys.argv) b = Browser() b.load("http://www.flyscoot.com/") sys.exit(app.exec_())Javascript 腳本示例
// 簡單起見,這里只演示部分頁面,腳本內(nèi)容摘自 Heng丶原貼文。 function handle(path) { // 首頁 if (path == "/zh") { document.getElementsByClassName("radio-inline")[1].click(); document.getElementById("oneway_from").value="廣州 (CAN)"; document.getElementById("oneway_to").value="新加坡 (SIN)"; document.getElementById("oneway_departuredate").value="2018年9月10日"; document.getElementsByClassName("btn--booking")[1].click(); return; } // 選擇航班 if (path == "/Book/Flight") { document.getElementsByClassName("price--sale")[0].click(); document.getElementsByClassName("heading-4")[0].click(); document.getElementsByClassName("btn-submit")[0].click(); return; } // 乘客信息 if (path == "/BookFlight/Passengers") { document.getElementsByClassName("fname1")[0].value = "匿名"; } } let host = document.location.hostname; if (host.endsWith(".flyscoot.com")) { handle(document.location.pathname); }
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://systransis.cn/yun/42341.html
摘要:跨域請求跨域問題,是開發(fā)中一直以來需要注意的問題。則發(fā)送通道的發(fā)送給監(jiān)聽此消息的外部擴(kuò)展。完整方法此處,為必選參數(shù),為回調(diào)函數(shù)?;卣{(diào)函數(shù)接收到的參數(shù)有三個(gè),分別是和,即消息內(nèi)容,消息發(fā)送者相關(guān)信息和相應(yīng)函數(shù)。是發(fā)起消息的標(biāo)簽。 第二章簡介 第二章較第一章復(fù)雜許多。書中雖寥寥21頁,內(nèi)容卻也不少。涵蓋了8個(gè)小節(jié)。 2.1 操作用戶正在瀏覽的頁面 2.2 跨域請求 2.3 常駐后臺 2....
摘要:和類似的預(yù)處理器還有等。的用處非常多,包括給自動(dòng)加前綴使用下一代語法等,目前越來越多的人開始用它,它很可能會(huì)成為預(yù)處理器的最終贏家。 webpack實(shí)戰(zhàn) 查看所有文檔頁面:全棧開發(fā),獲取更多信息??祚R加鞭,加班加點(diǎn),終于把這個(gè)文檔整理出來了,順便深入地學(xué)習(xí)一番,鞏固知識,就是太累人,影響睡眠時(shí)間和質(zhì)量。極客就是想要把事情做到極致,開始了就必須到達(dá)終點(diǎn)。 原文鏈接:webpack實(shí)戰(zhàn),原...
摘要:在瀏覽區(qū)中的性能,可以認(rèn)為是開發(fā)者所面臨的最嚴(yán)重的可用性問題。優(yōu)化這個(gè)問題的第一步從它的加載和執(zhí)行開始。這意味著在對象的事件觸發(fā)后再下載腳本。屬性指明本元素所含的腳本不會(huì)修改,因此代碼能夠安全地執(zhí)行,但是瀏覽器的支持情況不理想。 JavaScript在瀏覽區(qū)中的性能,可以認(rèn)為是開發(fā)者所面臨的最嚴(yán)重的可用性問題。 優(yōu)化這個(gè)問題的第一步從它的加載和執(zhí)行開始。 霸道的script標(biāo)簽scr...
摘要:是代替用戶完成指定的動(dòng)作,需要知道其他用戶頁面的代碼和數(shù)據(jù)包。 常見web安全及防護(hù)原理 sql注入原理 就是通過把SQL命令插入到Web表單遞交或輸入域名或頁面請求的查詢字符串,最終達(dá)到欺騙服務(wù)器執(zhí)行惡意的SQL命令 總的來說有以下幾點(diǎn) 永遠(yuǎn)不要信任用戶的輸入,要對用戶的輸入進(jìn)行校驗(yàn),可以通過正則表達(dá)式,或限制長度,對單引號和雙-進(jìn)行轉(zhuǎn)換等 永遠(yuǎn)不要使用動(dòng)態(tài)拼裝SQL,可以...
閱讀 2646·2021-10-14 09:47
閱讀 4940·2021-09-22 15:52
閱讀 3362·2019-08-30 15:53
閱讀 1458·2019-08-30 15:44
閱讀 689·2019-08-29 16:41
閱讀 1659·2019-08-29 16:28
閱讀 449·2019-08-29 15:23
閱讀 1628·2019-08-26 12:20