对于CDHtmlDialog和JavaScript、HTML配合使用的一些技术总结

转自:http://blog.csdn.net/pjl1119/article/details/7713892

CDHtmlDialog可以方便的将网页嵌入对话框,使得在程序设计中人机界面(DHTML网页)与控制逻辑(CDialog)可以很好的分离。

 

1、屏蔽安全性提示,不再弹出控件是否安全的提示框。

重载CanAccessExternal()函数,直接 return TRUE;
头文件中:virtual BOOL CanAccessExternal();

.cpp中:

BOOL CBaseDHtmlDialog::CanAccessExternal()
{
 return TRUE;
}

 

2、拦截系统默认的右键菜单

重载ShowContextMenu函数。

头文件中:virtual HRESULT STDMETHODCALLTYPE ShowContextMenu(DWORD dwID,POINT *ppt,IUnknown *pcmdtReserved,IDispatch *pdispReserved);

.cpp中:

HRESULT STDMETHODCALLTYPE CBaseDHtmlDialog::ShowContextMenu(DWORD dwID,  POINT *ppt,  IUnknown *pcmdtReserved,  IDispatch *pdispReserved)
{
      return S_OK;
}

 

3、拦截一些系统快捷键

头文件中:STDMETHOD(TranslateAccelerator)(LPMSG lpMsg, const GUID *pguidCmdGroup, DWORD nCmdID);

HRESULT STDMETHODCALLTYPE CBaseDHtmlDialog::TranslateAccelerator(LPMSG lpMsg, 
                 const GUID *pguidCmdGroup, 
                 DWORD nCmdID)
{
 if (lpMsg && lpMsg->message == WM_KEYDOWN)
 {
  bool bCtrl = (0x80 == (0x80 & GetKeyState(VK_CONTROL)));

  // prevent Ctrl+N
  if (lpMsg->wParam == 'N' && bCtrl)
  {
   return S_OK;
  }

  // prevent Ctrl+F
  if (lpMsg->wParam == 'F' && bCtrl)
  {
   return S_OK;
  }

  // prevent F5
   if (lpMsg->wParam == VK_F5)
   {
    return S_OK;
   }

  // prevent ESC
  if (lpMsg->wParam == VK_ESCAPE)
  {
   return S_OK;
  }

  // prevent ENTER
  if (lpMsg->wParam == VK_RETURN)
  {
   return S_OK;
  }
 }
 return CDHtmlDialog::TranslateAccelerator(lpMsg, pguidCmdGroup, nCmdID);
}

 

4、添加滚动条

BOOL CBaseDHtmlDialog::OnInitDialog()
{
 SetHostFlags(DOCHOSTUIFLAG_FLAT_SCROLLBAR|DOCHOSTUIFLAG_NO3DBORDER);//必须在 CDHtmlDialog::OnInitDialog();之前

 CDHtmlDialog::OnInitDialog();
 m_pBrowserApp->put_Silent(VARIANT_TRUE);//屏蔽警告提示

 return TRUE;
}

 


5.数据处理尽量交给JavaScript,Dialog只做有意义的事。

IE先于CDHtmlDialog处理个事件,如鼠标事件。


6.从CDHtmlDialog调用网页中JavaScript函数的方法。


其中pDoc指针参数可通过CDHtmlDialog::GetDHtmlDocument(&pDoc)函数获得; strFunctionName指示函数名; dispParams为传给函数的参数列表,其使用方法请查阅MSDN相关文档; varResult为函数返回值; exceptInfo为JavaScript函数执行时抛出的异常; nArgErr返回第一个出错的参数的下标,由于参数列表中参数的逻辑顺序为JavaScript函数定义的参数的顺序的逆序,所以应特别注意该返回值所指示的具体位置。

HRESULT CallJSFunction(IHTMLDocument2* pDoc2, CString strFunctionName, DISPPARAMS dispParams, VARIANT* varResult, EXCEPINFO* exceptInfo, UINT* nArgErr )

{

IDispatch *pDispScript = NULL;

HRESULT hResult;

hResult = pDoc2->get_Script(&pDispScript);

if(FAILED(hResult)) { return S_FALSE; }

DISPID dispid;

CComBSTR objbstrValue = strFunctionName;

BSTR bstrValue = objbstrValue.Copy();

OLECHAR *pszFunct = bstrValue ;

hResult = pDispScript->GetIDsOfNames(IID_NULL, &pszFunct,1, LOCALE_SYSTEM_DEFAULT, &dispid);
if (S_OK != hResult)

{ pDispScript->Release();

return hResult; }

varResult->vt = VT_VARIANT;

hResult = pDispScript->Invoke(dispid, IID_NULL, LOCALE_USER_DEFAULT, DISPATCH_METHOD, &dispParams, varResult, exceptInfo, nArgErr);

pDispScript->Release();

return hResult;

}

 

