首页 > 代码库 > AdjustWindowRect 与 SetWindowPos

AdjustWindowRect 与 SetWindowPos

这两个函数经常一起使用,所以放到一起讲:

1 AdjustWindowRect

    函数功能:该函数依据所需客户矩形的大小,计算需要的窗口矩形的大小。计算出的窗口矩形随后可以传递给CreateWindow函数,用于创建一个客户区所需大小的窗口。

    函数原型:BOOL AdjustWindowRect(LPRECT lpRect ,DWORD dwStyle,BOOL bMENU);

    参数:

    lpRect:指向RECT结构的指针,该结构包含所需客户区域的左上角和右下角的坐标。函数返回时,该结构容纳所需客户区域的窗口的左上角和右下角的坐标。

    dwStyle:指定将被计算尺寸的窗口的窗口风格。

    bMenu:指示窗口是否有菜单。

    返回值:如果函数成功,返回值为非零;如果函数失败,返回值为零。获取错误信息,参看GetLastError。

    备注:客户矩形是指完全包含一个客户区域的最小矩形;窗口矩形是指完全包含一个窗口的最小矩形,该窗口包含客户区与非客户区。

    当一个菜单条下拉出两行或更多行时,AdjustWindowRect函数不增加额外的空间。

 

2 SetWindowPos

    函数功能:该函数改变一个子窗口,弹出式窗口式顶层窗口的尺寸,位置和Z序。子窗口,弹出式窗口,及顶层窗口根据它们在屏幕上出现的顺序排序、顶层窗口设置的级别最高,并且被设置为Z序的第一个窗口。

    函数原型:BOOL SetWindowPos(HWN hWnd,HWND hWndlnsertAfter,int X,int Y,int cx,int cy,UNIT.Flags);

    参数:

    hWnd:窗口句柄。

    hWndlnsertAfter:在z序中的位于被置位的窗口前的窗口句柄。该参数必须为一个窗口句柄,或下列值之一:

    HWND_BOTTOM:将窗口置于Z序的底部。如果参数hWnd标识了一个顶层窗口,则窗口失去顶级位置,并且被置在其他窗口的底部。

    HWND_DOTTOPMOST:将窗口置于所有非顶层窗口之上(即在所有顶层窗口之后)。如果窗口己经是非顶层窗口则该标志不起作用。

    HWND_TOP:将窗口置于Z序的顶部。

    HWND_TOPMOST:将窗口置于所有非顶层窗口之上。即使窗口未被激活窗口也将保持顶级位置。

    查g看该参数的使用方法,请看说明部分。

    x:以客户坐标指定窗口新位置的左边界。

    Y:以客户坐标指定窗口新位置的顶边界。

    cx:以像素指定窗口的新的宽度。

    cy:以像素指定窗口的新的高度。

    uFlags:窗口尺寸和定位的标志。该参数可以是下列值的组合:

    SWP_ASNCWINDOWPOS:如果调用进程不拥有窗口,系统会向拥有窗口的线程发出需求。这就防止调用线程在其他线程处理需求的时候发生死锁。

    SWP_DEFERERASE:防止产生WM_SYNCPAINT消息。

    SWP_DRAWFRAME:在窗口周围画一个边框(定义在窗口类描述中)。

    SWP_FRAMECHANGED:给窗口发送WM_NCCALCSIZE消息,即使窗口尺寸没有改变也会发送该消息。如果未指定这个标志,只有在改变了窗口尺寸时才发送WM_NCCALCSIZE。

    SWP_HIDEWINDOW;隐藏窗口。

    SWP_NOACTIVATE:不激活窗口。如果未设置标志,则窗口被激活,并被设置到其他最高级窗口或非最高级组的顶部(根据参数hWndlnsertAfter设置)。

    SWP_NOCOPYBITS:清除客户区的所有内容。如果未设置该标志,客户区的有效内容被保存并且在窗口尺寸更新和重定位后拷贝回客户区。

    SWP_NOMOVE:维持当前位置(忽略X和Y参数)。

    SWP_NOOWNERZORDER:不改变z序中的所有者窗口的位置。

    SWP_NOREDRAW:不重画改变的内容。如果设置了这个标志,则不发生任何重画动作。适用于客户区和非客户区(包括标题栏和滚动条)和任何由于窗回移动而露出的父窗口的所有部分。如果设置了这个标志,应用程序必须明确地使窗口无效并区重画窗口的任何部分和父窗口需要重画的部分。

    SWP_NOREPOSITION;与SWP_NOOWNERZORDER标志相同。

    SWP_NOSENDCHANGING:防止窗口接收WM_WINDOWPOSCHANGING消息。

    SWP_NOSIZE:维持当前尺寸(忽略cx和Cy参数)。

    SWP_NOZORDER:维持当前Z序(忽略hWndlnsertAfter参数)。

    SWP_SHOWWINDOW:显示窗口。

    返回值:如果函数成功,返回值为非零;如果函数失败,返回值为零。若想获得更多错误消息,请调用GetLastError函数。

    备注:如果设置了SWP_SHOWWINDOW和SWP_HIDEWINDOW标志,则窗口不能被移动和改变大小。如果使用SetWindowLoog改变了窗口的某些数据,则必须调用函数SetWindowPos来作真正的改变。使用下列的组合标志:SWP_NOMOVEISWP_NOSIZEISWP_FRAMECHANGED。

    有两种方法将窗口设为最顶层窗口:一种是将参数hWndlnsertAfter设置为HWND_TOPMOST并确保没有设置SWP_NOZORDER标志;另一种是设置窗口在Z序中的位置以使其在其他存在的窗口之上。当一个窗口被置为最顶层窗口时,属于它的所有窗口均为最顶层窗口,而它的所有者的z序并不改变。

    如果HWND_TOPMOST和HWND_NOTOPMOST标志均未指定,即应用程序要求窗口在激活的同时改变其在Z序中的位置时,在参数hWndinsertAfter中指定的值只有在下列条件中才使用:

    在hWndlnsertAfter参数中没有设定HWND_NOTOPMOST和HWND_TOPMOST标志。

    由hWnd参数标识的窗口不是激活窗口。

    如果未将一个非激活窗口设定到z序的顶端,应用程序不能激活该窗口。应用程序可以无任何限制地改变被激活窗口在Z序中的位置,或激活一个窗口并将其移到最高级窗口的顶部或非最高级窗口的顶部。

    如果一个顶层窗口被重定位到z序的底部(HWND_BOTTOM)或在任何非最高序的窗口之后,该窗口就不再是最顶层窗口。当一个最顶层窗口被置为非最顶级,则它的所有者窗口和所属者窗口均为非最顶层窗口。

    一个非最顶端窗口可以拥有一个最顶端窗口,但反之则不可以。任何属于顶层窗口的窗口(例如一个对话框)本身就被置为顶层窗口,以确保所有被属窗口都在它们的所有者之上。

    如果应用程序不在前台,但应该位于前台,就应调用SetForegroundWindow函数来设置。

    Windows CE:如果这是一个可见的顶层窗口,并且未指定SWP_NOACTIVATE标志,则这个函数将激活窗口、如果这是当前的激活窗口,并且指定了SWP_NOACTIVATE或SWP_HIDEWINDOW标志,则激活另外一个可见的顶层窗口。

    当在这个函数中的nFlags参数里指定了SWP_FRAMECHANGED标志时,WindowsCE重画窗口的整个非客户区,这可能会改变客户区的大小。这也是重新计算客户区的唯一途径,也是通过调用SetwindowLong函数改变窗口风格后通常使用的方法。

    SetWindowPos将使WM_WINDOWPOSCHANGED消息向窗口发送,在这个消息中传递的标志与传递给函数的相同。这个函数不传递其他消息。

    Windows CE 1.0不支持在hWndlnsertAber参数中的HWND_TOPMOST和HWND_NOTOPMOST常量。

    Windows CE1.0不支持在fuFags参数中的SWP_DRAWFRAME和SWP_NOCOPYBITS标志。

 

