成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

PyQt5 內(nèi)嵌瀏覽器注入 Javascript 腳本實(shí)現(xiàn)自動(dòng)化操作

xiguadada / 2112人閱讀

摘要:概要應(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

相關(guān)文章

  • chrome擴(kuò)展開發(fā)之旅 第二篇

    摘要:跨域請求跨域問題,是開發(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....

    shadajin 評論0 收藏0
  • webpack實(shí)戰(zhàn)

    摘要:和類似的預(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),原...

    cyrils 評論0 收藏0
  • JavaScript性能優(yōu)化之加載與執(zhí)行

    摘要:在瀏覽區(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...

    call_me_R 評論0 收藏0
  • 延遲腳本的方式

    摘要:實(shí)現(xiàn)延遲加載的方式有方式一標(biāo)簽放在底部將節(jié)點(diǎn)放置在之前,這樣腳本會(huì)在頁面顯示出來之后再加載。盡可能地合并腳本。采用無阻塞下載腳本的方法使用標(biāo)簽的屬性僅適用于和以上版本使用動(dòng)態(tài)創(chuàng)建的元素來下載并執(zhí)行代碼使用對象下載代碼并注入頁面中。 昨晚‘今日頭條’筆試題中遇到這道題,今天專門查了一下。 從 IE 8、Firefox 3.5、Safari 4 和 Chrome 2 開始都允許并行下載 J...

    tinylcy 評論0 收藏0
  • web security

    摘要:是代替用戶完成指定的動(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,可以...

    livem 評論0 收藏0

發(fā)表評論

0條評論

最新活動(dòng)
閱讀需要支付1元查看
<