7.JavaScript通过external调用CDHtmlDialog的方法。


<1>让CDHtmlDialog对象自身支持自动化 EnableAutomation();(只要是从CCmdTarget派生下来的类都可以支持,放在构造函数或Create重载函数中)


<2>将自身暴露给Script引擎: SetExternalDispatch(GetIDispatch(TRUE)); (将浏览器控件的扩展接口设置为对话框自身的IDispatch ,放在CMyDHTMLDialog::OnInitDialog中调用)


<3>声明DISPATCH_MAP。在头文件中添加 DECLARE_DISPATCH_MAP()


<4>定义DISPATCH映射(MyDHTMLDialog.cpp)

BEGIN_DISPATCH_MAP(CMyDHtmlDialog, CDHtmlDialog)

   DISP_FUNCTION(CMobileThemesHtml,"OnExternalTrackMenu",OnTrackMenu,VT_NULL,VTS_I4)  

   DISP_FUNCTION(CMobileThemesHtml,"OnExternalCheckboxClick",OnCheckboxClick,VT_NULL,VTS_I4 VTS_I4 VTS_I4)

END_DISPATCH_MAP()


<5>函数实现

OnExternalTrackMenu为HTML中JavaScript调用的函数,如:

<INPUT id="Button1" type="button" value="Button1" name="Button1"

οnclick="external.OnExternalTrackMenu(this.id);">

OnTrackMenu为VC++中的响应函数。如:void CMyDHtmlDialog::OnTrackMenu(int id)

参数VT_NULL(以VT开头),表示传给JavaScript的参数,若无,用VT_NULL代替。

参数VTS_I4表示JavaScript中返回的参数,以VTS开头,如无参数,用VTS_NONE代替。可有多个,如:

void CMyDHtmlDialog::OnCheckboxClick(int nTotalNumber,int nSelectedNumber,int nSystemThemeNumber)


 

8、响应HTML中控件的另一种方法

BEGIN_DHTML_EVENT_MAP(CMyDHtmlDialog)
  DHTML_EVENT_CLASS(DISPID_HTMLELEMENTEVENTS_ONMOUSEDOWN, _T("img_cls"), OnButtonImage)
END_DHTML_EVENT_MAP()

 

DISPID_HTMLELEMENTEVENTS_ONMOUSEDOWN:为鼠标点击方式,有多种;

img_cls:为HTML中一控件的class名称;

OnButtonImage:VC++中的响应函数。

 

9、调用JavaScript函数时,向其传递多个参数。

 参数传入的顺序,与JavaScript中的参数反序对应。

BOOL CMyDHtmlDialog::AddThemeToHtml(LPCTSTR name,  float size,  LPCTSTR picpath,    int id,  bool bSystemTheme)
{
 CComVariant* pvars = new CComVariant[5];
 DISPPARAMS dispParams = { pvars, NULL, 5, 0 };
 
 //ID
 pvars[0].vt = VT_INT;
 pvars[0].intVal= id;

 //路径
 pvars[1].vt = VT_BSTR;
 CComBSTR bstr2 = picpath ;
 bstr2.CopyTo(&pvars[1].bstrVal);

 //大小

 CString csSize;
 csSize.Format(_T("%.1fM"), size);
 pvars[2].vt = VT_BSTR;
 CComBSTR bstr1 = csSize ;
 bstr1.CopyTo(&pvars[2].bstrVal);

 //名称
 pvars[3].vt = VT_BSTR;
 CComBSTR bstr0 = name ; 
 bstr0.CopyTo(&pvars[3].bstrVal);

  pvars[4].vt = VT_INT;
 pvars[4].intVal= bSystemTheme ? 1 : 0;
 
 BOOL bResult = CallJsFunc(_T("AddNode"),dispParams,pvars);
 delete [] pvars;

 return bResult;

}

 

BOOL CMyDHtmlDialog::CallJsFunc(CString szFuncName,DISPPARAMS dispParams,CComVariant* vResult)
{
 HRESULT hr = S_OK;

 CComPtr<IDispatch> spScript;
 if (m_spHtmlDoc==NULL)
 {
  return FALSE;
 }

 hr = m_spHtmlDoc->get_Script(&spScript);
 if (FAILED(hr))
 {
  return FALSE;
 }

 CComBSTR bstrFunc(szFuncName);
 DISPID dispid = 0;
 hr = spScript->GetIDsOfNames(IID_NULL, &bstrFunc, 1, LOCALE_SYSTEM_DEFAULT, &dispid);
 if (FAILED(hr))
 {
  spScript.Release();
  return FALSE;
 }

 EXCEPINFO except;
 memset(&except, 0, sizeof(except));

 UINT nArgErr = (UINT)-1; //initialize to invalid arg

 hr = spScript->Invoke(dispid, IID_NULL, 0, DISPATCH_METHOD,
  &dispParams, vResult, &except, &nArgErr);

 if (FAILED(hr))
 {
  spScript.Release();
  return FALSE;
 }
 spScript.Release();
 return TRUE;

}

 

