Qt嵌入瀏覽器(二)——QWebChannel實(shí)現(xiàn)與頁(yè)面的通信

本篇簡(jiǎn)介

本篇的小目標(biāo):

  • 借助QWebChannel明郭,實(shí)現(xiàn)瀏覽器與頁(yè)面的通信交互

上一節(jié)中我們實(shí)現(xiàn)了瀏覽器的基本功能和一個(gè)簡(jiǎn)易的調(diào)試界面糠睡。>>點(diǎn)這里回顧上節(jié)內(nèi)容

本節(jié)將嘗試在此基礎(chǔ)上實(shí)現(xiàn)瀏覽器與所加載頁(yè)面的相互通信迫靖。既然是相互通信膳灶,自然包含兩個(gè)方向:

  • 頁(yè)面向?yàn)g覽器發(fā)送消息
  • 瀏覽器向頁(yè)面發(fā)送消息

頁(yè)面向?yàn)g覽器發(fā)送消息

為了實(shí)現(xiàn)頁(yè)面向?yàn)g覽器發(fā)送消息悔常,需要在一定程度上實(shí)現(xiàn)在頁(yè)面中訪問Qt對(duì)象循帐。Qt官方的推薦方式是使用QWebChannel。借助官方提供的qwebchannel.js腳本惠遏,可以很方便地實(shí)現(xiàn)頁(yè)面對(duì)Qt對(duì)象的調(diào)用砾跃,同時(shí)便于實(shí)現(xiàn)C++和html/js的解耦,方便使用不同技術(shù)棧的開發(fā)人員的分工节吮。

注:qwebchannel.js在Qt發(fā)布版的文件夾下可以找到抽高。

Qt這邊,為了方便后續(xù)開發(fā)透绩,我們先設(shè)計(jì)一個(gè)JsContext類翘骂,聲明如下:

class JsContext : public QObject
{
    Q_OBJECT
public:
    explicit JsContext(QObject *parent = nullptr);

signals:
    void recvdMsg(const QString& msg);

public:
    // 向頁(yè)面發(fā)送消息
    void sendMsg(QWebEnginePage* page, const QString& msg);

public slots:
    // 接收到頁(yè)面發(fā)送來(lái)的消息
    void onMsg(const QString& msg);
};

這里需要注意的是,onMsg這個(gè)方法是提供給js直接調(diào)用的方法帚豪,按照Qt官方的說明碳竟,這個(gè)方法必須被聲明為一個(gè)slot。onMsg方法的具體實(shí)現(xiàn)如下:

void JsContext::onMsg(const QString &msg)
{
    emit recvdMsg(msg);
}

也就是發(fā)送了一個(gè)“已接收到消息”的信號(hào)志鞍,供感興趣的其他組件接收瞭亮。簡(jiǎn)便起見,我們直接把接收這個(gè)信號(hào)的部分實(shí)現(xiàn)到上一節(jié)中實(shí)現(xiàn)好的主界面初始化方法中——在接收到頁(yè)面發(fā)送的消息后固棚,把接收到的消息顯示在狀態(tài)欄里,實(shí)現(xiàn)如下:

m_jsContext = new JsContext(this);
m_webChannel = new QWebChannel(this);
m_webChannel->registerObject("context", m_jsContext);
m_webView->page()->setWebChannel(m_webChannel);
connect(m_jsContext, &JsContext::recvdMsg, this, [this](const QString& msg) {
    ui->statusBar->showMessage(QString("Received message:%1").arg(msg), 3000);
});

上面的實(shí)現(xiàn)里包含了QWebChannel的使用步驟:

  1. 創(chuàng)建QWebChannel對(duì)象;
  2. 向QWebChannel對(duì)象注冊(cè)Qt對(duì)象仙蚜。 例子里的Qt對(duì)象就是上面實(shí)現(xiàn)的JsContext對(duì)象此洲。
    注意:這里注冊(cè)時(shí)所用的名字“context”,就是js獲取qt對(duì)象時(shí)所使用的索引名委粉。
  3. 將設(shè)置好的QWebChannel對(duì)象設(shè)置為當(dāng)前頁(yè)面的通道呜师。即調(diào)用QWebEnginePage的setWebChannel方法。
    注意:如果沒有正確注冊(cè)webchannel贾节,在js調(diào)用qt對(duì)象時(shí)會(huì)出現(xiàn)qt對(duì)象為undefined的情況汁汗。

接下來(lái)我們來(lái)看看頁(yè)面這邊要如何實(shí)現(xiàn)對(duì)應(yīng)的接口衷畦。首先實(shí)現(xiàn)一個(gè)簡(jiǎn)單的js工具包msgutils.js:

