IE編程轉載

2個比較好的ie編程文章:牛人啊

http://blog.csdn.net/CathyEagle/archive/2006/03/11/621961.aspx

http://blog.csdn.net/CathyEagle/archive/2006/04/25/677425.aspx

 

順便記錄下

Internet Explorer 編程簡述(十)響應來自HTML Element的事件通知——幾個好用的類 收藏
關鍵字:HTML Element, Sink

 
1、概述
實現了對Webbrowser的resuing之後我們便會發現有時候我們還需要處理瀏覽器中的元素(HTML Element)。這種處理包括主動和被動兩個方面,像《FAQ:如何訪問WebBrowser的滾動條》、《FAQ:操縱下拉列表》、《FAQ:兩種方法訪問多層嵌套的frame》等文章所演示的就是主動的處理。通常我們從Webbrowser獲得一個Web文檔接口(IHTMLDocumentx),從它出發便可訪問到瀏覽器所包含的一切HTML元素。而被動的處理則是在COM技術中稱爲Sink的技術,我更喜歡的說法是事件通知。當文檔的下載進度發生變化時,我們可以獲得ProgressChange通知,當Webbrowser下載完HTML文檔時,我們可以獲得DocumentComplete的通知,而當鏈接被點擊,或圖片被拖動時,我們如何獲得通知呢?本文希望能夠給出部分的答案。
 
2、HtmlObj Template
如何Sink一個HTML Element並不是本文的重點,其理論我不是太瞭解,也懶得去搞透徹,所以使用現成的庫來實現。CodeProject上的一篇文章《CHtmlObj Template》給出的一個模板類CHtmlObj就非常好用。下面的例子是針對Html Anchor Element的一個實例化。
 
#include "HtmlObj.h"
 
class CHtmlAnchorElement : public CHtmlObj<IHTMLAnchorElement,
&DIID_HTMLAnchorEvents>
{
public:
CHtmlAnchorElement(CHtmlDocument2* pParentDoc2);
virtual ~CHtmlAnchorElement();
 
virtual HRESULT OnInvoke(DISPID dispidMember,REFIID riid,
LCID lcid, WORD wFlags,
DISPPARAMS * pdispparams, VARIANT * pvarResult,
EXCEPINFO * pexcepinfo,
UINT * puArgErr);
};

HRESULT CHtmlAnchorElement::OnInvoke(DISPID dispidMember,
REFIID riid, LCID lcid, WORD wFlags,
DISPPARAMS * pdispparams, VARIANT * pvarResult,
EXCEPINFO * pexcepinfo,
UINT * puArgErr)
{
HRESULT hr = E_NOTIMPL;
switch(dispidMember)
{
case DISPID_HTMLELEMENTEVENTS_ONMOUSEOVER :
{//當鼠標經過鏈接時,我們在這裏獲得通知
hr = S_OK;
      // TODO: add code to handle on mouse over events
break;
}
case DISPID_HTMLELEMENTEVENTS_ONMOUSEOUT :
{//當鼠標從鏈接上移開時,我們在這裏獲得通知,其它的Dispatch ID可根據需要添加
hr = S_OK;
      // TODO: add code to handle on mouse out events
break;
}
default:
{
break;
}
}
 
return hr;
}
 
當我們得到某個鏈接的HTML接口指針,便可調用CHtmlAnchorElement繼承自CHtmlObj的SetSite(IUnknown *pUnkSite)成員函數傳入該接口指針。在CHtmlObj類內部用一個智能指針m_spHtmlObj來保存相應的HTML Element接口指針,所以當上面的ONMOUSEHOVER和ONMOUSEOUT兩個事件通知到達時,從m_spHtmlObj就可以訪問IHTMLAnchorElement的所有成員,如從href獲得鏈接的Url等,此處不再贅述。
 
