如何拖动用duilib创建出来的窗口
Posted
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了如何拖动用duilib创建出来的窗口相关的知识,希望对你有一定的参考价值。
在基类中做默认的控件拖动判断, 在子类中做特殊控件判断, e.g. 点击某个静态控件时,要弹出一个对话框.[cpp] view
plaincopyprint?
class CMainDlg : public CXmlWnd
class CMainDlg : public CXmlWnd
基类:
[cpp] view
plaincopyprint?
virtual LRESULT OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled);
ate:
BOOL IsInStaticControl(CControlUI * pControl);
virtual LRESULT OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled);
private:
BOOL IsInStaticControl(CControlUI * pControl);
[cpp] view
plaincopyprint?
LRESULT CXmlWnd::OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
POINT pt;
RECT rcClient;
RECT rcCaption;
CControlUI * pControl = NULL;
rcCaption = m_PaintManager.GetCaptionRect();
GetClientRect(m_PaintManager.GetPaintWindow(), &rcClient);
pt.x = GET_X_LPARAM(lParam);
pt.y = GET_Y_LPARAM(lParam);
::ScreenToClient(m_PaintManager.GetPaintWindow(), &pt);
if (-1 == rcCaption.bottom) ///< xml中描述bottom为-1时,整个窗口区域都可以拖动
rcCaption.bottom = rcClient.bottom;
if ((pt.x >= rcClient.left)
&& (pt.x < rcClient.right)
&& (pt.y >= rcCaption.top)
&& (pt.y < rcCaption.bottom))
pControl = m_PaintManager.FindControl(pt);
if (IsInStaticControl(pControl))
return HTCAPTION;
return __super::OnNcHitTest(uMsg, wParam, lParam, bHandled);
BOOL CXmlWnd::IsInStaticControl(CControlUI * pControl)
CDuiString strClassName;
std::vector<CDuiString> vctStaticName;
std::vector<CDuiString>::iterator it;
if (NULL == pControl)
return FALSE;
strClassName = pControl->GetClass();
strClassName.MakeLower();
vctStaticName.push_back(L"controlui");
vctStaticName.push_back(L"textui");
vctStaticName.push_back(L"labelui");
vctStaticName.push_back(L"containerui");
vctStaticName.push_back(L"horizontallayoutui");
vctStaticName.push_back(L"verticallayoutui");
vctStaticName.push_back(L"tablayoutui");
vctStaticName.push_back(L"childlayoutui");
vctStaticName.push_back(L"dialoglayoutui");
it = std::find(vctStaticName.begin(), vctStaticName.end(), strClassName);
return (it != vctStaticName.end());
LRESULT CXmlWnd::OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
POINT pt;
RECT rcClient;
RECT rcCaption;
CControlUI * pControl = NULL;
rcCaption = m_PaintManager.GetCaptionRect();
GetClientRect(m_PaintManager.GetPaintWindow(), &rcClient);
pt.x = GET_X_LPARAM(lParam);
pt.y = GET_Y_LPARAM(lParam);
::ScreenToClient(m_PaintManager.GetPaintWindow(), &pt);
if (-1 == rcCaption.bottom) ///< xml中描述bottom为-1时,整个窗口区域都可以拖动
rcCaption.bottom = rcClient.bottom;
if ((pt.x >= rcClient.left)
&& (pt.x < rcClient.right)
&& (pt.y >= rcCaption.top)
&& (pt.y < rcCaption.bottom))
pControl = m_PaintManager.FindControl(pt);
if (IsInStaticControl(pControl))
return HTCAPTION;
return __super::OnNcHitTest(uMsg, wParam, lParam, bHandled);
BOOL CXmlWnd::IsInStaticControl(CControlUI * pControl)
CDuiString strClassName;
std::vector<CDuiString> vctStaticName;
std::vector<CDuiString>::iterator it;
if (NULL == pControl)
return FALSE;
strClassName = pControl->GetClass();
strClassName.MakeLower();
vctStaticName.push_back(L"controlui");
vctStaticName.push_back(L"textui");
vctStaticName.push_back(L"labelui");
vctStaticName.push_back(L"containerui");
vctStaticName.push_back(L"horizontallayoutui");
vctStaticName.push_back(L"verticallayoutui");
vctStaticName.push_back(L"tablayoutui");
vctStaticName.push_back(L"childlayoutui");
vctStaticName.push_back(L"dialoglayoutui");
it = std::find(vctStaticName.begin(), vctStaticName.end(), strClassName);
return (it != vctStaticName.end());
子类
[cpp] view
plaincopyprint?
LRESULT CMainDlg::OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled)
POINT pt;
CDuiString str;
CControlUI * pControl = NULL;
pt.x = GET_X_LPARAM(lParam);
pt.y = GET_Y_LPARAM(lParam);
::ScreenToClient(m_PaintManager.GetPaintWindow(), &pt);
pControl = m_PaintManager.FindControl(pt);
str = pControl->GetName();
/// 如果是要处理点击的界面元素是静态控件, 要单独处理, 返回 HTCLIENT
if ((str == ELEMENT_MAIN_DLG_CTRL_XX)
|| (str == ELEMENT_MAIN_DLG_CTRL_YY))
return HTCLIENT; ///< 要处理的控件不能拖动,可以处理点击
return __super::OnNcHitTest(uMsg, wParam, lParam, bHandled);
参考技术A
选中xml文件,在属性对话框中更改caption的四个值即可,如图:
参考技术B 在基类中做默认的控件拖动判断, 在子类中做特殊控件判断, e.g. 点击某个静态控件时,要弹出一个对话框.[cpp] view
plaincopyprint?
class CMainDlg : public CXmlWnd
class CMainDlg : public CXmlWnd
基类:
[cpp] view
plaincopyprint?
virtual LRESULT OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled);
ate:
BOOL IsInStaticControl(CControlUI * pControl);
virtual LRESULT OnNcHitTest(UINT uMsg, WPARAM wParam, LPARAM lParam, BOOL& bHandled);
private:
BOOL IsInStaticControl(CControlUI * pControl); 参考技术C 直接打开拖动。
Duilib中创建子窗口后,子窗口销毁的问题
自己在学习Duilib时,学习ListDemo中的右键生成菜单,然后当菜单被点击活着失去焦点时,这个菜单窗口如何销毁的问题。
1、在这里,创建菜单时使用了子窗口的概念,创建时,使用了WS_EX_TOOLWINDOW类型,表示是一个工具栏窗口。
2、这个菜单窗口有自己的消息处理函数HandleMessage。第一次的尝试是调用CWindowWnd中的Close函数,查看了这个函数的源码,发现其中是使用PostMessage给自己发送了一个WM_CLOSE的消息,然后在HandleMessage中拦截WM_CLOSE的消息,在消息处理函数中调用了PostQuitMessage(0),此种情况下,菜单窗口是退出了,但是父窗口也是跟着退出了。因为在CPaintManager的MessageLoop中的GetMessage,当收到WM_QUIT的消息时会返回0,此时,MessageLoop中的while循环就会给打破。整个消息循环就会退出,因此整个程序退出
3、第二次的尝试,是直接在菜单项被选中和菜单窗口失去焦点时调用PostQuitMessage(0),由于跟第一次尝试是换汤不换药的动作,因此也已失败告终。
4、由于前两次的失败,然后再网上各种查询也未找到相关的解决方法。后来在查看MSDN时,查看了消息WM_CLOSE的Remark,其中说道“应用程序通过处理WM_CLOSE消息,可以在窗口销毁前对用户做一些提示,如果用户确认提示的话,通过调用DestoryWindow来销魂窗口”。因此,此基础和第一二次尝试的基础上,对代码进行了修改。首先,在菜单项被选中和菜单窗口失去焦点时调用CWindowWnd的Close函数,然后在窗口的消息处理函数中拦截WM_CLOSE消息,在WM_CLOSE的消息处理函数中通过使用DestoryWindow来达到销毁菜单窗口的目的,在调用了DestoryWindow后,再调用delete this来删除窗体本身。这样才达到了效果。
当然,在做完第4步时,也遇到了其他的一些小麻烦,稍作调试即完成。也算对window应用编程又有了点新的体会。
以上是关于如何拖动用duilib创建出来的窗口的主要内容,如果未能解决你的问题,请参考以下文章