JavaScript中对应的函数:

function AddNode(bSystemTheme,name,size,picpath,id)
  {
   total_number ++;
   if (bSystemTheme==1) systheme_total ++;

            var myElement = document.createElement('li');
            myElement.className = "li_cls";
            myElement.id = id+"li";
            if (bSystemTheme==0)
            {
                myElement.οnmοuseοut=function(){Hide(this.id)};
                myElement.οnmοuseοver=function(){Show(this.id)};
            }
            document.getElementById("frame").appendChild(myElement);

           var chkbox = document.createElement('input');
           chkbox.type="checkbox";
           chkbox.id=id+"thm";
           chkbox.className="chk_cls";
           chkbox.name=id;
     chkbox.οnmοuseup= function(){ProcWhenSelectorChange(this.checked, bSystemTheme)};
           myElement.appendChild(chkbox); 
  
           var span1 = document.createElement('span');
           span1.className="cap_td";
           span1.title=name;
           span1.innerHTML=name;
           myElement.appendChild(span1);

           var span2 = document.createElement('span');
           span2.className="wgt_td";
           if (bSystemTheme==1) size=size+"(内置)";
           span2.innerHTML=size;
           myElement.appendChild(span2);

           var img1 = document.createElement('img');
           img1.className="img_show";
           img1.src=picpath; 
           img1.id=id+"thm"; 
           img1.οnmοusedοwn=function(){ClinkMouseBtnDown(event,this.id,bSystemTheme)};
           img1.οnmοuseup=function(){ClinkImageUp(event,id)};
           myElement.appendChild(img1); 
           if (bSystemTheme==0)
           {
                var btn = document.createElement('button');
                btn.className="lnkbtn";
                btn.innerHTML="卸载";
                btn.οnmοuseup= function(){ClinkButtonUp(event,id)};
                myElement.appendChild(btn); 
           }
     }

 

10、悬浮按钮的实现方法

鼠标移至某控件(如img)上时,出现按钮,鼠标移开后,按钮消失。

方法1:将图片作为按钮父元素的背景图片。

<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" />
<title>TEST</title>
<script language="JavaScript" type="text/javascript"> 
 function Show() { var tmp = document.getElementById("lnkbtn");
   tmp.style.display="inline";}   
  function Hide(){ var tmp = document.getElementById("lnkbtn");
   tmp.style.display="none";}    
</script>
<style  type="text/css">
#lnkbtn{font:normal bold 12px/130% "Microsoft YaHei",serif; white-space: nowrap;
   cursor:hand; color:#5f7e4b; display:none;}
