标签(空格分隔):【VS开发】
声明:引用请注明出处http://blog.csdn.net/lg1259156776/
说明:最近突发奇想想完成两件新的人机交互工程,一个是基于眼动仪的眼动控制,另一个是基于camera的手动控制,最直观的方法就是使用slider滑块,眼睛或手左右动,slider也左右动,眼睛或手上下动,slider也上下动,并配合着对话框的背景颜色的改变,来增强互动的效果,本文所记录的就是其中关于动态修改控件颜色的方法。
首先说基于眼动仪的眼动控制,由于提供的matlab接口开发起来比较方便,所以采用了matlab gui设计,而在matlab中关于dialog或者某个控件的颜色的修改简直太方便了,如下一行代码就可以搞定:
set(handles.axes1,'color',[0.5,0.5,0.5])
所以matlab gui设置各种事情是比较方便的。毕竟是解释性开发语言。
由于好久没有在VS2010下开发对话框的应用,以前都是在VC下开发dialog程序,在VS2010上开发的多是具有office风格的文档应用,但是曾经积累的经验是非常有帮助的。就不多说如何按照经验一步一步的尝试了,直接给出最终的较为简单的解决方案:
通过类向导,或者手动添加消息:WM_CTLCOLOR,其消息响应函数为:
afx_msg HBRUSH OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
在每个控件开始绘制之前,都会向其父窗口发送WM_CTLCOLOR通告消息,在该消息的处理函数中,可以设置控件显示文本的前景色、背景色以及字体。该消息处理函数还要求返回一个画刷的句柄,用于在控件具体的绘制之前擦除其客户区。当窗口重绘时,也会重新绘制每个控件,从而分别调用该函数,这就给了动态修改控件相关颜色特性的机会。
比如在对应的控件下的OnCtrColor函数中写入:
pDC->SetTextColor(RGB(255, 0, 0)); //设置文本前景色
pDC->SetBkColor(RGB(255, 255, 255)); //设置文本背景色
pDC->SetBkMode(TRANSPARENT); //TRANSPARENT或OPAQUE
pDC->SelectObject(...)
就可以实现修改某个控件的绘制属性。具体的实现可以参考下面的一段代码:
//
//m_font1与m_font2为CTestDlg的成员,类型为CFont
//
BOOL CTestDlg::OnInitDialog()
{
......
// TODO: Add extra initialization here
m_font1.CreatePointFont(120, TEXT("Impact"));
m_font2.CreatePointFont(120, TEXT("Arial"));
......
}
HBRUSH CTestDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
{
HBRUSH hbr = CDialog::OnCtlColor(pDC, pWnd, nCtlColor);
// TODO: Change any attributes of the DC here
if (nCtlColor == CTLCOLOR_STATIC)
{
switch (pWnd->GetDlgCtrlID())
{
case IDC_STATIC_1:
pDC->SetTextColor(RGB(255, 0, 0));
pDC->SetBkColor(RGB(255, 255, 255));
pDC->SetBkMode(TRANSPARENT);
pDC->SelectObject(&m_font1);
return (HBRUSH)::GetStockObject(BLACK_BRUSH);
break;
case IDC_STATIC_2:
pDC->SetTextColor(RGB(255, 255, 0));
pDC->SetBkColor(RGB(255, 255, 255));
pDC->SelectObject(&m_font2);
return (HBRUSH)::GetStockObject(BLACK_BRUSH);
break;
default:
break;
}
}
// TODO: Return a different brush if the default is not desired
return hbr;
}
当然如果是修改dialog的属性,可以直接在最后的return上返回一个画刷,填充dialog的背景颜色。
上面这种方法只是一种静态的修改,因为所有的属性都是一次性设定好了,似乎没有根据情况进行改变的可能。这个是时候就要用到上面所提到的一种方法:强迫窗口重绘,可用的函数有Invalidate()和UpdateWindow(),两者的区别如下:
Invalidate在消息队列中加入一条WM_PAINT消息,其无效区为整个客户区。而UpdateWindow直接发送一个WM_PAINT消息,其无效区范围就是消息队列中WM_PAINT消息(最多只有一条)的无效区。效果很明显,调用Invalidate之后,屏幕不一定马上更新,因为WM_PAINT消息不一定在队列头部,而调用UpdateWindow会使WM_PAINT消息马上执行的,绕过了消息队列。如果你调用Invalidate之后想马上更新屏幕,那就加上UpdateWindow()这条语句。
那么剩下的事情就比较简单了,可以通过设置一个COLORREF m_BrushColor;的成员变量,在调用窗口重绘的函数之前,修改m_BrushColor,然后在OnCtlColor函数中将画刷的颜色创建为该m_BrushColor:
m_bkBrush.DeleteObject();
m_bkBrush.CreateSolidBrush(m_BrushColor); //创建一把黄色的背景刷子
下面是我所修改的函数:
HBRUSH ChandControllerDlg::OnCtlColor(CDC* pDC, CWnd* pWnd, UINT nCtlColor)
{
HBRUSH hbr = CDialogEx::OnCtlColor(pDC, pWnd, nCtlColor);
// TODO: 在此更改 DC 的任何特性
if((CTLCOLOR_SCROLLBAR)&&(pWnd->GetDlgCtrlID()==IDC_SLIDER1 || pWnd->GetDlgCtrlID()==IDC_SLIDER2))
{
//此处设置背景的颜色
m_bkBrush.DeleteObject();
m_bkBrush.CreateSolidBrush(RGB(0,255,0)); //创建一把黄色的背景刷子
return m_bkBrush;
}
if((CTLCOLOR_BTN)&&(pWnd->GetDlgCtrlID()==IDOK))
{
m_bkBrush.DeleteObject();
m_bkBrush.CreateSolidBrush(RGB(0,255,0)); //创建一把黄色的背景刷子
return m_bkBrush;
}
m_bkBrush.DeleteObject();
m_bkBrush.CreateSolidBrush(m_BrushColor); //创建一把黄色的背景刷子
// TODO: 如果默认的不是所需画笔,则返回另一个画笔
return m_bkBrush;
}
通过上面的函数,可以实现对对话框中的控件或者对话框的背景颜色进行动态修改。
2015-10-31 调试记录 张朋艺