3、CHtmlElements類
有了CHtmlObj之後我們又會發現實踐中常常會需要多個相同類型的CHtmlObj。比如包含Frame的網頁中每個Frame的HTML Document都需要一個CHtmlObj來Sink其事件。所以我們還需要有效地管理這些相同類型的CHtmlObj。下面是我寫的一個簡單的模板類CHtmlElements,它通過CMap來管理多個CHtmlObj對象。
 
template<class THtmlElement> class CHtmlElements
{
typedef CMap<LPDISPATCH, LPDISPATCH, THtmlElement*, THtmlElement*> CMapDispToHtmlElement;
CMapDispToHtmlElement m_htmlElements;
BOOL IsSiteConnected( LPDISPATCH pDisp )
{
THtmlElement *pElement;
return m_htmlElements.Lookup( pDisp, pElement );
}
public:
CHtmlElements(void)
{
}
~CHtmlElements(void)
{
}
public:
void SetSite( LPDISPATCH pDisp )
{
if ( IsSiteConnected( pDisp ) ) //檢查以避免多餘的Sink
{
return;
}
THtmlElement *pElement = new THtmlElement; //通過模板類型創建相應的類的實例進行連接
pElement->SetSite( pDisp );
m_htmlElements.SetAt( pDisp, pElement );
}
 
//在合適的地方調用Clear釋放所管理的內存

void Clear(void)

{

POSITION pos = m_htmlElements.GetStartPosition();

THtmlElement *pElement = NULL;

LPDISPATCH pDisp = NULL;

while (pos != NULL)

{

m_htmlElements.GetNextAssoc( pos, pDisp, pElement );

m_htmlElements.RemoveKey( pDisp );

delete pElement;

}

}

};
 
假設我們有一個象CHtmlAnchorElement那樣派生自CHtmlObj的類CHtmlDocument2,使用CHtmlElements時這樣聲明:
 
typedef CHtmlElements<CHtmlDocument2> CHtmlDocuments;
typedef CHtmlElements<CHtmlAnchorElement> CHtmlAnchors;
 
class CMyView : public CHtmlView
{
private:
CHtmlDocuments m_htmlDocs;
CHtmlAnchors m_htmlAnchors;
}
 
在DocumentComplete時就可以這樣連接到瀏覽器的文檔對象:

void CMyView ::OnDocumentComplete(LPDISPATCH pDisp, LPCTSTR lpszURL)
{
m_htmlDocs.SetSite(pDisp);
}
 
如果想一次性連接上文檔中所有的Anchor Element,可以通過IHTMLDocument2::get_anchors獲得包含所有IHTMLAnchorElement接口指針的IHTMLElementCollection,再遍歷其中的每個元素,分別調用m_htmlAnchors.SetSite即可。當然,一次性的Sink全部鏈接可能並不是個好注意,我更願意在CHtmlDocument2中響應事件再通過其它手段來訪問當前位置的HTML Element。
 
4、結論
響應HTML Element的事件通知對於瀏覽器編程來說是一個非常強大的手段,它可以更深入細化地控制瀏覽器中的文檔及其HTML元素,實現更爲高級的功能,比如所謂的“超級拖放”(許多多窗口瀏覽器都提供了該功能,但實際上沒有哪個瀏覽器完美地實現了對URL、文字及圖片的拖放)。
 
5、參考資料
Codeproject: HtmlObj Template
 
引用地址:《Internet Explorer 編程簡述(十)響應來自HTML Element的事件通知——幾個好用的類》


本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/CathyEagle/archive/2006/03/11/621961.aspx



=============================================

 

Internet Explorer 編程簡述(十一)實現完美的Inplace Drag & Drop——“超級拖放” 收藏
關鍵字:超級拖放,GetDropTarget,ondragover,IHTMLDataTransfer
 
