代码改变世界

精通 VC++ 实效编程280例 - 01 窗口

2013-06-09 14:48  BytesLoop  阅读(1043)  评论(0编辑  收藏  举报

窗口是屏幕上的一个矩形区域。窗口分为3种:重叠窗口、弹出窗口和子窗口。每个窗口都有由系统绘制的“非客户区”和应用程序绘制的“客户区”。在 MFC 中,CWnd 类为各种窗口提供了基类。

1 通过 HWND 获得 CWnd 指针

通过 HWND 获得 Cwnd 指针可以调用 Cwnd::FromHandle 函数。

void CDemoDlg::OnButton1() 
{
	HWND hWnd = GetSafeHwnd();	//获得当前窗口的句柄
	CWnd* pWnd = CWnd::FromHandle(hWnd);	//通过HWND获得CWnd指针
	CString strText = _T("");
	strText.Format("pWnd=0x%X\nthis=0x%X\n",pWnd,this);
	AfxMessageBox(strText);
}

2 获得应用程序主窗口的指针

主窗口指针保存在 CWinThread::m_pMainWnd 中。可以首先调用 AfxGetApp 函数获得应用程序的指针,然后通过应用程序指针获得主窗口的指针。

void CDemoDlg::OnButton2() 
{
	CDemoApp* pApp = (CDemoApp*)AfxGetApp();	//获得应用程序指针
	CWnd* pMainWnd = pApp->m_pMainWnd;	//获得主窗口指针
	CString strText = _T("");
	strText.Format("pMainWnd=0x%X\nthis=0x%X\n",pMainWnd,this);
	AfxMessageBox(strText);
}

3 获得指定点的窗口

获得指定点的窗口可以调用 CWnd::WindowFromPoint 函数。

a 在 CDemoDlg 类中重载 CWnd::PreTranslateMessage 函数。

BOOL CDemoDlg::PreTranslateMessage(MSG* pMsg) 
{
	if (pMsg->message == WM_MOUSEMOVE)
	{
		CPoint point(LOWORD(pMsg->lParam),HIWORD(pMsg->lParam));
		::ClientToScreen(pMsg->hwnd, &point);	//客户区坐标转换为屏幕坐标
		OnMouseMove(0,point);
	}
	return CDialog::PreTranslateMessage(pMsg);
}

b 在 CDemoDlg 类中添加 WM_MOUSEMOVE 消息处理函数。

void CDemoDlg::OnMouseMove(UINT nFlags, CPoint point) 
{
	CWnd* pWnd = WindowFromPoint(point);	//获得指定点的窗口
	if (pWnd != NULL)
	{
		TRACE("%d\n",pWnd);
		if (IsChild(pWnd))
		{
			CString strText = _T("");
			pWnd->GetWindowText(strText);
			SetWindowText(strText);
		}
	}
	CDialog::OnMouseMove(nFlags, point);
}

4 最大化和最小化窗口

最大化和最小化窗口可以调用 CWnd::SendMessage 函数发送最大化或最小化窗口消息。

LRESULT SendMessage( 

UINT message,    //发送的消息,值为 WM_SYSCOMMAND 时表示系统命令消息。 

WPARAM wParam = 0,     //当 message 值为 WM_SYSCMMAND,参数 wParam 值为 SC_MAXIMIZE、SC_MINIMIZE、

LPARAM lParam = 0 );      //SC_RESTORE 时分别表示最大化窗口、最小化窗口、恢复窗口。

void CDemoDlg::OnButton3() 
{
	//最大化窗口
	SendMessage(WM_SYSCOMMAND,SC_MAXIMIZE,0);
}

void CDemoDlg::OnButton4() 
{
	//最小化窗口
	SendMessage(WM_SYSCOMMAND,SC_MINIMIZE,0);
}

void CDemoDlg::OnButton5() 
{
	//恢复窗口
	SendMessage(WM_SYSCOMMAND,SC_RESTORE,0);
}

5 关闭窗口

关闭窗口可以调用 CWnd::SendMessage 函数发送 WM_CLOSE 消息。框架将调用 CWnd::OnClose 函数处理 WM_CLOSE 消息。默认情况下,OnClose 函数将调用 CWnd::DestroyWindow 函数关闭窗口。

void CDemoDlg::OnButton6() 
{
	//关闭窗口
	SendMessage(WM_CLOSE,0,0);
}

void CDemoDlg::OnClose() //添加 WM_CLOSE 消息处理函数
{
	//判断是否关闭
	if (IDYES == MessageBox(_T("是否关闭窗口?"),NULL,MB_YESNO))
	{
		CDialog::OnClose();
	}
}

6 设置窗口的大小和位置

设置窗口大小和位置可通过两种方法:1.调用 CWnd::SetWindowPos 函数;2.调用 CWnd::MoveWindow 函数。

void CDemoDlg::OnButton7() 
{
	//设置窗口的大小和位置
	SetWindowPos(NULL,0,0,320,200,SWP_NOZORDER);
}

void CDemoDlg::OnButton8() 
{
	//设置窗口的大小和位置
	MoveWindow(0,200,200,320);
} 

7 居中显示窗口

使窗口居中显示可以调用 CWnd::CenterWindow 函数。

void CDemoDlg::OnButton9() 
{
	//居中显示窗口
	CenterWindow();
} 

8 顶层显示窗口

使窗口顶层显示,可以调用 CWnd::SetWindowPos 函数,设置对话框窗口的层次为最顶层。

void CDemoDlg::OnButton10() 
{
	//设置窗口层次
	SetWindowPos(&wndTopMost,0,0,0,0,SWP_NOSIZE | SWP_NOMOVE);	//SWP_NOSIZE:表示窗口保持当前的大小,SWP_NOMOVE:表示窗口保持当前的位置
}

9 设置窗口图标

首先调用 CWinApp::LoadIcon 函数加载图标资源,然后调用 CWnd::SetIcon 函数设置图标。

void CDemoDlg::OnButton11() 
{
	//加载图标
	HICON hIcon = AfxGetApp()->LoadIcon(IDI_ICON);
	//设置图标
	SetIcon(hIcon,FALSE);	//FALSE:设置程序小图标,TRUE:设置任务栏大图标
} 

10 获得和设置窗口的标题

获得和设置窗口标题可以分别调用 CWnd::GetWindowText 和 CWnd::SetWindowText 函数。

void CDemoDlg::OnButton12() 
{
	CString strText = _T("");
	GetWindowText(strText);				//获得窗口标题
	SetDlgItemText(IDC_TEXT,strText);	//设置编辑框文本
}

void CDemoDlg::OnButton13() 
{
	CString strText = _T("");
	GetDlgItemText(IDC_TEXT,strText);	//获得编辑框文本
	SetWindowText(strText);				//设置窗口标题
}

11 显示或隐藏窗口的标题栏

显示或隐藏窗口的标题栏可以调用 CWnd::ModifyStyle 函数。

void CDemoDlg::OnButton14() 
{
	//删除标题栏风格
	ModifyStyle(WS_CAPTION,0,SWP_FRAMECHANGED);
}

void CDemoDlg::OnButton15() 
{
	//添加标题栏风格
	ModifyStyle(0,WS_CAPTION,SWP_FRAMECHANGED);	
}

12 改变窗口形状

标准窗口的形状是矩形的。改变窗口的形状首先调用 CRgn 类的成员函数创建相应形状的区域,然后调用 CWnd::SetWindowRgn 函数将其设置为窗口区域。

CRgn 类的 CreateRectRgn、CreateEllipticRgn、CreatePolygonRgn 和 CreateRoundRectRgn 函数可以分别用来创建矩形、椭圆形、多边形和圆矩形区域。 

void CDemoDlg::OnButton16() 
{
	CRect rect;
	GetClientRect(rect);
	//创建矩形区域
	CRgn rgn;
	rgn.CreateRectRgn(rect.left,rect.top,rect.right,rect.bottom);
	//设置窗口的区域
	SetWindowRgn((HRGN)rgn,TRUE);
}

void CDemoDlg::OnButton17() 
{
	CRect rect;
	GetClientRect(rect);
	//创建椭圆形区域
	CRgn rgn;
	rgn.CreateEllipticRgn(0,0,rect.Width(),rect.Height());
	//设置窗口的区域
	SetWindowRgn((HRGN)rgn,TRUE);
}

void CDemoDlg::OnButton18() 
{
	CRect rect;
	GetClientRect(rect);
	CPoint point[6];
	point[0].x = 0;
	point[0].y = rect.Height() / 2;
	point[1].x = rect.Width() / 3;
	point[1].y = 0;
	point[2].x = 2 * rect.Width() / 3;
	point[2].y = 0;
	point[3].x = rect.Width();
	point[3].y = rect.Height() / 2;
	point[4].x = 2 * rect.Width() / 3;
	point[4].y = rect.Height();
	point[5].x = rect.Width() / 3;
	point[5].y = rect.Height();
	//创建多边形区域
	CRgn rgn;
	rgn.CreatePolygonRgn(point,6,ALTERNATE);
	//设置窗口的区域
	SetWindowRgn((HRGN)rgn,TRUE);
}

void CDemoDlg::OnButton19() 
{
	CRect rect;
	GetClientRect(rect);
	//创建圆矩形区域
	CRgn rgn;
	rgn.CreateRoundRectRgn(0,0,rect.Width(),rect.Height(),rect.Width() / 2,rect.Height() / 2);
	//设置窗口的区域
	SetWindowRgn((HRGN)rgn,TRUE);
}

13 设置窗口的透明区域

设置窗口的透明区域,首先调用 CRgn::CreateRectRgn 创建一个区域,然后调用 CRgn::CombineRgn 函数将需要透明的区域去掉,最后调用 CWnd::SetWindowRgn 函数将其设置为窗口区域。

void CDemoDlg::OnButton20() 
{
	CRect rect1;
	GetWindowRect(rect1);
	CRect rect2;
	GetClientRect(rect2);
	ClientToScreen(rect2);
	CRgn rgn1;
	rgn1.CreateRectRgn(rect1.left,rect1.top,rect1.right,rect1.bottom);
	CRgn rgn2;
	rgn2.CreateRectRgn(rect2.left,rect2.top,rect2.right,rect2.bottom);
	CRgn rgn;
	rgn.CreateRectRgn(0,0,1,1);
	rgn.CombineRgn(&rgn1,&rgn2,RGN_DIFF);
	//设置窗口区域
	SetWindowRgn((HRGN)rgn,TRUE);
}

14 透明窗口

实现透明窗口,首先调用 CWnd::ModifyStyleEx 函数,添加窗口的 WS_EX_LAYERED(0x00080000) 扩展风格,然后调用 SDK 的 SetLayeredWindowAttributes 函数设置窗口的透明度和透明色。

void CDemoDlg::OnButton21() 
{
	//添加 WS_EX_LAYERED(0x80000) 扩展风格
	ModifyStyleEx(0,0x80000);
	//加载 User32.DLL 动态链接库
	HMODULE hModule = LoadLibrary("User32.DLL");
	if (hModule != NULL)
	{
		typedef BOOL (WINAPI *FUNC)(HWND,COLORREF,BYTE,DWORD);
		//获得 SetLayeredWindowAttributes 函数指针
		FUNC func = (FUNC)GetProcAddress(hModule,"SetLayeredWindowAttributes");
		if (func != NULL)
		{
			func(GetSafeHwnd(),0,128,2);
		}
		FreeLibrary(hModule);
	}
}

15 窗口闪烁

使窗口闪烁可以调用 CWnd::FlashWindow 函数。

void CDemoDlg::OnButton22() 
{
	//设置定时器
	SetTimer(1,1000,NULL);
}

void CDemoDlg::OnButton23() 
{
	//关闭定时器
	KillTimer(1);
	//窗口返回原始状态
	FlashWindow(FALSE);
}

void CDemoDlg::OnTimer(UINT nIDEvent) 
{
	if (nIDEvent == 1)
	{
		//窗口从一种状态闪烁到另一种状态
		FlashWindow(TRUE);
	}	
	CDialog::OnTimer(nIDEvent);
}

16 图片窗口

实现图片窗口,首先调用 CRgn::CreateRectRgn 和 CRgn::CombineRgn 函数创建并合并多个区域,然后调用 CWnd::SetWindowRgn 函数将其设置为窗口区域。

VC对话框如何添加WM_ERASEBKGND消息(OnEraseBkgnd函数):

http://guohaiyang.blog.163.com/blog/static/3213403720081027104147/

void CDemoDlg::OnButton24() 
{
	CRect rect;
	GetWindowRect(&rect);
	//加载位图
	CBitmap bmp;
	bmp.LoadBitmap(IDB_BITMAP);
	//将位图选入设备环境
	CDC dc;
	CDC *pDC = GetDC();
	dc.CreateCompatibleDC(pDC);
	dc.SelectObject(&bmp);
	//将位图中黑色区域变成透明区域
	CRgn rgn1;
	rgn1.CreateRectRgn(0,0,rect.Width(),rect.Height());
	for (int x = 0;x < rect.Width();x++)
	{
		for (int y = 0;y < rect.Height();y++)
		{
			COLORREF cr = dc.GetPixel(x,y);
			if (cr == RGB(0,0,0))
			{
				CRgn rgn2;
				rgn2.CreateRectRgn(x,y,x+1,y+1);
				rgn1.CombineRgn(&rgn1,&rgn2,RGN_XOR);
			}
		}
	}
	//设置窗口区域
	SetWindowRgn((HRGN)rgn1,TRUE);
	ReleaseDC(pDC);
}
//添加 WM_ERASEBKGND 消息处理函数
BOOL CDemoDlg::OnEraseBkgnd(CDC* pDC) 
{
	CRect rect;
	GetWindowRect(&rect);
	CBitmap bmp;
	bmp.LoadBitmap(IDB_BITMAP);
	CDC dc;
	dc.CreateCompatibleDC(pDC);
	dc.SelectObject(&bmp);
	pDC->BitBlt(0,0,rect.Width(),rect.Height(),&dc,0,0,SRCCOPY);
	return TRUE;
}

17 动画窗口

实现动画窗口,可以调用 SDK 的 AnimateWindow 函数。

BOOL CDemoDlg::OnInitDialog()
{
	CDialog::OnInitDialog();
	// ...
        //窗口居中
	CenterWindow();
	//显示动画窗口
	AnimateWindow(GetSafeHwnd(), 3000, AW_BLEND);
	return TRUE; 
}

18 桌面窗口

获得桌面窗口,可以调用 CWnd::GetDesktopWindow 函数。

void CDemoDlg::OnButton25() 
{
	//获得桌面窗口
	CWnd* pWnd = CWnd::GetDesktopWindow();
	//获得窗口大小
	CRect rect;
	pWnd->GetClientRect(rect);
	CString strText = _T("");
	strText.Format(_T("桌面窗口大小:%dX%d"),rect.Width(),rect.Height());
	AfxMessageBox(strText);
}

19 最小化桌面所有窗口

Window 中可以利用快捷键 Win+M 最小化所有窗口。因此,可以通过向任务栏窗口发送 ID 为 0x1F5(Win+M) 的 WM_HOTKEY 消息,使桌面所有窗口最小化。首先调用 CWnd::FindWindow 函数获得窗口,然后调用 CWnd::SendMessage 函数向窗口发送消息。

void CDemoDlg::OnButton26() 
{
	//获得任务栏窗口
	CWnd* pWnd = CWnd::FindWindow(_T("Shell_TrayWnd"),NULL);
	//发送ID为0x1F5(Win+M)的WM_HOTKEY消息
	pWnd->SendMessage(WM_HOTKEY,0x1F5);
}

20 获取任务栏窗口

获得任务栏窗口,可以调用 CWnd::FindWinow 函数。

void CDemoDlg::OnButton27() 
{
	//获得任务栏窗口
	CWnd* pWnd = CWnd::FindWindow(_T("Shell_TrayWnd"),NULL);
	//获得窗口大小
	CRect rect;
	pWnd->GetClientRect(rect);
	CString strText = _T("");
	strText.Format(_T("任务栏窗口大小:%dX%d"),rect.Width(),rect.Height());
	AfxMessageBox(strText);
}

21 显示或隐藏任务栏

显示或隐藏任务栏,首先调用 CWnd::FindWindow 函数获得窗口,然后调用 CWnd::ShowWindow 函数隐藏或显示窗口。 

void CDemoDlg::OnButton28() 
{
	//获得任务栏窗口
	CWnd* pWnd = CWnd::FindWindow(_T("Shell_TrayWnd"),NULL);
	//隐藏窗口
	if (pWnd->IsWindowVisible())
	{
		pWnd->ShowWindow(SW_HIDE);
	}
}

void CDemoDlg::OnButton29() 
{
	//获得任务栏窗口
	CWnd* pWnd = CWnd::FindWindow(_T("Shell_TrayWnd"),NULL);
	//显示窗口
	if (!pWnd->IsWindowVisible())
	{
		pWnd->ShowWindow(SW_SHOW);
	}
	
}

22 枚举桌面所有顶层窗口

BOOL CDemoDlg::OnInitDialog()
{
	CDialog::OnInitDialog();
	// ...
	//初始化列表框控件
	CListCtrl* pList = (CListCtrl*)GetDlgItem(IDC_LIST);
	pList->ModifyStyle(LVS_ICON | LVS_SMALLICON | LVS_LIST,LVS_REPORT);
	pList->SetExtendedStyle(LVS_EX_GRIDLINES | LVS_EX_FULLROWSELECT);
	pList->InsertColumn(0,_T("窗口类名"),LVCFMT_LEFT,115);
	pList->InsertColumn(1,_T("窗口标题"),LVCFMT_LEFT,150);
	return TRUE;
}

枚举桌面所有顶层窗口有以下两种方法:

a 调用 CWnd::GetDesktopWindow 和 CWnd::GetWindow 函数:首先调用 CWnd::GetDesktopWindow 函数,获得桌面窗口,然后调用 CWnd::GetWindow 函数,枚举所有子窗口。

void CDemoDlg::OnButton30() 
{
	CListCtrl* pList = (CListCtrl*)GetDlgItem(IDC_LIST);
	pList->DeleteAllItems();
	pList->SetRedraw(FALSE);
	//获得桌面窗口
	CWnd* pDesktopWnd = CWnd::GetDesktopWindow();
	//获得第一个子窗口
	CWnd* pWnd = pDesktopWnd->GetWindow(GW_CHILD);
	while(pWnd != NULL)
	{
		int nItem = pList->GetItemCount();
		//获得窗口类名
		CString strClassName = _T("");
		::GetClassName(pWnd->GetSafeHwnd(),strClassName.GetBuffer(256),256);
		strClassName.ReleaseBuffer();
		pList->InsertItem(nItem,strClassName);
		//获得窗口标题
		CString strWindowText = _T("");
		::GetWindowText(pWnd->GetSafeHwnd(),strWindowText.GetBuffer(256),256);
		strWindowText.ReleaseBuffer();
		pList->SetItemText(nItem,1,strWindowText);
		//继续获得下一个子窗口
		pWnd =  pWnd->GetWindow(GW_HWNDNEXT);
	}
	pList->SetRedraw(TRUE);
}

b 调用 SDK 的 EnumWindows 函数。

//添加全局函数
BOOL CALLBACK EnumWndPorc(HWND hWnd,LPARAM lParam)
{
	if (hWnd == NULL)
	{
		return FALSE;
	}
	CListCtrl* pList = (CListCtrl*)lParam;
	int nItem = pList->GetItemCount();
	//获得窗口类名
	CString strClassName = _T("");
	::GetClassName(hWnd,strClassName.GetBuffer(256),256);
	strClassName.ReleaseBuffer();
	pList->InsertItem(nItem,strClassName);
	//获得窗口标题
	CString strWindowText = _T("");
	::GetWindowText(hWnd,strWindowText.GetBuffer(256),256);
	strWindowText.ReleaseBuffer();
	pList->SetItemText(nItem,1,strWindowText);
	return TRUE;
}

void CDemoDlg::OnButton31() 
{
	CListCtrl* pList = (CListCtrl*)GetDlgItem(IDC_LIST);
	pList->DeleteAllItems();
	pList->SetRedraw(FALSE);
	//枚举窗口
	::EnumWindows(EnumWndPorc,(LPARAM)pList);
	pList->SetRedraw(TRUE);
}