有关Qt的WebEngine模块与JS交互的研究(二)



  • 0_1524459226096_131313.gif
    大家上午好哦,上一篇文章简单介绍了我对WebEngine和JS交互相关的理解,这篇文章则是介绍了如何实验如何制作一个小例子,实现WebEngine和JS的交互。

    这个例子的截图是这样的:
    0_1524455742527_eb728e46-d777-4d39-b620-52a5d7e6c991-image.png

    我们看一下文件的结构:

    webengine_2.pro
    main.cpp
    html/libs/bootstrap.min.css
    html/index.html
    qml/main.qml
    html.qrc
    qml.qrc

    文件的结构还是非常简单的。由于我们对按钮实行了美化,因此需要依赖bootstrap.min.css文件,除此之外,就是普通的html文件了。这里需要关注的核心就是index.html文件和main.qml文件了。因为我们的通信主要是借助Chromium和Qt的C++代码,在QML环境和html环境中做通信。

    首先是index.html文件:

    <!doctype html>
    <html lang="en">
    <head>
        <meta charset="UTF-8">
        <title>bootsrap按钮</title>
        <link rel="stylesheet" href="libs/bootstrap.min.css">
        <script type="text/javascript" src="qrc:///qtwebchannel/qwebchannel.js"></script>
    </head>
    <body class="vert-center-parent">
        <div class="container text-center vert-center">
            <button id="deleteButton" class="btn btn-danger">删除所有文件</button>
        </div>
    </body>
    <style>
    .vert-center
    {
        line-height: 600px;
    }
    .vert-center-parent
    {
        height: 600px;
    }
    </style>
    <script type="text/javascript">
    window.onload = function ( )
    {
        new QWebChannel( qt.webChannelTransport, function( channel ) {
            //makedialogobjectaccessibleglobally
            var fileField = channel.objects.fileField;
            var deleteButton = document.getElementById( "deleteButton" );
            deleteButton.onclick = function( ) {
                alert( "删除了" + fileField.text + "文件。" );
            }
        } );
    }
    </script>
    </html>
    

    因为我们需要使用WebChannel来找到QML环境中的变量,因此需要通过

    <script type="text/javascript" src="qrc:///qtwebchannel/qwebchannel.js"></script>
    

    的方式引入Qt自己搭建的JS方法。如果想在纯Chrome或者是其它浏览器上运行这个网页,那么src属性不能是以qrc为前缀了,需要将Qt这个文件复制到合适的位置才行。



  • 在页面加载的过程中,通过设置window.onload回调函数来执行我们自己的逻辑。其内容主要是创建QWebChannel对象,并且在回调函数中拿到QML对象的Id,这样就像QML环境中使用该对象了。我为了演示的便利,使用alert方法删除一个文件。文件的内容包含的是fileField的属性text,这和QML环境中访问同名属性一致。
    还有一点值得注意的是,因为我们确信网页是执行在WebEngineView中的,因此我们在建立QWebChannel的时候使用Qt在WebEngine中默认的传输管道qt.webChannelTransport就行了,内部看文档介绍是使用进程间通信(IPC)的方法。如果是在纯浏览器中建立QWebChannel,那么就需要使用WebSocket来达到目的了,查一查目标浏览器是否支持WebSocket

    接下来介绍的就是main.qml文件。

    import QtQuick 2.10
    import QtQuick.Window 2.10
    import QtQuick.Controls 2.3
    import QtQuick.Layouts 1.3
    import QtWebEngine 1.5
    import QtWebChannel 1.0
    
    Window
    {
        id: window
        width: 640
        height: 480
        visible: true
        title: "WebEngine的例子 2"
    
        RowLayout
        {
            anchors.fill: parent
            ColumnLayout
            {
                Layout.fillHeight: true
                Button
                {
                    text: "进入一个地址"
                    onClicked:
                    {
                        webEngineView.url = urlField.text;
                    }
                }
                TextField
                {
                    id: urlField
                    selectByMouse: true
                    text: "qrc:///html/index.html"
                }
                Label
                {
                    text: "服务器中的文件"
                }
                TextField
                {
                    id: fileField
                    WebChannel.id: "fileField"
                    text: "HelloWorld.txt"
                    selectByMouse: true
                }
            }
            WebEngineView
            {
                id: webEngineView
                Layout.fillWidth: true
                Layout.fillHeight: true
                url: "qrc:///html/index.html"
                webChannel: WebChannel
                {
                    registeredObjects: [ fileField ]
                }
            }
        }
    }
    

    Window代码块前面是一些控件,不必多言。注意的是id名为fileField的控件。这个控件有一个附加属性:WebChannel.id。这个id就是我们需要从html中拿到的id或者说属性,不这么设定的话,我们是无法在html中找到这个对象的。
    最后则是WebEngineView,它的后端使用的是WebEngine,并且指定了webChannel,它有一个属性registeredObjects,指向的是已经注册过的对象。这里的对象就包含了fileField,也只有注册过的对象才能暴露在html环境中。



  • 其余的文件则是初始化必要的文件。也不必多言,除了main.cpp,它里面要使用QtWebEngine::initialize( )初始化一下。main.cpp的内容是:

    #include <QGuiApplication>
    #include <QQmlApplicationEngine>
    #include <QQuickWindow>
    #include <QtWebEngine>
    
    int main( int argc, char *argv[] )
    {
        QGuiApplication a( argc, argv );
    
        QtWebEngine::initialize( );
    
        // 设置字体渲染方式
        QQuickWindow::setTextRenderType( QQuickWindow::NativeTextRendering );
    
        QQmlApplicationEngine engine;
        engine.load( QUrl( "qrc:///qml/main.qml" ) );
    
        return a.exec();
    }
    

    很简单吧。程序运行的时候,我们点击“删除所有文件”按钮,就会弹出一个对话框,里面显示的是我们在TextField中设定的文字。运行符合预期,说明我们的尝试成功了。
    0_1524459184313_a783721a-2141-4627-b374-b969da346699-image.png



  • 今天没有时间弄了,明天试试



  • @大黄老鼠 嗯,也是非常简单的一个例子啦。



  • @jiangcaiyang 博主您好 谢谢您的贡献,按照您的代码可以运行,但是如果 我想调用qml 页面中的一个函数 function 该怎么办呢,谢谢博主



  • @倦鸟归巢 WebChannel可以让双端解耦。这样的话,通过QML调用Html的JS函数以及JS函数调用QML的函数都是可行的。