例如,下面创建的窗口位于最顶层,且指定客户区大小

 

 

  1 #include <windows.h> 
  2 
  3 
  4 HINSTANCE g_hIns; 
  5 HWND g_hWnd; 
  6 
  7 
  8 LRESULT CALLBACK WndProc(HWND,UINT,WPARAM,LPARAM); 
  9 
 10 
 11 int WINAPI WinMain(HINSTANCE hInstance,HINSTANCE hPrevInstance,LPSTR lpCmdLine,int iCmdShow) 
 12 { 
 13     g_hIns=hInstance; 
 14 
 15     static TCHAR szAppName[]=TEXT("HelloWin"); 
 16     HWND hWnd; 
 17     MSG msg; 
 18     WNDCLASS wndClass; 
 19 
 20     wndClass.style=CS_HREDRAW|CS_VREDRAW; 
 21     wndClass.lpfnWndProc=WndProc; 
 22     wndClass.cbClsExtra=0; 
 23     wndClass.cbWndExtra=0; 
 24     wndClass.hInstance=hInstance; 
 25     wndClass.hIcon=LoadIcon(NULL,IDI_APPLICATION); 
 26     wndClass.hCursor=LoadCursor(NULL,IDC_ARROW); 
 27     wndClass.hbrBackground=(HBRUSH)GetStockObject(WHITE_BRUSH); 
 28     wndClass.lpszMenuName=NULL; 
 29     wndClass.lpszClassName=szAppName; 
 30 
 31     if(!RegisterClass(&wndClass)) 
 32     { 
 33         MessageBox(NULL,TEXT("error"),szAppName,MB_ICONERROR|MB_OK); 
 34         return 0; 
 35     } 
 36 
 37     hWnd=CreateWindow(szAppName,TEXT("The hello program"),WS_OVERLAPPEDWINDOW, 
 38         CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,CW_USEDEFAULT,NULL,NULL,hInstance,NULL); 
 39 
 40      
 41 
 42     int iReturn; 
 43 
 44     //也可以放在此 
 45 
 46     ShowWindow(hWnd,iCmdShow); 
 47 
 48     UpdateWindow(hWnd); 
 49 
 50 
 51 
 52 
 53     while(GetMessage(&msg,hWnd,0,0)) 
 54     { 
 55         TranslateMessage(&msg); 
 56         DispatchMessage(&msg); 
 57     } 
 58 
 59     return msg.wParam; 
 60 } 
 61 
 62 LRESULT CALLBACK WndProc(HWND hWnd,UINT message,WPARAM wParam,LPARAM lParam) 
 63 { 
 64     static HDC hdc; 
 65     static HDC hBitmapDC; 
 66 
 67     PAINTSTRUCT ps; 
 68 
 69     int iReturn; 
 70 
 71     static RECT rect = {0,0,400,200}; 
 72 
 73     switch (message) 
 74     { 
 75     case WM_CREATE: 
 76         iReturn = AdjustWindowRect(&rect,WS_OVERLAPPEDWINDOW,true); 
 77 
 78         SetWindowPos(hWnd,HWND_TOPMOST,100,100,rect.right-rect.left,rect.bottom-rect.top, SWP_NOMOVE);//顶层 
 79          
 80         break; 
 81 
 82     case   WM_PAINT : 
 83         hdc=BeginPaint(hWnd,&ps); 
 84 
 85         EndPaint (hWnd, &ps) ; 
 86 
 87         break; 
 88 
 89     case WM_DESTROY : 
 90         PostQuitMessage (0) ; 
 91 
 92         break ; 
 93 
 94     default: 
 95         return DefWindowProc (hWnd, message, wParam, lParam) ; 
 96 
 97     } 
 98 
 99     return 1; 
100 
101 }