var context;
// 初始化
function init()
{
    if (typeof qt != 'undefined')
    {
        new QWebChannel(qt.webChannelTransport, function(channel)
        {
        context = channel.objects.context;
        }
        );
    }
    else
    {
        alert("qt對(duì)象獲取失敗知牌!");
    }
}
// 向qt發(fā)送消息
function sendMessage(msg)
{
    if(typeof context == 'undefined')
    {
        alert("context對(duì)象獲取失斊碚!");
    }
    else
    {
        context.onMsg(msg);
    }
}
// 控件控制函數(shù)
function onBtnSendMsg()
{
    var cmd = document.getElementById("待發(fā)送消息").value;
    sendMessage(cmd);   
}
init();

這里有幾點(diǎn)需要注意的地方:

  • 初始化時(shí)創(chuàng)建了一個(gè)QWebChannel對(duì)象角寸,里面的第一個(gè)參數(shù)qt.webChannelTransport菩混,只有Qt端正確地向頁(yè)面設(shè)置了webchannel才能取到,否則會(huì)是undefined扁藕。
  • QWebChannel對(duì)象里的channel.objects對(duì)應(yīng)了上面Qt實(shí)現(xiàn)里向webchannel注冊(cè)的對(duì)象表沮峡,channel.objects.context即為從表中取出名為"context"的對(duì)象。
  • context.onMsg即為對(duì)context對(duì)象所提供的onMsg方法的調(diào)用亿柑。如果該對(duì)象沒有提供這個(gè)方法邢疙,或者這個(gè)方法沒有被聲明為一個(gè)slot,則Qt端會(huì)報(bào)找不到方法的錯(cuò)望薄。

以這個(gè)腳本為基礎(chǔ)秘症,實(shí)現(xiàn)一個(gè)簡(jiǎn)單的交互頁(yè)面:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>webchannel測(cè)試</title>
</head>
<body>
    <p>webchannel test</p>
    <script type="text/javascript" src="./qwebchannel.js"></script>
    <script type="text/javascript" src="./msgutils.js"></script>
    <input id="待發(fā)送消息" type="text" name="msgText" />
    <input type="button" value="發(fā)送消息到瀏覽器" onclick="onBtnSendMsg()" />
</body>
</html>

如此,頁(yè)面向?yàn)g覽器發(fā)送消息的通道就實(shí)現(xiàn)完成了式矫。

瀏覽器向頁(yè)面發(fā)送消息

這個(gè)方向相對(duì)比較簡(jiǎn)單乡摹。QWebEnginePage提供了執(zhí)行JS的方法runJavaScript,因此只需要在頁(yè)面上提供一個(gè)接收消息的JS接口采转,就能實(shí)現(xiàn)瀏覽器向頁(yè)面的消息發(fā)送聪廉。

首先Qt端,上面的JsContext類已經(jīng)聲明了發(fā)送消息的方法sendMsg故慈,其具體實(shí)現(xiàn)如下:

void JsContext::sendMsg(QWebEnginePage* page, const QString& msg)
{
    page->runJavaScript(QString("recvMessage('%1');").arg(msg));
}

然后在msgutils.js里添加上對(duì)應(yīng)的js接口recvMessage:

// 接收qt發(fā)送的消息
function recvMessage(msg)
{
    alert("接收到Qt發(fā)送的消息:" + msg);
}

最后板熊,我們?cè)跒g覽器的主界面上加上一個(gè)輸入框(msgEdit)和一個(gè)按鈕(btnSend),并實(shí)現(xiàn)發(fā)送消息的槽:

connect(ui->btnSend, &QPushButton::clicked, this, [this]() {
    QString msg = ui->msgEdit->text();
    if (!msg.isEmpty())
    {
        m_jsContext->sendMsg(m_webView->page(), msg);
    }
});

這樣察绷,瀏覽器向頁(yè)面發(fā)送消息的接口就也實(shí)現(xiàn)完成了干签。

來(lái)試試效果,如下圖:


頁(yè)面向?yàn)g覽器發(fā)送消息.png

瀏覽器向頁(yè)面發(fā)送消息.png

有關(guān)舊版本的坑

如開篇簡(jiǎn)介中所說拆撼,這里的范例是基于Qt5.10版本實(shí)現(xiàn)的容劳。如果使用Qt5.7.1之前的版本,在使用WebChannel時(shí)闸度,會(huì)有一個(gè)嚴(yán)重的bug:在刷新頁(yè)面后竭贩,頁(yè)面會(huì)丟失window.qt對(duì)象[3]。所以如果要借助WebChannel實(shí)現(xiàn)Qt對(duì)象與頁(yè)面的交互莺禁,一定要將Qt升級(jí)到5.7.1之后的版本留量。