#Layer1 {position:absolute; width:500px; height:315px; z-index:1;
 background-image: url(E://test.jpg);background-repeat: no-repeat;}
</style>
</head>
<body >
    <div id="Layer1" οnmοuseοver="Show()" οnmοuseοut="Hide()">
        <button id="lnkbtn"> 刷新</button>
    </div>
</body>
</html>


 

方法2:图片和按钮都采用绝对坐标

<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<meta http-equiv="Content-Type" content="text/html; charset=gb2312" />
<title>TEST</title>
<script language="JavaScript" type="text/javascript"> 
 function Show(id) { var tmp = document.getElementById(id);
   tmp.nextSibling.style.display="inline";}   
  function Hide(id){ var tmp = document.getElementById(id);
   tmp.nextSibling.style.display="none";}    
</script>
<style  type="text/css">
.li_cls {background: transparent;position: relative;display: inline-block;
    width:150px;height:280px; overflow: hidden;
    float: left;margin: 0px 30px 30px 4px;border:1px solid rgb(60,60,235);}   
#lnkbtn {font:normal bold 12px/130% "Microsoft YaHei",serif;white-space: nowrap;cursor:hand;
   position: absolute;color:#5f7e4b;display:none;width:40px;height:25px;left:55px; top:221px; }
#img_show{position: absolute; height:230px; width:138px; left:5px;top:46px;}
</style>
</head>
<body >
    <div class="li_cls">
        <img id="img_show" src="E://test.jpg" alt="郁金香"  οnmοuseοver="Show(this.id)" οnmοuseοut="Hide(this.id)"/>
        <button id="lnkbtn"> 刷新 </button>
    </div>
</body>
</html>

 

11、在JavaScript中删除HTML的节点时,获取的节点数会随着删除动作的进行而逐渐减小。

function RemoveAllNodes()
  {
      var myform = document.getElementById("frame");
      var nodes = myform.childNodes;
     
      while (nodes.length != 0){
           for(var i = 0; i < nodes.length; i++){
                  myform.removeChild(nodes[i]);
          }
      }
  }

如上:每次removeChild成功后,nodes数组的大小会减1。若想删除所有的节点,需采用上面双循环的方式。
  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
### 回答1: vc2020 cdhtmldialog是继承自MFC中的CDHtmlDialog类的。CDHtmlDialog类是MFC框架提供的一个用于创建基于HTML和脚本的自定义对话框的类。 CDHtmlDialog类可以让开发者使用HTML和脚本语言(如JavaScript)来创建对话框上的控件、界面和交互行为。它基于IE浏览器引擎,能够支持HTML和脚本语言的强大功能,使得开发者能够以更灵活和可扩展的方式来构建用户界面。 继承自CDHtmlDialog类的vc2020 cdhtmldialog可以根据具体需求来定制化自己的对话框。通过继承CDHtmlDialog类,可以在自己的对话框上添加各种HTML控件,如按钮、文本框、下拉框等,并通过JavaScript脚本来实现它们的交互行为。 CDHtmlDialog类提供了一系列重要的成员函数和事件处理函数,用于处理HTML控件的事件、脚本的执行和数据传递等。开发者可以根据需要重写这些函数,实现自己对控件和脚本的处理逻辑。 与传统的MFC对话框相比,vc2020 cdhtmldialog具有更高的扩展性和灵活性。通过使用HTML和脚本语言,开发者可以实现更复杂和动态的用户界面,以及更强大的交互功能。同时,CDHtmlDialog类的封装也简化了开发过程,降低了开发难度。 综上所述,vc2020 cdhtmldialog继承自MFC的CDHtmlDialog类,通过HTML和脚本语言来构建自定义对话框,具有高度的灵活性和扩展性,能够实现复杂的界面和交互功能。 ### 回答2: vc2020 cdhtmldialog是一个可以用来创建基于HTML的对话框的类。它使用CDHtmlDialog类作为基类,通过继承该类,我们可以方便地创建自己的HTML对话框。 继承是面向对象编程的一个重要概念,通过继承,我们可以从已有的类派生出新的类,新的类将拥有原有类的属性和方法。对于vc2020 cdhtmldialog来说,继承可以让我们自定义HTML对话框的行为和外观,以满足我们的特定需求。 继承vc2020 cdhtmldialog时,我们可以重写其父类的方法,实现自己的逻辑。例如,我们可以重写OnInitDialog方法,在对话框初始化时执行一些自定义操作;或者重写OnDocumentComplete方法,当HTML文档加载完成时执行一些需要的操作。 另外,我们还可以通过继承来添加新的属性和方法,以扩展vc2020 cdhtmldialog的功能。我们可以在新的类中添加新的成员变量和成员函数,并在代码中使用它们。 总之,通过继承vc2020 cdhtmldialog,我们可以根据自己的需求创建出功能丰富、个性化的HTML对话框。继承的优势在于代码的复用和扩展性,使得我们可以更加灵活地定制和开发HTML对话框应用程序。 ### 回答3: vc2020的cdhtmldialog是基于CDHtmlDialog类的,它允许我们创建使用HTML和脚本的用户界面。继承是一种面向对象编程的概念,它允许我们从已有的类中派生出新的类,并继承它的属性和方法。 继承CDHtmlDialog类有几个好处。首先,我们可以利用CDHtmlDialog已有的功能来构建我们自己的用户界面。CDHtmlDialog类已经集成了许多常用的HTML和脚本控件,比如按钮、文本框、下拉列表等,我们可以直接使用它们来设计我们的界面,而无需从头编写这些控件的样式和行为。 此外,通过继承CDHtmlDialog类,我们可以重写其函数以实现我们的逻辑。CDHtmlDialog类提供了一些虚函数,比如OnInitDialog()、DoDataExchange()和OnDocumentComplete()等,通过重写这些函数,我们可以在特定的事件发生时执行我们自己的代码。例如,在OnInitDialog()函数中,我们可以初始化一些数据或者执行一些必要的操作;在OnDocumentComplete()函数中,我们可以处理网页加载完成后的逻辑。 另外,继承CDHtmlDialog类还使得我们能够对话框进行更加灵活的自定义。我们可以通过添加新的成员函数和成员变量来实现我们自己的功能,或者通过重写基类的函数来改变原有的行为。这种灵活性使得我们可以根据具体需求对对话框进行定制,满足项目的特殊要求。 总之,通过继承CDHtmlDialog类,我们可以构建基于HTML和脚本的用户界面,并且能够对其进行进一步的定制和自定义。这样,我们能够更加方便地开发出功能强大且美观的用户界面,提升用户体验。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值