COM组件开发实践(八)---多线程ActiveX控件和自动调整ActiveX控件大小(下)

本文介绍如何在ActiveX控件中实现多线程处理,并通过子线程发送自定义消息通知主线程,同时解决了控件在不同页面中自适应大小的问题。

源代码下载:MyActiveX20081229.rar

声明:本文代码基于CodeProject的文章《A Complete ActiveX Web Control Tutorial》修改而来,因此同样遵循Code Project Open License (CPOL)

在上一篇文章《COM组件开发实践(七)---多线程ActiveX控件和自动调整ActiveX控件大小()》中介绍了ActiveX控件中使用多线程的基本需求,并提出了一个简单的线程模型,但却出现了意想不到的问题,本文将尝试给出问题的一个可行的解法,并同时解决上文中提出的第二个问题。

其实解决的思路也很简单,一开始我也早就想到了的,就是使用让子线程PostMessage来发出自定义的消息来通知主线程,特定的事件已经发生了,需求主线程去响应。这不是什么了不起的想法,但我对子线程PostMessage非常恐惧,因为以前的一个项目中就是这个问题导致了内存泄露,所以这个方案一开始就被我否定了。

遍寻解决之道不可得时,只得在csdn的论坛上发贴求教高手了,具体的讨论请参考这个帖子:

http://topic.youkuaiyun.com/u/20081226/17/9bf0ae08-c54d-4934-b1b2-91baa27ff76e.html

看到jameshooo(胡柏华)的回帖后,还是决定回到起点,尝试用PostMessage这个方案。

首先自定义两个事件,分别表示操作成功和操作失败

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> #define WM_OPTSUCCESSWM_APP+101 // 操作成功
#define WM_OPTFAILEDWM_APP+102 // 操作失败

然后回调函数中就变得非常简单,只需要post对应的事件即可。

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> /////////////////////
// 回调函数
//////////////////////// /
void CMyActiveXCtrl::OnSuccesful()
{
// 操作成功
this -> PostMessage(WM_OPTSUCCESS,(WPARAM)NULL,(LPARAM)NULL);
}

void CMyActiveXCtrl::OnFailed()
{
// 操作失败
this -> PostMessage(WM_OPTFAILED,(WPARAM)NULL,(LPARAM)NULL);
}

再重载消息处理函数WindowProc,在其中调用外部的JavaScript函数或者Fire出外部页面可以响应的事件。

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> LRESULTCMyActiveXCtrl::WindowProc(UINTmsg,WPARAMwParam,LPARAMlParam)
{
switch (msg)
{
case WM_OPTSUCCESS:
{
// 操作成功,通知外部页面
CStringstrOnLoaded( " OnLoaded " );
this -> CallJScript(strOnLoaded);
return 0 ;
}
case WM_OPTFAILED:
{
// 操作失败,通知外部页面
// 这里不写了,同上面
}
}
return COleControl::WindowProc(msg,wParam,lParam);
}

OnCreate函数中加入启动工作线程代码:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->
int CMyActiveXCtrl::OnCreate(LPCREATESTRUCTlpCreateStruct)
{
if (COleControl::OnCreate(lpCreateStruct) == - 1 )
return - 1 ;
m_MainDialog.Create(IDD_MAINDIALOG,
this );
pThread.SetICallBack(
this ); // 设置主线程回调函数
pThread.Start(); // 启动工作线程
return 0 ;
}

重载掉OnClose函数,在其中加入关闭工作线程的代码:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> void CMyActiveXCtrl::OnClose(DWORDdwSaveOption)
{
pThread.Stop(
true ); // 强行关闭工作线程
COMRELEASE(pWebBrowser);
COMRELEASE(pHTMLDocument);
COleControl::OnClose(dwSaveOption);
}

到此为止,一个多线程的ActiveX控件就诞生了。这里是不会发生以前我遇到的内存泄露的,因为情况不同了,只是在回调函数中简单的post一个message,并没有new一个内存区域并将这块内存作为参数post给主线程,后面这种情况是可能会内存泄露的。

