如何拖动用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创建出来的窗口的主要内容,如果未能解决你的问题,请参考以下文章

Duilib界面拖动问题

duilib入门问题集

Duilib中创建子窗口后,子窗口销毁的问题

DuiLib的Combo控件点击无响应的问题

Duilib学习日记1

解决Duilib创建的win32窗口拖到屏幕上边缘自动最大化