如何从 PyQt 的 QWebEnginePage.runJavaScript() 获取返回值 [英] How to get return Value from PyQt's QWebEnginePage.runJavaScript()

查看:552
本文介绍了如何从 PyQt 的 QWebEnginePage.runJavaScript() 获取返回值的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

我正在创建一个简单的 QWebEngineView,我试图通过调用 js 函数来检索字符串,但是我还没有找到这样做的方法.

I'm creating a simple QWebEngineView where I'm trying to retrieve a string by calling a js function, however I haven't found a way doing so.

这是一个仅调用 js 函数的工作示例

This is a working example with just calling a js function

from PySide2.QtWebEngineWidgets import QWebEngineView


class View:
    def __init__(self):
       self.view = QWebEngineView()
       self.view.load(QUrl("https://mytestpage.com"))
       self.view.show()


def callback(a):
    print a
if __name__ == '__main__': 
    view = View()
    view.view.page().runJavaScript("window.getMail()", callback)

getMail 在浏览器上执行,但是根据文档,要获得结果,我需要传入回调函数作为第二个参数,但是这样做会产生:

getMail is executed on the browser, however according to the doc's, to get the result, I would need to pass in a callback function as a second argument, however doing so yields a:

TypeError: `runJavaScript() takes exactly one argument (2 given) # 

推荐答案

一个可能的解决方案是注入一个 QObject,它允许使用 Qt WebChannel 与 DOM 通信:

A possible solution is to inject a QObject that allows to communicate with the DOM using Qt WebChannel:

import os
import sys
from PySide2 import QtCore, QtWidgets, QtWebEngineWidgets, QtWebChannel
from jinja2 import Template


class Element(QtCore.QObject):
    loaded = QtCore.Signal()

    def __init__(self, name, parent=None):
        super(Element, self).__init__(parent)
        self._name = name
        self._is_loaded = False

    @property
    def name(self):
        return self._name

    @property
    def is_loaded(self):
        return self._is_loaded

    @QtCore.Slot()
    def set_loaded(self):
        self._is_loaded = True
        self.loaded.emit()

    def render_script(self, script, **kwargs):
        kwargs["name"] = self.name
        return Template(script).render(**kwargs)


class TestObject(Element):
    @QtCore.Slot(str)
    def test(self, res):
        print(res)


class WebEnginePage(QtWebEngineWidgets.QWebEnginePage):
    def __init__(self, *args, **kwargs):
        super(WebEnginePage, self).__init__(*args, **kwargs)
        self.loadFinished.connect(self.onLoadFinished)
        self._objects = []

    def add_object(self, obj):
        self._objects.append(obj)

    @QtCore.Slot(bool)
    def onLoadFinished(self, ok):
        if ok:
            self.load_qwebchannel()
            self.load_objects()

    def load_qwebchannel(self):
        file = QtCore.QFile(":/qtwebchannel/qwebchannel.js")
        if file.open(QtCore.QIODevice.ReadOnly):
            content = file.readAll()
            file.close()
            self.runJavaScript(content.data().decode())
        if self.webChannel() is None:
            channel = QtWebChannel.QWebChannel(self)
            self.setWebChannel(channel)

    def load_objects(self):
        if self.webChannel() is not None:
            objects = {obj.name: obj for obj in self._objects}
            self.webChannel().registerObjects(objects)
            _script = r"""
            {% for obj in objects %}
            var {{obj}} = null;
            {% endfor %}
            new QWebChannel(qt.webChannelTransport, function (channel) {
                {% for obj in objects %}
                    {{obj}} = channel.objects.{{obj}};
                    {{obj}}.set_loaded()
                {% endfor %}
            }); 
            """
            self.runJavaScript(Template(_script).render(objects=objects.keys()))


class WebPage(QtWebEngineWidgets.QWebEngineView):
    def __init__(self, parent=None):
        super().__init__(parent)

        page = WebEnginePage(self)
        self.setPage(page)

        self.test_object = TestObject("test_object", self)
        self.test_object.loaded.connect(self.test_object_loaded)
        page.add_object(self.test_object)

        self.load(QtCore.QUrl("https://mytestpage.com"))

    @QtCore.Slot()
    def test_object_loaded(self):
        script = self.test_object.render_script(
            r"""
        {{name}}.test(window.getMail());
        """
        )
        self.page().runJavaScript(script)


if __name__ == "__main__":
    app = QtWidgets.QApplication(sys.argv)
    web = WebPage()
    web.show()
    sys.exit(app.exec_())

这篇关于如何从 PyQt 的 QWebEnginePage.runJavaScript() 获取返回值的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