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

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

源代码下载:MyActiveX20081229.rar

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

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

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

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

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

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

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

复制代码
#define  WM_OPTSUCCESS WM_APP+101  // 操作成功
#define  WM_OPTFAILED WM_APP+102     // 操作失败
复制代码

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

复制代码
/////////////////////
// 回调函数
//////////////////////// /
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出外部页面可以响应的事件。

复制代码
LRESULT CMyActiveXCtrl::WindowProc(UINT msg, WPARAM wParam, LPARAM lParam)
{
    
switch  (msg)
    {
    
case  WM_OPTSUCCESS:
        {
// 操作成功,通知外部页面
            CString strOnLoaded( " OnLoaded " );
            
this -> CallJScript(strOnLoaded);
            
return  0 ;
        }
    
case  WM_OPTFAILED:
        {
// 操作失败,通知外部页面
            
// 这里不写了,同上面
        }
    }
    
return    COleControl::WindowProc(msg,wParam,lParam);   
}
复制代码

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

复制代码

int  CMyActiveXCtrl::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
    
if  (COleControl::OnCreate(lpCreateStruct)  ==  - 1 )
        
return  - 1 ;
    m_MainDialog.Create(IDD_MAINDIALOG, 
this );
    pThread.SetICallBack(
this ); // 设置主线程回调函数
    pThread.Start(); // 启动工作线程
     return  0 ;
}
复制代码

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

复制代码
void  CMyActiveXCtrl::OnClose(DWORD dwSaveOption)
{
    pThread.Stop(
true ); // 强行关闭工作线程
    COMRELEASE(pWebBrowser);
    COMRELEASE(pHTMLDocument);
    COleControl::OnClose(dwSaveOption);
}
复制代码

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

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

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

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

 

     一种是特殊模式,

 

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

复制代码
< PARAM  NAME ="IsSpecial"  VALUE ="TRUE" >
复制代码

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

复制代码
CString m_bIsSpecial; // 是否是"特殊"页面
复制代码

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

复制代码

void  CMyActiveXCtrl::DoPropExchange(CPropExchange *  pPX)
{
    ExchangeVersion(pPX, MAKELONG(_wVerMinor, _wVerMajor));
    COleControl::DoPropExchange(pPX);

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

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

复制代码
public :
    CMyDlgTwo m_dlgSpecial;
// 特殊模式
    CMyDlgThree m_dlgCommon; // 普通模式
复制代码

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

复制代码
int  CMyActiveXCtrl::OnCreate(LPCREATESTRUCT lpCreateStruct)
{
    
if  (COleControl::OnCreate(lpCreateStruct)  ==  - 1 )
        
return  - 1 ;

    CRect newRc; 
    
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/
本文版权归作者和博客园共有,欢迎转载,但请保留此段声明,并在文章页面明显位置给出原文连接。

作者:洞庭散人

出处:http://phinecos.cnblogs.com/    

本博客遵从 Creative Commons Attribution 3.0 License,若用于非商业目的,您可以自由转载,但请保留原作者信息和文章链接URL。
目录
相关文章
|
6月前
|
IDE C# 开发工具
2000条你应知的WPF小姿势 基础篇<40-44 启动关闭,Xaml,逻辑树>
2000条你应知的WPF小姿势 基础篇<40-44 启动关闭,Xaml,逻辑树>
31 0
|
7月前
|
API C# 数据安全/隐私保护
C#之二十一 创建MDI应用程序和组件开发
C#之二十一 创建MDI应用程序和组件开发
65 0
|
11月前
|
C# C++ Windows
3.只使用代码创建WPF应用程序
3.只使用代码创建WPF应用程序
81 0
WPF界面无法正常显示(资源引用,如转换器),但程序正常运行
WPF界面无法正常显示(资源引用,如转换器),但程序正常运行
WPF界面无法正常显示(资源引用,如转换器),但程序正常运行
|
XML 程序员 C语言
Qt编写控件属性设计器1-加载插件
一、前言 加载插件是整个属性设计器的第一步要打通的功能,插件中的控件都加载不了,后面就别搞别玩下去了没法玩的,要从一个动态库中加载出来控件,肯定需要用到反射机制,以前做.NET开发的时候就觉得反射这个东西相当强大,居然可以读取DLL加载出来控件,现在用Qt,发现Qt也有反射机制,也许这东东可能各大.
1009 0
|
消息中间件 C# Windows
WPF的消息机制(一)- 让应用程序动起来
原文:WPF的消息机制(一)- 让应用程序动起来 版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/powertoolsteam/article/details/6106485 ...
1032 0
|
C# 前端开发
WPF Adorner+附加属性 实现控件友好提示
原文:WPF Adorner+附加属性 实现控件友好提示 标题太空泛,直接上图   无论是在验证啊,还是提示方面等一些右上角的角标之类的效果,我们会怎么做? 这里介绍一种稍微简单一些的方法,利用附加属性和Adorner来完成。
949 0
|
C# 前端开发 设计模式
WPF设计界面不执行代码
原文:WPF设计界面不执行代码 一般在我们在设计WPF XAML界面时,XAML 引用一些后端的类。比如UserControl、Converter、MVVM,引用 xmlns:ALLUserControl="clr-namespace:程序集名.ALLUserControl;assembly=程序集名"。
1031 0
|
JavaScript 前端开发 C++
COM组件开发实践(七)---多线程ActiveX控件和自动调整ActiveX控件大小(上)
声明:本文代码基于CodeProject的文章《A Complete ActiveX Web Control Tutorial》修改而来,因此同样遵循Code Project Open License (CPOL)。
840 0