MoveWindow() SetWindowPos()的区别于联系

  敲代码时,突然发现有一个背景图片无法显示,百思不得其解,最终发现是MoveWindow() SetWindowPos()这两个函数的使用不当造成的。

  这里把这两个函数的前世今生给分析一下。

  先看MoveWindow()函数,用于改变窗口的位置和尺寸,对于顶层窗口,按照屏幕对齐与左上角,对于子窗口,则对齐与父窗口的左上角。

BOOL MoveWindow( HWND hWnd,int x, int y, int nWidth, int nHeight,BOOL bRepaint = TRUE);

  第一个表示窗口句柄;

  参数x,y表示窗口的左上角起点;

  参数nwidth,nHeight 表示窗口高度和宽度;

  最后一个bRepaint表示是否立即重绘。为true时系统会立即发送WM_PAINT到窗口程序(会调用UpdateWindow()函数),为false时不会发生任何类型的重绘操作。

  MoveWindow会给窗口发送WM_WINDOWPOSCHANGING,WM_WINDOWPOSCHANGED,WM_MOVE,WM_SIZE和WM_NCCALCSIZE消息。

  接下来看SetWindowPos(),这个函数同样用来实现窗口移动操作,只不过功能更多一点。

BOOL SetWindowPos(HWND hWnd,const CWnd* pWndInsertAfter, int x, int y,int cx, int cy, UINT nFlags);

  第一个参数是窗口句柄;

  第二个参数是窗口的Z order,就是有多个窗口时,本窗口显示的顺序,最上面或者最下面,可选参数有HWND_BOTTOM,HWND_NOTOPMOST,HWND_TOP,HWND_TOPMOST.

  HWND_BOTTOM:将窗口置于Z序的底部。如果参数hWnd标识了一个顶层窗口,则窗口失去顶级位置,并且被置在其他窗口的底部。 
   HWND_NOTOPMOST:将窗口置于所有非顶层窗口之上(即在所有顶层窗口之后)。如果窗口已经是非顶层窗口则该标志不起作用。 
   HWND_TOP:将窗口置于Z序的顶部。 
   HWND_TOPMOST:将窗口置于所有非顶层窗口之上。即使窗口未被激活窗口也将保持顶级位置。

  接下来四个参数是窗口的位置和尺寸。

  最后一个参数nFlags是窗口显示方式的标志,这个参数很多,可以组合使用,详细如下:

    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:显示窗口。

  注意事项:

  使用SetWindowPos()如果设置了SWP_SHOWWINDOWS或者SWP_HIDEWINDOW,那么窗口将不能被移动和改变大小,我使用时就是设置了SWP_SHOWWINDOW,从而导致不能重绘背景。

  两者的比较:

  两者都能实现移动和改变窗口大小。

  MoveWindow()功能比较单一,只能移动和改变窗口大小,而SetWindowPos()可以设置更多的参数,实现Zorder及显示方式标志。

  MoveWindow()发送WM_WINDOWPOSCHANGING,WM_WINDOWPOSCHANGED,WM_MOVE,WM_SIZE和WM_NCCALCSIZE等消息到窗口SetWindowPos()只发送WM_WINDOWPOSCHANGED消息到窗口。所以说如果需要发送更多消息时还是使用MoveWindow()的好。

时间: 2024-10-28 14:35:28

MoveWindow() SetWindowPos()的区别于联系的相关文章

VC/MFC 编程技巧大总结