1、概述
許多多窗口瀏覽器都提供了一種被稱爲“超級拖放”(或“超級拖拽”、“隨心拖放”等等,不一而足)的功能。作爲對IE拖拽行爲對擴展,“超級拖放”實現了一些非常實用的功能:
拖放網頁鏈接:通常是在新窗口中打開
拖放選中的文字:保存文字、作爲關鍵字通過搜索引擎搜索網絡、作爲Url打開等
拖放圖片:通常是保存圖片到指定文件夾
當然,還有很關鍵的一點:拖動對象時鼠標指針反饋不同的拖拽效果
在《Internet Explorer 編程簡述(十)響應來自HTML Element的事件通知——幾個好用的類》中曾提到,儘管許多瀏覽器都提供了超級拖放的功能,但與IE的缺省實現相比,除了具備鼠標指針拖拽效果外,還沒有哪個瀏覽器的實現能夠實現:
文字在頁面內與輸入框之間的交互拖放(這一點最爲重要)
來自外部的文字與網頁輸入框之間的交互拖放
拖拽時滾動頁面(這一點是被忽略了)
 
本文的目的,一是介紹實現超級拖放的兩種方法,二是說明如何實現“完美”的拖放——即擴展IE拖拽行爲的同時,保留IE默認的拖拽行爲。三是給出一個最爲直接和簡潔的實現,至於拖放不同的對象以實現不同的功能,不在本文討論的範圍,略去。
 
 
2、標準的實現方法
標準方法即通過IDocHostUIHandler的GetDropTarget成員函數來實現,在MSDN這樣說到:
IDocHostUIHandler::GetDropTarget Method——Called by MSHTML when it is used as a drop target. This method enables the host to supply an alternative IDropTarget interface.
即在適當的時候,MSHTML引擎會調用IDocHostUIHandler的GetDropTarget方法,爲應用程序提供一個機會來替換MSHTML缺省的DropTarget實現。我們就可以通過這個自定義的DropTarget實現來完成上述的“超級拖放”功能。方法示例如下,其中略去的部分可參考MFC中CHtmlControlSite和CHtmlView的源代碼:
 
STDMETHODIMP CHtmlControlSite::XDocHostUIHandler::GetDropTarget(
LPDROPTARGET pDropTarget, LPDROPTARGET* ppDropTarget)
{
METHOD_PROLOGUE_EX_(CHtmlControlSite, DocHostUIHandler)
*ppDropTarget = g_pDropTarget;//將自定義的實現告知MSHTML引擎
return S_OK;
}
 
其中g_pDropTarget指向某個全局的IDropTarget接口的實現,我們假定爲CIEDropTarget,CIEDropTarget實現了IDropTarget的幾個成員函數DragEnter、DragOver、DragLeave和Drop。在DragEnter中可以決定是否接受一個Drop以及如果接受這個Drop的話該提供怎樣的鼠標拖拽反饋,在持續觸發的DragOver中同樣可以設定鼠標拖拽反饋,從而實現在拖放不同的對象(文字、鏈接、圖像等)時提供不同的拖拽視覺效果,實現相當簡單,此處不再贅述。
但上面的實現存在一些問題。首先是選中的文字在頁面內與輸入框之間交互的拖放沒有了。這是自然的,既然我們用自定義的DropTarget替換掉了IE的缺省實現,那這種交互的拖放理應由我們自己實現。難處並非在於不能實現,而是在於實現起來比較麻煩——光是得到鼠標下的HTML Element就夠我們煩了;當輸入框中有文字的時候,光標還應該隨着鼠標的移動而移動——所以這個費力還不一定討好的功能似乎沒有哪個瀏覽器去做。其次,作爲輸入框文字拖放的衍生物,拖拽滾動沒有了。當鼠標向某個方向拖拽時,網頁應該隨着將不可見的部分滾動出來,比如某個輸入框,讓我們有機會將文字拖拽過去。這個Feature的實現並不困難,不過一來是被忽略了(注意到拖拽滾動的人並不多),二來主要Feature都沒有實現,這個滾動也意義不大了。
 