Ok,下面来考虑第二个问题,先简单介绍下具体需求:就是一个AcitveX控件会用到不同的页面中,每个页面对这个控件的需求也不同,也就要求在两个不同的页面中,控件显示的大小也不同。

jameshooo(胡柏华) 回帖说:改变控件大小要通知容器,由容器再反过来通知控件改变大小,不然没有任何效果。调用IOleInPlaceSite::OnPosRectChange即可。因此就根据这个来尝试提出一个解决方案来。

假设有两种模式的控件,一种是普通模式, 如下图所示:

一种是特殊模式,

为了区别开两者,就考虑在web页面中通过设置参数的方式来通知ActiveX控件,对于不同的模式填充不同的对话框就可以了。我们在web页面中控件部分加入如下参数:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> < PARAM NAME ="IsSpecial" VALUE ="TRUE" >

相应的在CMyActiveXCtrl类中加入一个变量,这里为简单起见,选择了类型为CString型,主要是为了传参数方便。

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> CStringm_bIsSpecial; // 是否是"特殊"页面

Web页面传入的参数值在下面这个函数中读取:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />-->
void CMyActiveXCtrl::DoPropExchange(CPropExchange * pPX)
{
ExchangeVersion(pPX,MAKELONG(_wVerMinor,_wVerMajor));
COleControl::DoPropExchange(pPX);

PX_String(pPX,_T(
" IsSpecial " ),m_bIsSpecial); // 读取外部设置的参数
}

为了供控件选择,这里提供了两种模式的对话框:

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> public :
CMyDlgTwom_dlgSpecial;
// 特殊模式
CMyDlgThreem_dlgCommon; // 普通模式

然后在创建和绘制对话框时,通过检测参数是否为空就知道待创建的对话框类型到底是普通还是特殊了。

<!--<br /><br />Code highlighting produced by Actipro CodeHighlighter (freeware)<br />http://www.CodeHighlighter.com/<br /><br />--> int CMyActiveXCtrl::OnCreate(LPCREATESTRUCTlpCreateStruct)
{
if (COleControl::OnCreate(lpCreateStruct) == - 1 )
return - 1 ;

CRectnewRc;
if (m_bIsSpecial.Compare(_T( "" )) == 0 )
{
// 没设置参数,"普通“模式
this -> m_dlgCommon.Create(IDD_DIALOG3, this );
// 设置控件的大小

newRc.left
= 0 ;
newRc.top
= 0 ;
newRc.right
= 200 ;
newRc.bottom
= 200 ;
}
else
{
// 设置了参数,”特殊“模式
this -> m_dlgSpecial.Create(IDD_DIALOG2, this );
// 设置控件的大小
newRc.left = 0 ;
newRc.top
= 0 ;
newRc.right
= 200 ;
newRc.bottom
= 200 ;
}
this -> m_pInPlaceSite -> OnPosRectChange( & newRc);
return 0 ;
}

void CMyActiveXCtrl::OnDraw(
CDC
* pdc, const CRect & rcBounds, const CRect & rcInvalid)
{
if ( ! pdc)
return ;
if (m_bIsSpecial.Compare(_T( "" )) == 0 )
{
// 没设置参数,"普通“模式
this -> m_dlgCommon.MoveWindow(rcBounds,TRUE);
}
else
{
// 设置了参数,”特殊“模式
this -> m_dlgSpecial.MoveWindow(rcBounds,TRUE);
}
}

这种方法对于我目前的需求刚好是满足的,但也许还有其他更好的方法,也希望有知道的能贡献出来,一起学习下。

作者:phinecos(洞庭散人)
出处:http://phinecos.cnblogs.com/
本文版权归作者和博客园共有,欢迎转载,但请保留此段声明,并在文章页面明显位置给出原文连接。

评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值