在VC++中绑定网页元素的事件(如onclick),通常会使用HTMLElementEvents2事件接口进行。大致如下:

void   CSomeObject:ConnectElement(IHTMLElement*   pElement)   

{

//取得连接点容器

CComQIPtr<IConnectionPointContainer> pCPC=pElement; 

//查找连接点

CComPtr<IConnectionPoint> pCP;

pCPC->FindConnectionPoint( DIID_HTMLElementEvents2, &pCP);

//连接事件接口

IUnknownPtr pUnk=this; //由CSomeObject实现HTMLElementEvents2事件接口

DWORD dwCookie;

pCP->Advise( pUnk, &dwCookie);

}

 

而在javascript中使用attachEvent绑定网页元素事件的方式却显得更自然,实现也更直接,我们是否也可以用VC++实现呢?

  1. function fn_onclick()
  2. {
  3.    alert("Hello world!");
  4. }
  5. document.all("button1").attachEvent("onclick",fn_onclick);
  6. document.all("button1").detachEvent("onclick",fn_onclick);

先看看在PIMShell中用VC++是怎样实现以上功能的。

  1. class ATL_NO_VTABLE CVCEventSinkDemo:
  2.  public IDispatchImpl<IVCEventSinkDemo>
  3. {
  4. public:
  5.  CVCEventSinkDemo()
  6.  {
  7.  }
  8.  DECLARE_PROTECT_FINAL_CONSTRUCT()
  9.  HRESULT FinalConstruct()
  10.  {
  11.   return S_OK;
  12.  }
  13.  void FinalRelease()
  14.  {
  15.  }
  16. private:
  17.    //代理对象
  18.    PIMShellCore::IAjaxDelegatePtr m_pDelegate_onevents;
  19.    //回调函数
  20.    static HRESULT CALLBACK Sink_onevents( VARIANT vEvent, VARIANT vContext, IUnknown* pInstance, VARIANT* pvarResult);
  21.    //绑定事件或取消绑定
  22.    void __attachEvents(bool bAttach);
  23. };
  24. void CVCEventSinkDemo::__attachEvents(bool bAttach)
  25. {
  26.    if(bAttach)
  27.    {
  28.       //生成一个代理对象,传入this指针和回调函数
  29.       IUnknownPtr pThis=this;
  30.       m_pDelegate_onevents = o->Sys2->CreateDelegateVC1( pThis, (LONGLONG)&Sink_onevents);
  31.       //绑定事件
  32.        o->Control->AttachEvent(L"onclick", m_pDelegate_onevents);
  33.    }
  34.    else
  35.    {
  36.       if(m_pDelegate_onevents!=NULL)
  37.       {
  38.          //取消绑定
  39.           o->Control->DetachEvent( L"onclick", m_pDelegate_onevents);
  40.          //
  41.          m_pDelegate_onevents=NULL;
  42.       }
  43.    }
  44. }
  45. //当事件发生时,调用此函数
  46. HRESULT CVCEventSinkDemo::Sink_onevents( VARIANT vEvent, VARIANT vContext, IUnknown* pInstance, VARIANT* pvarResult)
  47. {
  48. __SAFECALL_BEGIN;
  49.  //取得this指针
  50.  CVCEventSinkDemo* pThis=dynamic_cast<CVCEventSinkDemo*>(pInstance);
  51.  //取得event对象
  52.  MSHTML::IHTMLEventObjPtr e=vEvent.pdispVal;
  53.  //event.type
  54.  CString sType=e->type;
  55.  //
  56.  if(sType==L"click")
  57.  {
  58.      //执行响应代码
  59.  }
  60. __SAFECALL_END;
  61. }

1、先生成一个代理对象,这个对象记录CVCEventSinkDemo的实例指针和回调函数,与C#中delegate的设计理念一致。

2、将代理对象传入attachEvent实现对onclick事件的绑定。

3、当事件发生时,系统调用代理对象的0方法,

4、在代理对象的0方法中,调用先前记录的回调函数,并将CVCEventSinkDemo的实例指针传入。

 

 

我们接下来看看CreateDelegateVC1是如何生成代理对象的。

  1. HRESULT CreateDelegateVC1( IUnknown* pInstance, LONGLONG lnCallback, IDispatch** ppDelegate)
  2. {
  3.    //生成代理对象
  4.    CComObject<CVCDelegateDemo>* pDelegate;
  5.    HRESULT hr=CComObject<CVCDelegateDemo>::CreateInstance(&pDelegate);
  6.    if(FAILED(hr))
  7.       return hr;
  8.    //记录实例指针和回调函数
  9.    pDelegate->__record(pInstance,lnCallback);
  10.    //ok
  11.    return pDelegate->QueryInterface(IID_IDispatch,(void**)ppDelegate);
  12. }

接下来,我们看CVCDelegateDemo的0方法是如何实现的

  1. class ATL_NO_VTABLE CVCDelegateDemo:
  2.  public IDispatchImpl<CVCDelegateDemo>
  3. {
  4. public:
  5.  CVCDelegateDemo()
  6.  {
  7.  }
  8.  DECLARE_PROTECT_FINAL_CONSTRUCT()
  9.  HRESULT FinalConstruct()
  10.  {
  11.   return S_OK;
  12.  }
  13.  void FinalRelease()
  14.  {
  15.  }
  16. //声明回调函数的格式
  17. typedef HRESULT (CALLBACK* _delegateVC1)( VARIANT vParam1, VARIANT vContext, IUnknown* pInstance, VARIANT* pvarResult);
  18. private:
  19.     IUnknownPtr       m_pInstance;
  20.     _delegateVC1      m_pCallback1;
  21. public:
  22.     
  23.     //record
  24.     void __record(IUnknown* pInstance,LONGLONG lnCallback)
  25.     {
  26.         //记录实例指针和回调函数
  27.          this->m_pInstance=pInstance;
  28.         this->m_pCallback1=(_delegateVC1)lnCallback;
  29.     }
  30. public:
  31.     //实现0方法,并调用回调函数
  32.     STDMETHOD(Invoke)( DISPID dispidMember, REFIID riid, LCID lcid, WORD wFlags,
  33.                        DISPPARAMS* pDispParams, VARIANT* pvarResult,
  34.                        EXCEPINFO*  pExcepInfo,  UINT* puArgErr)
  35.     {
  36.        //只支持0方法
  37.         if(dispidMember!=0)
  38.           return E_INVALIDARG;
  39.        //如果是通过attachEvent绑定网页元素的事件,那么传入的第一个参数就是event对象
  40.         _variant_t vEvent=pDispParams->rgvarg[0];
  41.       //调用回调函数,传入event对象和实例指针
  42.        (*m_pCallback1)(vEvent,vtMissing,m_pInstance,pvarResult);
  43.       //ok
  44.       return  S_OK; 
  45.     }
  46. };
Logo

权威|前沿|技术|干货|国内首个API全生命周期开发者社区

更多推荐