Log in to reply
 

走马观花

最近的回复

  • E

    是不是好久都没有人了,站主找到工作了,还是说划水了

    read more
  • 请问一下如果原Qt桌面软件支持拖拽外部文件进去(比如把在用资源管理器里的一张图片拖拽到软件界面上就会显示这张图片),转成Webassembly之后这种拖拽操作还是否有效?劳烦大佬帮我试试看,可以的话我就学QML了

    read more
  • H
    Toou 2D 拿来即用,为简单而生。

    简称T2D,是一款采用自身模块规范编写的轻量级UI框架,遵循Qt书写与组织形式门槛极低无需深入学习简单易用可拿来即用,丰富的控件模块适合界面的快速开发,让程序人员拥有更多的精力来实现业务逻辑与算法。

    统一交互规范,丰富的Ui控件几十种常用控件放弃了Qt Controls 及 Controls 2 来提高性能。

    完善的主题系统,业务逻辑与界面主题设计分离,可通过简单修改变量自定义主题皮肤。灵活的多主题皮肤绑定机制、在不需要重启App即实现一键换肤

    ini皮肤配置规则与每一个控件融合。可在应用内配置也可在应用外动态扩展配置。

    框架自动化安装支持动态库、静态库多模式编译。使用方便更安全更自由。

    提供丰富Demo、全面的帮助文档,Api查阅快速方便。项目必备开源框架!

    已经集成最新版 Font Awesome 4.7

    开源地址:https://github.com/ShowFL/Toou-2D

    read more
  • 刚刚毕业,工作用QT开发,以后有问题多多请教各位前辈😬 抱拳了。

    read more

关注我们

微博
QQ群