3、打入MSHTML內部
既然從GetDropTarget提供外部實現難以得到與輸入框的交互式拖放,那就換個角度來考慮問題,讓我們打入MSHTML的內部。
着手點是IHTMLDocumentX接口——操縱IE的DOM的法寶。我們注意到IHTMLDocument2有個ondragstart事件,進而想到應該也有諸如ondragenter、ondragover、ondrop之類的事件(事實上也是有的),如果響應這些事件,處理同輸入框的交互式拖放應該就能夠解決。因爲這些拖放在MSHTML的缺省DropTarget實現中發生,因而當鼠標拖拽到某個輸入框上時,肯定會觸發一個ondragover事件,而在IHTMLEventObj的輔助下我們能輕鬆得到相關的HTML Element,其它的操作就容易進行了。再細心一點,我們還發現IHTMLEventObj2接口有個dataTransfer屬性——可以得到一個IHTMLDataTransfer的指針,而IHTMLDataTransfer接口正是瀏覽器內部用於數據交換的重要手段之一(看看它的屬性就知道會很有用了):
IHTMLDataTransfer Members
clearData——Removes one or more data formats from the clipboard through dataTransfer or clipboardData object.
dropEffect——Sets or retrieves the type of drag-and-drop operation and the type of cursor to display.
effectAllowed——Sets or retrieves, on the source element, which data transfer operations are allowed for the object.
getData——Retrieves the data in the specified format from the clipboard through the dataTransfer or clipboardData objects.
setData——Assigns data in a specified format to the dataTransfer or clipboardData object.
 
更進一步,從IHTMLDataTransfer接口還可以訪問到IDataObject接口,在進行Ole拖放時,數據就是通過IDataObject接口來傳遞的。具體用法稍後討論。
 
4、打入MSHTML內部——思路
提供鼠標反饋效果與實現GetDropTarget的方法類似,有了IHTMLDataTransfer接口,便可在ondragstart及ondragover事件觸發時通過dropEffect屬性設置拖拽的效果(可根據需要自行設定,不設置的話使用默認的效果)。再者,“拖”和“放”都在MSHTML的缺省實現中發生,我們從IHTMLEventObj的SrcElement即可得知鼠標所位置的HTML Element是否是輸入框。
 
5、打入MSHTML內部——實現
要接收到ondragstart之類的事件,可以採用《Internet Explorer 編程簡述(十)響應來自HTML Element的事件通知——幾個好用的類》中提到的CHtmlObj類和CHtmlElements類,並在適當的地方連接到Document,示例代碼如下所示:
 
HRESULT CHtmlDocument2::OnInvoke(DISPID dispidMember, REFIID riid, LCID lcid, WORD wFlags,
DISPPARAMS * pdispparams, VARIANT * pvarResult,EXCEPINFO * pexcepinfo,
UINT * puArgErr)
{
......
//如果只是要設置鼠標拖拽效果的話,這個事件可以不處理
case DISPID_HTMLELEMENTEVENTS_ONDRAGSTART :
{
OnDragStart();
break;
}
//重點在這裏
case DISPID_HTMLELEMENTEVENTS_ONDRAGOVER :
{
OnDragOver();
break;
}
case DISPID_HTMLELEMENTEVENTS_ONDROP :
{
OnDrop();
break;
}
......
}
 
void CHtmlDocument2::OnDragOver(void)
{
SetDragEffect();               //設置鼠標拖拽效果
}
 
void CHtmlDocument2::SetDragEffect(void)
{
CComQIPtr<IHTMLWindow2>  pWindow;
CComQIPtr<IHTMLEventObj>  pEventObj;
CComQIPtr<IHTMLEventObj2>  pEventObj2;
CComQIPtr<IHTMLElement>  pElement;
 
HRESULT hr = m_spHtmlObj->get_parentWindow( &pWindow );
hr = pWindow->get_event( &pEventObj );
 
//ondragover發生時IE的默認行爲是“沒有鼠標拖拽效果”。
//將IHTMLEventObj的返回值設爲false即可取消該事件的默認行爲,所以執行完下面這句話,拖拽效果就出現了。
AllowDisplayDragCursor(pEventObj, FALSE);  
 
CComBSTR bstrTagName;
pEventObj->get_srcElement(&pElement);    //獲得當前HTML Element
pElement->get_tagName(&bstrTagName);    
if ( IsEditArea(bstrTagName) ) //根據Tag Name判斷是否鼠標位於輸入框,以便設置焦點使得光標隨鼠標移動
{
CComQIPtr<IHTMLElement2>  pElement2;
if ( SUCCEEDED(pElement->QueryInterface(IID_IHTMLElement2, (void **) &pElement2 ))
&& pElement2 )
{
pElement2->focus();
}
//默認情況下,當拖拽文檔到輸入框時,鼠標會變成拖拽的光標,所以這裏使用IE的默認行爲。
AllowDisplayDragCursor(pEventObj, TRUE);
}
}
 
