關(guān)鍵字:itravellogstg, ienumtravellogentry, itravellogentry
1、概述
internet explorer的瀏覽歷史菜單在4.0版本開始出現(xiàn),但直到5.5之前,微軟都未公布用于訪問瀏覽歷史的com接口,如今已是ie6.0大行其道的年代,用于訪問瀏覽歷史的接口也早已公布多時(shí),本文的目的則是試圖拋磚引玉,簡單介紹用于訪問瀏覽歷史的travel log接口,并用一個(gè)小小的類cietravellog來實(shí)現(xiàn)對(duì)travel log的封裝。
2、iomhistory接口
在早些時(shí)候的msdn中,我們能夠查閱到關(guān)于瀏覽歷史的接口僅有iomhistory,而該接口實(shí)際上對(duì)應(yīng)的是瀏覽器中可以通過腳本訪問的“history”對(duì)象。對(duì)于“history”對(duì)象,msdn中是這樣說的:
for security reasons, the history object does not expose the actual urls in the browser history. it does allow navigation through the browser history by exposing the back, forward, and go methods. a particular document in the browser history can be identified as an index relative to the current page. for example, specifying -1 as a parameter for the go method is the equivalent of clicking the back button.
this object is available in script as of microsoft internet explorer 3.0.
即為了安全的原因,iomhistory接口僅提供了有限的幾個(gè)方法來完成在瀏覽器中前進(jìn)、后退等操作,并沒有提供訪問歷史列表url的能力。這也難怪,該接口在ie 3.0時(shí)代已經(jīng)存在,而當(dāng)時(shí)ie并不成熟,可編程能力也不甚強(qiáng)大。一直到ie 4.0通過與windows 98捆綁銷售一統(tǒng)天下之后,相關(guān)的文檔才逐漸豐富,多窗口瀏覽器等基于internet explorer/webbrowser control的應(yīng)用軟件也才鋪天蓋地開來。但在ie 5.5接口公布之前,要完全模擬ie的travel log行為,并不是一件容易的事。最容易想到的方法就是在beforenavigate、documentcomplete等事件發(fā)生之時(shí)記錄/修改url并加以保存(我在早些時(shí)候也這樣做過),但是效果不甚理想,尤其是瀏覽包含frame的網(wǎng)頁時(shí),處理更是麻煩。當(dāng)然,要完全模擬亦非難事,只不過開發(fā)人員都知道微軟公布接口是早晚的事,所以也沒有人花大力氣在模擬ie的travel log行為上。
3、travel log簡介
internet explorer 5.5推出以后,travel log接口也就開始出現(xiàn)在msdn中,它是專門為ole嵌入webbrowser control的應(yīng)用程序設(shè)計(jì)的,其目的是“提高和加強(qiáng)用戶的訪問日志體驗(yàn)”(improve and enhance the user's travel log experience)。事實(shí)上,稍后我會(huì)提到,travel log接口正日益成為應(yīng)用程序中的重要接口之一。
微軟公布的travel log共包含三個(gè)接口:itravellogstg, ienumtravellogentry和itravellogentry。
itravellogstg——該接口提供了用于在travel log中添加、刪除、枚舉日志(瀏覽歷史)的方法,本文需要用到的幾個(gè)方法列舉如下:
方法名 用途
enumentries 為訪問日志項(xiàng)創(chuàng)建枚舉器(ienumtravellogentry接口指針)
getrelativeentry 返回一個(gè)日志項(xiàng)
travelto 訪問一個(gè)日志項(xiàng)
ienumtravellogentry——該接口提供用于枚舉日志項(xiàng)所必需的方法,本文只用到一個(gè)方法:
方法名 用途
next 枚舉下一個(gè)日志項(xiàng)(返回itravellogentry接口指針)
itravellogentry——該接口只有兩個(gè)方法,分別用于返回日志項(xiàng)的title和url:
方法名 用途
gettitle 返回日志項(xiàng)的title
geturl 返回日志項(xiàng)的url
接口準(zhǔn)備好了,我們也就很容易得知它們之間的關(guān)系:
要得到相對(duì)于當(dāng)前頁面的日志項(xiàng)列表,首先應(yīng)通過itravellogstg接口創(chuàng)建一個(gè)枚舉器(ienumtravellogentry接口)。
通過ienumtravellogentry枚舉器的next方法枚舉出一個(gè)個(gè)的日志項(xiàng)(itravellogentry接口)。
由itravellogentry接口獲取日志項(xiàng)所代表的網(wǎng)頁的title和url并加以處理。
訪問相對(duì)于當(dāng)前頁面的某個(gè)日志項(xiàng)時(shí),首先由itravellogstg的getrelativeentry方法根據(jù)與當(dāng)前頁的距離得到itravellogentry接口,再將后者傳入itravellogstg的travelto方法以達(dá)到訪問日志項(xiàng)的目的(如前進(jìn)和后退)。
也許不是太恰當(dāng),我對(duì)uml也不熟悉,借用一個(gè)偽uml序列圖表示其關(guān)系如下:
4、封裝travel log
接下來,我們就用一個(gè)簡單的類來完成對(duì)travel log的封裝。如下所示,tlogstg.h包含了travel log的相關(guān)接口聲明,該頭文件可以在platform sdk中找到。
#include "tlogstg.h"
class cietravellog
{
private:
itravellogstg *m_ptravellogstg;
ienumtravellogentry *m_penumlogentry;
itravellogentry *m_ptravallogentry;
iwebbrowser2* m_pwebbrowser;
public:
cietravellog(void);
~cietravellog(void);
void setwebbrowser(iwebbrowser2* pwebbrowser);
void buildhistorymenu(cmenu * pmenu, unsigned char ncount, bool bforward);
void travelto(int nposition);
};
cietravellog::cietravellog(void)
: m_ptravellogstg(null), m_penumlogentry(null), m_ptravallogentry(null), m_pwebbrowser(null)
{
}
cietravellog::~cietravellog(void)
{
if ( m_ptravellogstg != null )
{
m_ptravellogstg->release();
}
if ( m_penumlogentry != null )
{
m_penumlogentry->release();
}
if ( m_ptravallogentry != null )
{
m_ptravallogentry->release();
}
if ( m_pwebbrowser != null )
{
m_pwebbrowser->release();
}
}
//將瀏覽器的iwebbrowser2接口指針賦予cietravellog的實(shí)例
void cietravellog::setwebbrowser(iwebbrowser2* pwebbrowser)
{
if ( (m_pwebbrowser == pwebbrowser) || (m_pwebbrowser == null) )
{
return;
}
if ( m_pwebbrowser != null )
{
m_pwebbrowser->release();
}
m_pwebbrowser = pwebbrowser;
iserviceprovider *psp;
hresult hr = m_pwebbrowser->queryinterface(iid_iserviceprovider, (lpvoid*)&psp);
m_pwebbrowser->release();
if (succeeded(hr))
{
hr = psp->queryservice(sid_stravellogcursor, iid_itravellogstg, (lpvoid*)&m_ptravellogstg);
psp->release();
}
}
//創(chuàng)建瀏覽歷史菜單,bforward指明是前進(jìn)還是后退菜單
void cietravellog::buildhistorymenu(cmenu * pmenu, unsigned char ncount, bool bforward)
{
if ( m_ptravellogstg == null )
{
return;
}
tlenumf eflag = bforward ? tlef_relative_fore : tlef_relative_back;
if ( failed(m_ptravellogstg->enumentries( eflag, &m_penumlogentry ) ) )
{
return;
}
ulong ufetched;
int i=0;
if ( m_penumlogentry !=null )
{
while ( succeeded( m_penumlogentry->next( 1, &m_ptravallogentry, &ufetched ) ) &&
m_ptravallogentry && i<10 )//我們最多只需要10條歷史菜單,可根據(jù)實(shí)際情況修改
{
lpolestr psztitle;
m_ptravallogentry->gettitle( &psztitle );
cstring strtitle = psztitle;
if ( bforward )
{
//id_iehistory_middle是預(yù)定義的某個(gè)菜單項(xiàng)id,從該id開始前后可以創(chuàng)建10個(gè)菜單項(xiàng),參見下一節(jié)
pmenu->insertmenu( 0, mf_string, id_iehistory_middle + ++i, strtitle );
}
else
{
pmenu->insertmenu( 0, mf_string, id_iehistory_middle - ++i, strtitle );
}
cotaskmemfree( psztitle );
m_ptravallogentry->release();
}
}
}
//根據(jù)與當(dāng)前頁面的相對(duì)距離來訪問歷史網(wǎng)頁
void cietravellog::travelto(int nposition)
{
if ( m_ptravellogstg == null )
{
return;
}
if succeeded(m_ptravellogstg->getrelativeentry( nposition, &m_ptravallogentry ))
{
m_ptravellogstg->travelto( m_ptravallogentry );
}
}
5、使用cietravellog
假設(shè)是在我們自己編寫的多窗口瀏覽器中使用travel log。為簡單起見,我們聲明一個(gè)cietravellog的全局對(duì)象g_ietravellog,以便在任何地方調(diào)用。然后在適當(dāng)?shù)牡胤剑鏲mainframe的tbn_dropdown消息(工具條菜單下拉消息)處理函數(shù)ondropdown中,添加下面的代碼,用以創(chuàng)建瀏覽歷史菜單:
//getactivewebbrowserptr返回活動(dòng)的瀏覽器iwebbrowser2接口指針
ietravellog.setwebbrowser( getactivewebbrowserptr );
//bforward為true則創(chuàng)建“前進(jìn)”菜單,否則創(chuàng)建“后退”菜單
ietravellog.buildhistorymenu( &menu, 10, bforward);
以下定義為菜單項(xiàng)id的范圍,前后共可以容納10個(gè)菜單項(xiàng),可根據(jù)實(shí)際情況修改。
#define id_iehistory_first 60200
#define id_iehistory_middle 60210
#define id_iehistory_last 60220
添加命令處理函數(shù)ontravelhistoryurl用以響應(yīng)從id_iehistory_first到id_iehistory_last的菜單命令。
on_command_range(id_iehistory_first, id_iehistory_last, ontravelhistoryurl)
void cmainframe::ontravelhistoryurl(uint nid /* command id */)
{
//nid - id_iehistory_middle即為要訪問的瀏覽歷史到當(dāng)前頁面的距離
g_ietravellog.travelto( nid - id_iehistory_middle );
}
6、再談travel log
前面我提到“travel log接口正日益成為應(yīng)用程序中的重要接口之一”,此處加以說明。從微軟平臺(tái)的開發(fā)模式及導(dǎo)向來看,基于internet explorer/webbrowser control的應(yīng)用勢(shì)必會(huì)成為主流。在下一代的操作系統(tǒng)longhorn中,應(yīng)用程序界面的描述將完全由xml的一個(gè)特化——xaml來完成,而xaml的解析將由瀏覽器完成。微軟說未來應(yīng)用程序的部署將會(huì)十分容易,本地應(yīng)用和基于瀏覽器的應(yīng)用之間的差異將會(huì)被逐漸淡化,而實(shí)現(xiàn)這一目標(biāo)的一個(gè)重要表現(xiàn)就是,在將來的操作系統(tǒng)平臺(tái)上,應(yīng)用程序?qū)嶋H上時(shí)刻都將運(yùn)行在internet explorer中,internet explorer在某種程度上來說變成了一個(gè)容器。
于是,扎根于internet explorer的travel log自然而然地就被整合到了我們的應(yīng)用程序中。君不見,我們每天在資源管理器和瀏覽器上完成的工作,不就是在travel log中來來回回地跑嗎?如果所有的應(yīng)用程序都嵌入到internet explorer中運(yùn)行,那么我們?cè)趹?yīng)用程序中所作的操作便自然得到了記錄,“前進(jìn)”和“后退”也就很easy了。
很多軟件都已經(jīng)或多或少地開始采用基于internet explorer的模式,如microsoft money、microsoft encarta、visual studio.net的安裝程序等等,都是很好的范例。所以,就目前來說,將我們的應(yīng)用程序按這種模式編寫(可參考《利用瀏覽器實(shí)現(xiàn)程序界面與實(shí)現(xiàn)的分離》),不是可以早一點(diǎn)獲得“訪問日志的體驗(yàn)”嗎?何樂而不為。
新聞熱點(diǎn)
疑難解答
圖片精選