有窗口句柄的子窗口
IRealWndHandler
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 |
struct IRealWndHandler : public IObjRef { /** * SRealWnd::OnRealWndCreate * @brief 窗口创建 * @param SRealWnd *pRealWnd -- 窗口指针 * * Describe 窗口创建 */ virtual HWND OnRealWndCreate(SRealWnd *pRealWnd)=NULL; /** * SRealWnd::OnRealWndDestroy * @brief 销毁窗口 * @param SRealWnd *pRealWnd -- 窗口指针 * * Describe 销毁窗口 */ virtual void OnRealWndDestroy(SRealWnd *pRealWnd)=NULL; /** * SRealWnd::OnRealWndInit * @brief 初始化窗口 * @param SRealWnd *pRealWnd -- 窗口指针 * @return BOOL -- FALSE:交由系统处理,TRUE:用户处理 * * Describe 初始化窗口 */ virtual BOOL OnRealWndInit(SRealWnd *pRealWnd)=NULL; /** * SRealWnd::OnRealWndSize * @brief 调整窗口大小 * @param SRealWnd *pRealWnd -- 窗口指针 * @return BOOL -- FALSE:交由SOUI处理; TRUE:用户管理窗口的移动 * * Describe 调整窗口大小 */ virtual BOOL OnRealWndSize(SRealWnd *pRealWnd)=NULL; }; |
实现
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 |
#pragma once #include <unknown/obj-ref-impl.hpp> namespace SOUI { class CSouiRealWndHandler :public TObjRefImpl2<IRealWndHandler,CSouiRealWndHandler> { public: CSouiRealWndHandler(void); ~CSouiRealWndHandler(void); /** * SRealWnd::OnRealWndCreate * @brief 创建真窗口 * @param SRealWnd * pRealWnd -- 窗口指针 * @return HWND -- 创建出来的真窗口句柄 * Describe */ virtual HWND OnRealWndCreate(SRealWnd *pRealWnd); /** * SRealWnd::OnRealWndDestroy * @brief 销毁窗口 * @param SRealWnd *pRealWnd -- 窗口指针 * * Describe 销毁窗口 */ virtual void OnRealWndDestroy(SRealWnd *pRealWnd); /** * SRealWnd::OnRealWndInit * @brief 初始化窗口 * @param SRealWnd *pRealWnd -- 窗口指针 * * Describe 初始化窗口 */ virtual BOOL OnRealWndInit(SRealWnd *pRealWnd); /** * SRealWnd::OnRealWndSize * @brief 调整窗口大小 * @param SRealWnd *pRealWnd -- 窗口指针 * @return BOOL -- TRUE:用户管理窗口的移动;FALSE:交由SOUI自己管理。 * Describe 调整窗口大小, 从pRealWnd中获得窗口位置。 */ virtual BOOL OnRealWndSize(SRealWnd *pRealWnd); }; } |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 |
#include "StdAfx.h" #include "SouiRealWndHandler.h" namespace SOUI { CSouiRealWndHandler::CSouiRealWndHandler(void) { } CSouiRealWndHandler::~CSouiRealWndHandler(void) { } HWND CSouiRealWndHandler::OnRealWndCreate( SRealWnd *pRealWnd ) { const SRealWndParam ¶m=pRealWnd->GetRealWndParam(); if(param.m_strClassName==_T("button")) {//只实现了button的创建 //分配一个MFC CButton对象 CButton *pbtn=new CButton; //创建CButton窗口,注意使用pRealWnd->GetContainer()->GetHostHwnd()作为CButton的父窗口 //把pRealWnd->GetID()作为真窗口的ID pbtn->Create(param.m_strWindowName,WS_CHILD|WS_VISIBLE|BS_PUSHBUTTON,::CRect(0,0,0,0),CWnd::FromHandle(pRealWnd->GetContainer()->GetHostHwnd()),pRealWnd->GetID()); //把pbtn的指针放到SRealWnd的Data中保存,以便在窗口destroy时释放pbtn对象。 pRealWnd->SetData(pbtn); //返回成功创建后的窗口句柄 return pbtn->m_hWnd; }else { return 0; } } void CSouiRealWndHandler::OnRealWndDestroy( SRealWnd *pRealWnd ) { const SRealWndParam ¶m=pRealWnd->GetRealWndParam(); if(param.m_strClassName==_T("button")) {//销毁真窗口,释放窗口占用的内存 CButton *pbtn=(CButton*) pRealWnd->GetData(); if(pbtn) { pbtn->DestroyWindow(); delete pbtn; } } } //不处理,返回FALSE BOOL CSouiRealWndHandler::OnRealWndSize( SRealWnd *pRealWnd ) { return FALSE; } //不处理,返回FALSE BOOL CSouiRealWndHandler::OnRealWndInit( SRealWnd *pRealWnd ) { return FALSE; } } |
xml
1 2 3 4 5 6 7 8 9 10 11 |
<SOUI title="DUI-DEMO" width="600" height="400" appwin="0" ncRect="5,5,5,5" resize="1" translucent="0"> <root skin="skin.bkframe" cache="1"> <caption pos="0,0,-0,29"> <text pos="11,9" >%title% ver:%ver%</text> <imgbtn id="1" name="btn_close" skin="skin.btnclose" pos="-45,0" tip="close" animate="0"/> </caption> <window pos="0,29,-0,-0"> <realwnd pos="10,10,-10,-10" name="mfcbtn" wndclass="button" id="100" wndname="MFC Button"/> </window> </root> </SOUI> |
消息响应
- 由于真窗口是SOUI主窗口的子窗口,因此真窗口的消息可以在SOUI主窗口的消息映射表中处理。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 |
#pragma once class CRealWndDlg : public SOUI::SHostDialog { public: CRealWndDlg(void); ~CRealWndDlg(void); //响应MFC.button的按下消息, nID==100为在XML中指定的realwnd的id属性。 void OnBtnClick( UINT uNotifyCode, int nID, HWND wndCtl ) { if(uNotifyCode == BN_CLICKED && nID == 100) { SOUI::SMessageBox(m_hWnd,_T("the real mfc button is clicked!"),_T("mfc.demo"),MB_OK|MB_ICONEXCLAMATION); } } //消息映射表 BEGIN_MSG_MAP_EX(CMainDlg) MSG_WM_COMMAND(OnBtnClick) CHAIN_MSG_MAP(SOUI::SHostDialog) REFLECT_NOTIFICATIONS_EX() END_MSG_MAP() }; |
注意
- 有真窗口时,SOUI主窗口不能设置
translucent="1"
,因为任何子窗口在半透明窗口上都不能正常显示。
定时器
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 |
class SOUI_EXP STimerID { public: DWORD Swnd:24; //窗口句柄,如果窗口句柄超过24位范围,则不能使用这种方式设置定时器 DWORD uTimerID:7; //定时器ID,一个窗口最多支持128个定时器。 DWORD bSwndTimer:1; //区别通用定时器的标志,标志为1时,表示该定时器为SWND定时器 STimerID(SWND hWnd,char id) { SASSERT(hWnd<0x00FFFFFF && id>=0); bSwndTimer=1; Swnd=hWnd; uTimerID=id; } STimerID(DWORD dwID) { memcpy(this,&dwID,sizeof(DWORD)); } operator DWORD &() const { return *(DWORD*)this; } }; |
- 低24位用来存储SWindow的窗口ID(swnd)。
- 高8位中保留最高位设置为1,用来区别直接使用
::SetTimer
创建的定时器(不可以把最高位置1)。 - 剩下7位用于
SWindow
中作为定时器ID。因此在SOUI
中,一个SWindow
最多可以创建0-127个定时器。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
/** * SWindow::SetTimer * @brief 利用窗口定时器来设置一个ID为0-127的SWND定时器 * @param char id -- 定时器ID * @param UINT uElapse -- 延时(MS) * @return BOOL * * Describe 参考::SetTimer */ BOOL SWindow::SetTimer(char id,UINT uElapse); /** * KillTimer * @brief 删除一个SWND定时器 * @param char id -- 定时器ID * @return void * * Describe */ void KillTimer(char id); |
响应
1 2 3 4 5 |
SOUI_MSG_MAP_BEGIN() MSG_WM_TIMER_EX(OnTimer) //定时器消息 MSG_WM_PAINT_EX(OnPaint) //窗口绘制消息 MSG_WM_SHOWWINDOW(OnShowWindow)//窗口显示状态消息 SOUI_MSG_MAP_END() |
32位定时器
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 |
/** * SetTimer2 * @brief 利用函数定时器来模拟一个兼容窗口定时器 * @param UINT_PTR id -- 定时器ID * @param UINT uElapse -- 延时(MS) * @return BOOL * * Describe 由于SetTimer只支持0-127的定时器ID,SetTimer2提供设置其它timerid * 能够使用SetTimer时尽量不用SetTimer2,在Kill时效率会比较低 */ BOOL SetTimer2(UINT_PTR id,UINT uElapse); /** * KillTimer2 * @brief 删除一个SetTimer2设置的定时器 * @param UINT_PTR id -- SetTimer2设置的定时器ID * @return void * * Describe 需要枚举定时器列表 */ void KillTimer2(UINT_PTR id); |
响应32位定时器
1 2 3 4 5 6 7 8 9 10 11 |
#define WM_TIMER2 (WM_USER+5432) //定义一个与HWND定时器兼容的SOUI定时器 #define MSG_WM_TIMER2(func) \ if (uMsg == WM_TIMER2) \ { \ SetMsgHandled(TRUE); \ func(wParam); \ lResult = 0; \ if(IsMsgHandled()) \ return TRUE; \ } |
注意
- 前面两种定时器都是在控件开发的时候使用定时器的方法。
- 在应用层,可以为宿主窗口直接使用
::SetTimer
或者宿主窗口的基类:CSimpleWnd::SetTimer
来创建定时器(注意最高位必须是0)。- 在响应这类定时器时,一样可以在宿主窗口的消息映射表中使用MSG_WM_TIMER来响应定时器消息。
- 需要注意的是,这个映射宏会截获所有分发给宿主窗口的定时器,如果不是自己创建的定时器,则需要继续交给基类处理。
- 可以调用:
SetMsgHandled(FALSE);
或者:SHostWnd::OnTimer(UINT_PTR idEvent);
实现。
本文为原创文章,版权归Aet所有,欢迎分享本文,转载请保留出处!
你可能也喜欢
- ♥ 各平台调试方法总结记述一09/25
- ♥ C++并发编程 _ 基于锁的数据结构08/19
- ♥ C++_指针引用09/19
- ♥ CommandLine库gflags使用01/12
- ♥ X86_64汇编学习记述三08/08
- ♥ 预处理指令记录:一07/09