1 toolbar默认位图左上角那个点的颜色是透明色,不喜欢的话可以自己改. 2 VC++中 WM_QUERYENDSESSION WM_ENDSESSION 为系统关机消息. 3 Java学习书推荐:<java编程思想> 4 在VC下执行DOS命令 a. system("md c:\\12"); b. WinExec("Cmd.exe /C md c:\\12", SW_HIDE); c. ShellExecute ShellExecute(NULL,

Windows API-----top level window

原文地址: http://blog.163.com/cumt_xl/blog/static/19071504420136911838683/ Q: What is a top-level window? A: A top-level window is a window that is not child, i.e. it has not WS_CHILD style set. Notes unlike the child windows, a top-level window can be d

MoveWindow and SetWindowPos

转自:http://blog.sina.com.cn/s/blog_82c346de0100u7kq.html MoveWindow and SetWindowPos (2011-09-14 15:56:51) 转载▼ 标签: 杂谈 分类: MFC MoveWindow仅仅能设置窗体的大小和位置:SetWindowPos拥有MoveWindow的所有功能之外.还能够设置窗体的层叠关系(如.把指定的窗体放在所有窗体的最上层--always   on   top就能够用这个函数.或者把指定的窗体放在

MoveWindow和SetWindowPos

1.MoveWindow()用来改变窗口位置或大小.void MoveWindow(int x, int y, int nWidth, int nHeight, BOOL bRepaint = TRUE);void MoveWindow(LPCRECT lpRect, BOOL bRepaint = TRUE); 2.SetWindowPos不仅可以用来改变窗口位置和大小,还可以设置窗口Z序特性.BOOL SetWindowPos( const CWnd* pWndInsertAfter, in

SetWindowPos函数详解

WinAPI: SetWindowPos - 改变窗口的位置与状态 SetWindowPos(hWnd: HWND; {窗口句柄}hWndInsertAfter: HWND; {窗口的 Z 顺序}X, Y: Integer; {位置}cx, cy: Integer; {大小}uFlags: UINT {选项}): BOOL; //hWndInsertAfter 参数可选值:HWND_TOP = 0; {在前面}HWND_BOTTOM = 1; {在后面}HWND_TOPMOST = HWND(-

sendmessage和postmessage的区别

sendmessage和postmessage的区别PostMessage只负责将消息放到消息队列中,不确定何时及是否处理    SendMessage要等到受到消息处理的返回码(DWord类型)后才继续    PostMessage执行后马上返回    SendMessage必须等到消息被处理后才会返回. SendMessage 和 PostMessage 的区别 1.首先是返回值意义的区别,我们先看一下 MSDN 里的声明: LRESULT SendMessage(        HWND 

vc 使用SetWindowPos改变窗体的大小,实现折叠,展开

void CWork::DoDataExchange(CDataExchange* pDX) { CDialog::DoDataExchange(pDX); SetWindowPos(NULL,0,0,900,460,SWP_NOMOVE | SWP_NOZORDER); 上面是初始化窗体的大小. 增加事件响应来 实现窗体的折叠,展开(改变大小) //折叠窗体 void CWork::Onkuozhan() { // TODO: Add your control notification han

除了创建时指定窗口位置之外,还有3种移动窗口位置的办法(移动的同时往往可以改变窗口大小)(SetWindowPos最有用,它有许多标志位)

首先,在创立窗口对象的时候,CreateWindowEx就可以指定窗口的位置.除此之外,还有三种方法可以改变窗口的位置: procedure TWinControl.CreateWindowHandle(const Params: TCreateParams); begin // 根据之前准备的Params参数使用API创建窗口.其10个参数都是Params的参数,0表示Menu,WindowClass的十项内容只用到了hInstance一项 // important 控件移到正确的显示位置,就

PostMessage和SendMessage有什么区别?(有EnumChildWindowsProc的例子)

PostMessage只是把消息放入队列,不管其他程序是否处理都返回,然后继续执行;而SendMessage必须等待其他程序处理消息后才返回,继续执行.PostMessage的返回值表示PostMessage函数执行是否正确;而SendMessage的返回值表示其他程序处理消息后的返回值. 字号: 大大  中中  小小 教你:Delphi获取句柄sendmessage 查找另外一个窗口的句柄: handle := FindWindow(nil,PChar('窗口的标题'));//查到窗体句柄查找