BOOL CHtmlDocument2::IsEditArea(CComBSTR bstrTagName)
{
return bstrTagName == "INPUT" || bstrTagName == "TEXTAREA";
}
 
void CHtmlDocument2::AllowDisplayDragCursor(CComQIPtr<IHTMLEventObj> pEventObj, BOOL bAllow)
{
VARIANT v;
v.vt = VT_BOOL;
 
v.boolVal = !bAllow ? VARIANT_FALSE : VARIANT_TRUE;
pEventObj->put_returnValue(v);
}
 
void CHtmlDocument2::OnDrop(void)
{
CComQIPtr<IHTMLWindow2>  pWindow;
CComQIPtr<IHTMLEventObj>  pEventObj;
CComQIPtr<IHTMLEventObj2>  pEventObj2;
CComQIPtr<IHTMLElement>  pElement;
CComQIPtr<IHTMLDataTransfer>   pdt; //此處演示如何使用IHTMLDataTransfer
 
HRESULT hr = m_spHtmlObj->get_parentWindow( &pWindow );
hr = pWindow->get_event( &pEventObj );
hr = pEventObj->QueryInterface(IID_IHTMLEventObj2, (void **) &pEventObj2 );
hr = pEventObj2->get_dataTransfer(&pdt);
 
CComBSTR bstrFormat = "URL"; //首先嚐試獲取URL
VARIANT Data;
hr = pdt->getData(bstrFormat, &Data);
if ( Data.vt != VT_NULL )
{     //獲取成功,拖放的對象是Url
DoOpenUrl(CString(Data.bstrVal));
}
else
{     //否則嘗試獲取選中的文本
bstrFormat = "Text";
hr = pdt->getData(bstrFormat, &Data);
if ( Data.vt != VT_NULL )
{     //獲取成功,拖放的內容是文本
CComBSTR bstrTagName;
pEventObj->get_srcElement(&pElement);
pElement->get_tagName(&bstrTagName);
if ( IsEditArea(bstrTagName) )
{
//Drop target是輸入框,不做任何操作,由IE進行默認處理
return;
}
else
{     //否則我們自己處理文本,或保存,或檢測是否鏈接後打開,等等
DoProcessText(CString(Data.bstrVal));
//Process the text
}
}
else
{     //既不是鏈接,也不是文本,可認爲是來自外部(如Windows Shell)的文件拖放
DoOnDropFiles(pdt);
}
}
}
 
//演示如何從IHTMLDataTransfer得到IDataObject
void CHtmlDocument2::DoOnDropFiles(CComQIPtr<IHTMLDataTransfer> pDataTransfer)

{
CComQIPtr<IServiceProvider>  psp;
CComQIPtr<IDataObject>  pdo;
if ( FAILED(pDataTransfer->QueryInterface(IID_IServiceProvider, (void **) &psp)) )
{
return;
}
if ( FAILED(psp->QueryService(IID_IDataObject, IID_IDataObject, (void **) &pdo)) )
{
return;
}
 
COleDataObject DataObject;
DataObject.Attach(pdo);
......
}
 
