如何利用 QT 進行 web 與本地混合應用開發(fā)
如何利用 QT 進行 web 與本地混合應用開發(fā)是本文要講述的內(nèi)容,Qt提供了本地 C++對象與JavaScript的無縫集成,是進行本地與web混合應用開發(fā)的理想平臺。
一、Qt Webkit 集成
利用Qt的Webkit 集成與QtNetwork模塊,你完全可以進行本地桌面與web混合應用開發(fā),你可以自由地混合JavaScript,樣式表,Web內(nèi)容和Qt組件。 Webkit是一個非常成熟的web瀏覽引擎。Qt中集成了這個大名鼎鼎的引擎,通過QtWebkit,你可以在C++ 中執(zhí)行JavaScript,或者在網(wǎng)頁中集成C++對象,并且通過JavaScript和這些對象進行交互。
一個現(xiàn)代的HTML渲染引擎只 是混合開發(fā)的一半,另一半就是本地應用和渲染對象的交互。QT的Webkit 集成提供了這種解決方案:
1.使用object標簽嵌入Qt Widgets組件。這可以讓使用C++代碼的Qt組件包含在網(wǎng)頁中,作為網(wǎng)頁的部分外觀。
2.在JavaScript中訪問C++對象。你 可以在JavaScript環(huán)境中插入C++對象,讓網(wǎng)頁腳本直接訪問你的數(shù)據(jù)結(jié)構(gòu)。
3.在Qt中執(zhí)行JavaScript。你可以在C++ 調(diào)用網(wǎng)頁環(huán)境中的JavaScript函數(shù),觸發(fā)網(wǎng)頁事件。
4.共享客戶端存儲。在JavaScript和C++中你都具有訪問數(shù)據(jù)庫的能 力,這樣當下線時也能共享大量數(shù)據(jù)。
二、與嵌入的Qt對象交互
使用 QWebView 組件,有兩種方法可以在網(wǎng)頁中嵌入C++對象。你可以在網(wǎng)頁的JavaScript中添加C++對象,或者也可以創(chuàng)建一個插件,然后在網(wǎng)頁中使用 object標簽嵌入。
第二種方法更容易入手。當在網(wǎng)頁中放入 Widget組件時,它的所有public slots就像普通函數(shù)一樣被網(wǎng)頁中的JavaScript函數(shù)訪問。
要在網(wǎng)頁中添加一個Widget,首先要告訴你的QWebPage對象,該Widget可用,這個通過子類化QWebPlugFactory完成,你需要 重新實現(xiàn)兩個方法:plugs和create。plugs方法通知網(wǎng)頁該Widget可用,create方法根據(jù)請求創(chuàng)建widget。
在 HTML網(wǎng)頁中,widgets使用object標簽創(chuàng)建。比如,下面這個標簽試圖創(chuàng)建一個 application/x-qt-colorlabel 組件。
- <object type="application/x-qt-colorlabel" width="50px" height="20px" id="label" />
要利用這種創(chuàng)建,必須要允許使用插件并且要告訴QWebpage插件的工廠類。在下面的代碼中,ColorLabelFactory將 會根據(jù)application/x-qt-colorlabel的請求創(chuàng)建相應實例。
- QWebSettings::globalSettings()->setAttribute(QWebSettings::PluginsEnabled, true);
- webView->page()->setPluginFactory(new ColorLabelFactory(this));
ColorLabel有一個公開的slot: chagneColor(),這個對于網(wǎng)頁中的JavaScript自動可用。在網(wǎng)頁中插入一個指向該元素的鏈接,可以以一種簡單的方式激活C++函數(shù)。
- <a href='javascript:document.getElementById("label").changeColor();'>Change color!</a>
要反方向推進事件,必須要使你的對象在JavaScript文檔上下文中可用。要對QWebPage的每一個 QWebFrame,調(diào)用addToJavaScriptWindowObject方法。這個方法允許你根據(jù)名字把一個對象添加到JavaScipt上下 文中。
- webView->page()->mainFrame()->addToJavaScriptWindowObject( "eventSource", new eventSource( this ) );
要連接剛添加對象eventSource的信號,要加上一段JavaScript代碼,使用evaluateJavaScript方法 完成。下面的代碼將把eventSource對象的signalName信號連接到一個JavaScript函數(shù)destFunction。
- webView->page()->mainFrame()->evaluateJavaScript( "eventSource.signalName.connect(destFunction);" );
如果你把一個對象添加到一個以標準瀏覽器查看的JavaScript頁面中,有一個信號需要知道。每一次JavaScript 內(nèi)容被清除,F(xiàn)rame都會釋放 javaScriptWindowObjectCleared 信號。為了使你的Qt對象一直可用,你需要連接這個信號,并且在這里調(diào)用 addToJavaScriptWindowObject函數(shù)。
#p#
三、使用客戶端存儲共享數(shù)據(jù)
隨著HTML5,WEB標準日益靠近于桌面,同樣,桌面也開始集成WEB。這方面一個最大的變化就是客戶端存儲。這在客戶機上給了每一方(比如每一個頁 面)一個能利用SQL的數(shù)據(jù)庫引擎,可以緩存本地數(shù)據(jù),減少流量,使頁面能脫機使用。還可以用來存儲大量的結(jié)構(gòu)化的,可搜索的數(shù)據(jù)。
客戶端存儲可以在JavaScript中使用,從JavaScript代碼中搜索數(shù)據(jù)庫,然后從搜索結(jié)果中生成頁面。這要使用 openDatabase和transaction函數(shù)。
假設(shè)現(xiàn)在有個數(shù)據(jù)庫, 代碼如下所示:
- db = openDatabase("TestDb ", "1.0", "Client side storage test", 200000);
- db.transaction(function(tx) {
- tx.executeSql("SELECT id, text FROM Texts", [], function(tx, result) {
- for (var i = 0; i < result.rows.length; ++i) {
- var row = result.rows.item(i);
- processText( row['id'], row['text'] );
- }
- }, function(tx, error) {
- alert('Failed to retrieve texts from the database - ' + error.message);
- return;
- });
- });
使用QtWebkit,你可以使用QtSql模塊訪問同一個數(shù)據(jù)庫。這在混合開發(fā)中是一個非常有用的特征。比如,你的應用中的web頁面在和本地部分共享 數(shù)據(jù)時,可以用同樣的機制來保存數(shù)據(jù)。
為了避免安全問題,客戶端的數(shù)據(jù)庫只能由具有正確權(quán)限的一方在JavaScript中訪問。本地的C++代碼也可以通過靜態(tài)的QWebSecurityOrigin::allOrigins 方法,或者通過QWebFrame::securityOrigin 來訪問所有安全對象。
通過databases方法,可以訪問一個 QWebDatabaew方法列表,每一個web數(shù)據(jù)庫對象有一個filename屬性,可以用來在本地代碼中訪問數(shù)據(jù)庫。
- QWebDatabase webdb = mySecurityOrigin.databases()[index];
- QSqlDatabase sqldb = QSqlDatabase::addDatabase("QSQLITE", "webconnection");
- sqldb.setDatabaseName(webdb.fileName());
- if (sqldb.open()) {
- QStringList tables = sqldb.tables();
- }
在WEB和本地應用連接事件的能力,再加上共享數(shù)據(jù)的機制,更易于模糊WEB和桌面之間的界限。
四、WEB轉(zhuǎn)換
很多通過WEB得到的數(shù)據(jù)并不適合直接顯示。比如網(wǎng)絡(luò)新聞, 地理數(shù)據(jù),以及其它特定數(shù)據(jù)格式的應用。QT的網(wǎng)絡(luò)模塊可以以一種很簡單的方式下載這樣的數(shù)據(jù),然后解析數(shù)據(jù),把它轉(zhuǎn)換成能被自己的代碼處理的合適的格 式。也可以通過QtXmlPatterns模塊處理,當輸出格式是XML或者想在XHMTL網(wǎng)頁中顯示時,這樣更加方便。
我們通過一個小例子來大致了解一下這個有意思的部分。我們下載一個新聞源,用XSLT把它從XML轉(zhuǎn)換成XHTML,然后通過QWebPage把它顯示出 來。(圖略)
QNetworkAccessManager 類讓你很容易處理客戶端和WEB服務(wù)器之間的交互。它幫你處理諸如代理,保存設(shè)置,COOKIE以及SSL會話這些細節(jié)??傊?,它不但讓你在象上傳下載這 些常見情況更加容易,還可以幫你處理登錄,認證這些復雜會話。
為了下載我們例子中的新聞源,我們所有要做的就是創(chuàng)建一個QNetworkAccessManager 然后調(diào)用它的get方法。結(jié)果通過 finished(QNetworkReply*) 信號返回。
- {
- QNetworkAccessManager *manager = new QNetworkAccessManager( this );
- connect( manager, SIGNAL(finished(QNetworkReply*)), this, SLOT(handleReply(QNetworkReply*)) );
- connect( manager, SIGNAL(finished(QNetworkReply*)), m_progressBar, SLOT(hide()) );
- QNetworkReply *reply = manager->get( QNetworkRequest( QUrl( feedUrl ) ) );
- connect( reply, SIGNAL(downloadProgress(qint64, qint64)), this, SLOT(updateProgress(qint64,qint64)) );
- }
原文參見 http://www.qtsoftware.com/forms/whitepapers/reg-whitepaper-hybrid. (限于英文水平,只譯其大概)
小結(jié)::關(guān)于如何利用 QT 進行 web 與本地混合應用開發(fā)的內(nèi)容介紹完了,希望本文對你有所幫助!