對(duì)于基于WebChannel的頁(yè)面與瀏覽器間的通信實(shí)現(xiàn)就先介紹到這里。下一節(jié)會(huì)就使用Qt WebEngine實(shí)現(xiàn)https雙向認(rèn)證進(jìn)行簡(jiǎn)要的分析。

>>返回系列索引

參考鏈接

[1] Qt5.10 QWebEngine幫助文檔
[2] 實(shí)現(xiàn)QT與HTML頁(yè)面通信
[3] 5.7.1版之前WebChannel的bug report

?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末楼熄,一起剝皮案震驚了整個(gè)濱河市忆绰,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌可岂,老刑警劉巖错敢,帶你破解...
    沈念sama閱讀 217,657評(píng)論 6 505
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異青柄,居然都是意外死亡伐债,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,889評(píng)論 3 394
  • 文/潘曉璐 我一進(jìn)店門致开,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)峰锁,“玉大人,你說我怎么就攤上這事双戳『缃” “怎么了?”我有些...
    開封第一講書人閱讀 164,057評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵飒货,是天一觀的道長(zhǎng)魄衅。 經(jīng)常有香客問我,道長(zhǎng)塘辅,這世上最難降的妖魔是什么晃虫? 我笑而不...
    開封第一講書人閱讀 58,509評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮扣墩,結(jié)果婚禮上哲银,老公的妹妹穿的比我還像新娘。我一直安慰自己呻惕,他們只是感情好荆责,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,562評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著亚脆,像睡著了一般做院。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上濒持,一...
    開封第一講書人閱讀 51,443評(píng)論 1 302
  • 那天键耕,我揣著相機(jī)與錄音,去河邊找鬼弥喉。 笑死郁竟,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的由境。 我是一名探鬼主播,決...
    沈念sama閱讀 40,251評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼虏杰!你這毒婦竟也來(lái)了讥蟆?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,129評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤纺阔,失蹤者是張志新(化名)和其女友劉穎瘸彤,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體笛钝,經(jīng)...
    沈念sama閱讀 45,561評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡质况,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,779評(píng)論 3 335
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了玻靡。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片结榄。...
    茶點(diǎn)故事閱讀 39,902評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖囤捻,靈堂內(nèi)的尸體忽然破棺而出臼朗,到底是詐尸還是另有隱情,我是刑警寧澤蝎土,帶...
    沈念sama閱讀 35,621評(píng)論 5 345
  • 正文 年R本政府宣布视哑,位于F島的核電站,受9級(jí)特大地震影響誊涯,放射性物質(zhì)發(fā)生泄漏挡毅。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,220評(píng)論 3 328
  • 文/蒙蒙 一暴构、第九天 我趴在偏房一處隱蔽的房頂上張望跪呈。 院中可真熱鬧,春花似錦丹壕、人聲如沸庆械。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,838評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)缭乘。三九已至,卻和暖如春琉用,著一層夾襖步出監(jiān)牢的瞬間堕绩,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,971評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工邑时, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留奴紧,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 48,025評(píng)論 2 370
  • 正文 我出身青樓晶丘,卻偏偏與公主長(zhǎng)得像黍氮,于是被迫代替她去往敵國(guó)和親唐含。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,843評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • Spring Cloud為開發(fā)人員提供了快速構(gòu)建分布式系統(tǒng)中一些常見模式的工具(例如配置管理沫浆,服務(wù)發(fā)現(xiàn)捷枯,斷路器,智...
    卡卡羅2017閱讀 134,656評(píng)論 18 139
  • 問答題47 /72 常見瀏覽器兼容性問題與解決方案专执? 參考答案 (1)瀏覽器兼容問題一:不同瀏覽器的標(biāo)簽?zāi)J(rèn)的外補(bǔ)...
    _Yfling閱讀 13,751評(píng)論 1 92
  • 從三月份找實(shí)習(xí)到現(xiàn)在淮捆,面了一些公司,掛了不少本股,但最終還是拿到小米攀痊、百度、阿里拄显、京東苟径、新浪、CVTE凿叠、樂視家的研發(fā)崗...
    時(shí)芥藍(lán)閱讀 42,246評(píng)論 11 349
  • Android 自定義View的各種姿勢(shì)1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,116評(píng)論 25 707
  • 中學(xué)難度 妻子告訴丈夫說:”昨晚我夢(mèng)見你給我買了一個(gè)鉆戒和一件貂皮大衣.”丈夫放下手中的報(bào)紙說:”好吧涩笤!今晚再睡著...
    奶爸Max閱讀 167評(píng)論 0 0