6、再次回到標準方法
上述通過Event Sink響應網頁拖拽的方法已經能夠很好地工作,可說“趨於完美”了,但仍有兩個“小”問題:第一,必須與document建立連接才能工作,而建立連接的時機不容易掌握(MSDN中推薦的位置是DocumentComplete,但在NavigateComplete中也可,或者是檢測到WebBrowser的readystate變爲READYSTATE_INTERACTIVE時進行連接)。第二,實現方法還是略顯複雜。
有沒有更簡單的方法呢?我決定再次對GetDropTarget進行“調研”。所謂“踏破鐵鞋無覓處,得來全不費功夫”,晃了一眼GetDropTarget方法的聲明後,靈機一動,我忽然想到了辦法。事實證明,這是完美的解決辦法。
 
讓我們再來看看GetDropTarget的聲明,其中第一個參數指向MSHTML提供的缺省DropTarget實現,而第二個參數用以返回應用程序的自定義DropTarget實現,如果在GetDropTarget中返回S_OK,MSHTML將以應用程序提供的自定義DropTarget替換缺省的DropTarget實現。
HRESULT GetDropTarget( IDropTarget *pDropTarget, IDropTarget **ppDropTarget);

參數說明

pDropTarget

[in] Pointer to an IDropTarget interface for the current drop target object supplied by MSHTML.

ppDropTarget

[out] Address of a pointer variable that receives an IDropTarget interface pointer for the alternative drop target object supplied by the host.

想到了嗎?解決問題的關鍵就在於第一個參數pDropTarget。相信很多瀏覽器在處理的時候都忽略掉了第一個參數而只是將自己的實現通過第二個參數告知MSHTML,因而丟失了IE缺省的行爲。既然如此,將缺省的IDropTarget接口的指針保存下來,在適當的時候調用,不就能夠保留IE的原始拖放行爲了嗎?

 

 

 
7、完美實現
完整的代碼就不再給出,我們只列出關鍵的部分作爲示例。假設我們用來實現IDropTarget接口的類叫做CBrowserDropTarget:
//構造函數,傳入參數即是從GetDropTarget得到的那個pDropTarget,它是MSHTML的缺省實現
CBrowserDropTarget::CBrowserDropTarget(IDropTarget *pOrginalDropTarget)
:  m_bDragTextToInputBox(FALSE)
//這個布爾變量用來判斷是否正在向InputBox拖拽文字
,  m_pOrginalDropTarget(pOrginalDropTarget)
//m_pOrginalDropTarget用來保存MSHTML的缺省實現
{
}
 
STDMETHODIMP CBrowserDropTarget::DragEnter(/* [unique][in] */IDataObject __RPC_FAR *pDataObj,
/* [in] */ DWORD grfKeyState,
/* [in] */ POINTL pt,
/* [out][in] */ DWORD __RPC_FAR *pdwEffect)
{
//調用缺省的行爲
return m_pOrginalDropTarget->DragEnter(pDataObj, grfKeyState, pt, pdwEffect);
}
 
STDMETHODIMP CBrowserDropTarget::DragOver(/* [in] */ DWORD grfKeyState,
/* [in] */ POINTL pt,
/* [out][in] */ DWORD __RPC_FAR *pdwEffect)
{
//在網頁內拖拽文字時這個值是DROPEFFECT_COPY(拖拽的文字不屬於輸入框中)
//或DROPEFFECT_COPY | DROPEFFECT_MOVE(拖拽的文字是輸入框中的文字)
DWORD dwTempEffect = *pdwEffect;
 
//接下來調用IE的缺省行爲
HRESULT hr = m_pOrginalDropTarget->DragOver(grfKeyState, pt, pdwEffect);
 
//判斷是否是往輸入框拖拽文字
m_bDragTextToInputBox = IsDragTextToInputBox(dwOldEffect, *pdwEffect);
if ( !m_bDragTextToInputBox )
{
//不是往輸入框拖拽文字,則使用原始的拖拽效果。否則和IE的缺省效果一樣——也就是沒有效果
*pdwEffect = dwTempEffect;
}
return S_OK;
}
 
//根據調用缺省行爲前後的Effect值判斷是否是往輸入框拖拽文字
BOOL CBrowserDropTarget::IsDragTextToInputBox(DWORD dwOldEffect, DWORD dwNewEffect)
{
//如果是把非輸入框中文字往輸入框拖動,則dwOldEffect與dwNewEffect相等,都是DROPEFFECT_COPY
BOOL bTextSelectionToInputBox = ( dwOldEffect == DROPEFFECT_COPY )
&& ( dwOldEffect == dwNewEffect );
 
//如果是把文字從一個輸入框拖到另一個輸入框,則dwOldEffect爲DROPEFFECT_COPY | DROPEFFECT_MOVE,
//而dwNewEffect的值可能爲DROPEFFECT_MOVE(默認情況),也可能爲DROPEFFECT_COPY(按下Ctrl鍵時)
BOOL bInputBoxToInputBox = ( dwOldEffect == (DROPEFFECT_COPY | DROPEFFECT_MOVE) )
&& ( dwNewEffect == DROPEFFECT_MOVE || dwNewEffect == DROPEFFECT_COPY );
 
//來自Microsoft Word的拖拽特殊一些,dwOldEffect是所有效果的組合值
BOOL bMSWordToInputBox =
( dwOldEffect == (DROPEFFECT_COPY | DROPEFFECT_MOVE | DROPEFFECT_LINK) )
&& ( dwNewEffect == DROPEFFECT_MOVE || dwNewEffect == DROPEFFECT_COPY );
 
//來自Edit Plus的拖拽過也特殊一些,dwOldEffect是個負數(懷疑是Edit Plus的拖拽實現有問題)
BOOL bEditPlusToInputBox = ( dwOldEffect < 0 )
&& ( dwNewEffect == DROPEFFECT_MOVE || dwNewEffect == DROPEFFECT_COPY );
 
//也許還有些例外,可再添加
......
return bTextSelectionToInputBox || bInputBoxToInputBox || bMSWordToInputBox || bEditPlusToInputBox;
}
 
STDMETHODIMP CBrowserDropTarget::DragLeave()
{
//調用缺省的行爲
return m_pOrginalDropTarget->DragLeave();
}
 
STDMETHODIMP CBrowserDropTarget::Drop(/* [unique][in] */ IDataObject __RPC_FAR *pDataObj,
/* [in] */ DWORD grfKeyState,
/* [in] */ POINTL pt,
/* [out][in] */ DWORD __RPC_FAR *pdwEffect)
{
if ( m_bDragTextToInputBox )
{
//是文字拖放,調用IE的缺省行爲
return m_pOrginalDropTarget->Drop(pDataObj, grfKeyState, pt, pdwEffect);
}
 
//否則是拖放鏈接、圖片、文件等,按常規的IDataObject處理方式
......
return S_OK;
}
 
至此,我們就得到了一個完美的“超級拖放”的基本框架,它在擴展的同時保留了IE的默認行爲:
文字在頁面內與輸入框之間能夠交互拖放。
來自外部的文字與網頁輸入框之間也能交互拖放
拖拽時能夠自動滾動頁面
 
其餘的功能,如向不同的方向拖拽以完成不同的工作,左鍵右鍵拖放執行不同的功能,按住Alt保存文字等等,可根據需要自行實現,不再討論。
8、修正
今天和Stanley Xu聊了幾個鐘頭,受益匪淺。根據Stanley的提議,毋須再作是否往輸入框拖拽文字的判斷,因爲我們需要的只是在IE的缺省行爲沒有鼠標拖拽效果的時候讓它有拖拽效果,因此只需要簡單地判斷調用IE缺省行爲後的Effect值是否爲0即可,如下:
//判斷是否是往輸入框拖拽文字
m_bDragTextToInputBox = *pdwEffect != 0;
簡單而直接,當然更重要的是:可用。
 
9、參考資料
MSDN: IHTMLEventObj Interface
MSDN: IHTMLDataTransfer Interface
《Internet Explorer 編程簡述(十)響應來自HTML Element的事件通知——幾個好用的類》


本文來自CSDN博客,轉載請標明出處:http://blog.csdn.net/CathyEagle/archive/2006/04/25/677425.aspx

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章