1 /* $Id: defwnd.c,v 1.80 2003/09/07 11:52:54 weiden Exp $
3 * COPYRIGHT: See COPYING in the top level directory
4 * PROJECT: ReactOS user32.dll
5 * FILE: lib/user32/windows/window.c
6 * PURPOSE: Window management
7 * PROGRAMMER: Casper S. Hornstrup (chorns@users.sourceforge.net)
9 * 06-06-2001 CSH Created
12 /* INCLUDES ******************************************************************/
17 #include <user32/wininternal.h>
26 /* GLOBALS *******************************************************************/
28 static HBITMAP hbSysMenu
;
29 /* TODO: widgets will be cached here.
30 static HBITMAP hbClose;
31 static HBITMAP hbCloseD;
32 static HBITMAP hbMinimize;
33 static HBITMAP hbMinimizeD;
34 static HBITMAP hbRestore;
35 static HBITMAP hbRestoreD;
36 static HBITMAP hbMaximize;
37 static HBITMAP hbScrUp;
38 static HBITMAP hbScrDwn;
39 static HBITMAP hbScrLeft;
40 static HBITMAP hbScrRight;
44 static COLORREF SysColours
[29] =
46 RGB(224, 224, 224) /* COLOR_SCROLLBAR */,
47 RGB(58, 110, 165) /* COLOR_BACKGROUND */,
48 RGB(0, 0, 128) /* COLOR_ACTIVECAPTION */,
49 RGB(128, 128, 128) /* COLOR_INACTIVECAPTION */,
50 RGB(192, 192, 192) /* COLOR_MENU */,
51 RGB(192, 192, 192) /* COLOR_WINDOW */,
52 RGB(192, 192, 192) /* COLOR_WINDOWFRAME */,
53 RGB(0, 0, 0) /* COLOR_MENUTEXT */,
54 RGB(0, 0, 0) /* COLOR_WINDOWTEXT */,
55 RGB(255, 255, 255) /* COLOR_CAPTIONTEXT */,
56 RGB(128, 128, 128) /* COLOR_ACTIVEBORDER */,
57 RGB(255, 255, 255) /* COLOR_INACTIVEBORDER */,
58 RGB(255, 255, 232) /* COLOR_APPWORKSPACE */,
59 RGB(224, 224, 224) /* COLOR_HILIGHT */,
60 RGB(0, 0, 128) /* COLOR_HILIGHTTEXT */,
61 RGB(192, 192, 192) /* COLOR_BTNFACE */,
62 RGB(128, 128, 128) /* COLOR_BTNSHADOW */,
63 RGB(192, 192, 192) /* COLOR_GRAYTEXT */,
64 RGB(0, 0, 0) /* COLOR_BTNTEXT */,
65 RGB(192, 192, 192) /* COLOR_INACTIVECAPTIONTEXT */,
66 RGB(255, 255, 255) /* COLOR_BTNHILIGHT */,
67 RGB(32, 32, 32) /* COLOR_3DDKSHADOW */,
68 RGB(192, 192, 192) /* COLOR_3DLIGHT */,
69 RGB(0, 0, 0) /* COLOR_INFOTEXT */,
70 RGB(255, 255, 192) /* COLOR_INFOBK */,
71 RGB(184, 180, 184) /* COLOR_ALTERNATEBTNFACE */,
72 RGB(0, 0, 255) /* COLOR_HOTLIGHT */,
73 RGB(16, 132, 208) /* COLOR_GRADIENTACTIVECAPTION */,
74 RGB(181, 181, 181) /* COLOR_GRADIENTINACTIVECAPTION */,
77 static ATOM AtomInternalPos
;
79 /* Bits in the dwKeyData */
80 #define KEYDATA_ALT 0x2000
82 /* FUNCTIONS *****************************************************************/
85 IsMaxBoxActive(HWND hWnd
)
87 ULONG uStyle
= GetWindowLongW(hWnd
, GWL_STYLE
);
88 return (uStyle
& WS_MAXIMIZEBOX
);
92 IsCloseBoxActive(HWND hWnd
)
94 ULONG uStyle
= GetWindowLongW(hWnd
, GWL_STYLE
);
95 return (uStyle
& WS_SYSMENU
);
99 IsMinBoxActive(HWND hWnd
)
101 ULONG uStyle
= GetWindowLongW(hWnd
, GWL_STYLE
);
102 return (uStyle
& WS_MINIMIZEBOX
);
106 UIGetFrameSizeX(HWND hWnd
)
108 ULONG uStyle
= GetWindowLongW(hWnd
, GWL_STYLE
);
110 if ( uStyle
& WS_THICKFRAME
)
111 return GetSystemMetrics(SM_CXSIZEFRAME
);
113 return GetSystemMetrics(SM_CXFRAME
);
117 UIGetFrameSizeY(HWND hWnd
)
119 ULONG uStyle
= GetWindowLongW(hWnd
, GWL_STYLE
);
121 if (uStyle
& WS_THICKFRAME
)
122 return GetSystemMetrics(SM_CYSIZEFRAME
);
124 return GetSystemMetrics(SM_CYFRAME
);
128 UserSetupInternalPos(VOID
)
131 AtomInternalPos
= GlobalAddAtomA(Str
);
138 GetSysColor(int nIndex
)
140 return SysColours
[nIndex
];
147 GetSysColorPen(int nIndex
)
149 return CreatePen(PS_SOLID
, 1, SysColours
[nIndex
]);
156 GetSysColorBrush(int nIndex
)
158 return CreateSolidBrush(SysColours
[nIndex
]);
165 DefFrameProcA( HWND hWnd
,
179 DefFrameProcW(HWND hWnd
,
190 UserGetInternalPos(HWND hWnd
)
193 lpPos
= (PINTERNALPOS
)GetPropA(hWnd
, (LPSTR
)(DWORD
)AtomInternalPos
);
198 DefWndRedrawIconTitle(HWND hWnd
)
200 PINTERNALPOS lpPos
= (PINTERNALPOS
)GetPropA(hWnd
, (LPSTR
)(DWORD
)AtomInternalPos
);
204 if (lpPos
->IconTitle
!= NULL
)
206 SendMessageA(lpPos
->IconTitle
, WM_SHOWWINDOW
, TRUE
, 0);
207 InvalidateRect(lpPos
->IconTitle
, NULL
, TRUE
);
215 UserHasMenu(HWND hWnd
, ULONG Style
)
217 return (!(Style
& WS_CHILD
) && GetMenu(hWnd
) != 0);
221 UserHasAnyFrameStyle(ULONG Style
, ULONG ExStyle
)
223 return ((Style
& (WS_THICKFRAME
| WS_DLGFRAME
| WS_BORDER
)) ||
224 (ExStyle
& WS_EX_DLGMODALFRAME
) ||
225 (!(Style
& (WS_CHILD
| WS_POPUP
))));
229 UserHasDlgFrameStyle(ULONG Style
, ULONG ExStyle
)
231 return ((ExStyle
& WS_EX_DLGMODALFRAME
) ||
232 ((Style
& WS_DLGFRAME
) && (!(Style
& WS_THICKFRAME
))));
236 UserHasThickFrameStyle(ULONG Style
, ULONG ExStyle
)
238 return ((Style
& WS_THICKFRAME
) &&
239 (!((Style
& (WS_DLGFRAME
| WS_BORDER
)) == WS_DLGFRAME
)));
243 UserHasThinFrameStyle(ULONG Style
, ULONG ExStyle
)
245 return ((Style
& WS_BORDER
) || (!(Style
& (WS_CHILD
| WS_POPUP
))));
249 UserHasBigFrameStyle(ULONG Style
, ULONG ExStyle
)
251 return ((Style
& (WS_THICKFRAME
| WS_DLGFRAME
)) ||
252 (ExStyle
& WS_EX_DLGMODALFRAME
));
256 UserGetInsideRectNC(HWND hWnd
, RECT
*rect
)
262 Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
263 ExStyle
= GetWindowLongW(hWnd
, GWL_EXSTYLE
);
264 GetWindowRect(hWnd
, &WindowRect
);
265 rect
->top
= rect
->left
= 0;
266 rect
->right
= WindowRect
.right
- WindowRect
.left
;
267 rect
->bottom
= WindowRect
.bottom
- WindowRect
.top
;
269 if (Style
& WS_ICONIC
)
274 /* Remove frame from rectangle */
275 if (UserHasThickFrameStyle(Style
, ExStyle
))
277 InflateRect(rect
, -GetSystemMetrics(SM_CXFRAME
),
278 -GetSystemMetrics(SM_CYFRAME
));
282 if (UserHasDlgFrameStyle(Style
, ExStyle
))
284 InflateRect(rect
, -GetSystemMetrics(SM_CXDLGFRAME
),
285 -GetSystemMetrics(SM_CYDLGFRAME
));
286 /* FIXME: this isn't in NC_AdjustRect? why not? */
287 if (ExStyle
& WS_EX_DLGMODALFRAME
)
288 InflateRect( rect
, -1, 0 );
292 if (UserHasThinFrameStyle(Style
, ExStyle
))
294 InflateRect(rect
, -GetSystemMetrics(SM_CXBORDER
),
295 -GetSystemMetrics(SM_CYBORDER
));
302 UserDrawSysMenuButton(HWND hWnd
, HDC hDC
, LPRECT Rect
, BOOL down
)
305 HBITMAP hSavedBitmap
;
309 hbSysMenu
= (HBITMAP
)LoadBitmapW(0, MAKEINTRESOURCEW(OBM_CLOSE
));
311 hDcMem
= CreateCompatibleDC(hDC
);
316 hSavedBitmap
= SelectObject(hDcMem
, hbSysMenu
);
323 BitBlt(hDC
, Rect
->left
+ 2, Rect
->top
+ 3, 16, 14, hDcMem
,
324 (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
) ?
325 GetSystemMetrics(SM_CXSIZE
): 0, 0, SRCCOPY
);
327 SelectObject(hDcMem
, hSavedBitmap
);
334 * Cache bitmaps, then just bitblt instead of calling DFC() (and
335 * wasting precious CPU cycles) every time
338 UserDrawCaptionButton(HWND hWnd
, HDC hDC
, BOOL bDown
, ULONG Type
)
341 INT iBmpWidth
= GetSystemMetrics(SM_CXSIZE
) - 2;
342 INT iBmpHeight
= GetSystemMetrics(SM_CYSIZE
) - 4;
343 INT OffsetX
= UIGetFrameSizeX(hWnd
);
344 INT OffsetY
= UIGetFrameSizeY(hWnd
);
346 if (!(GetWindowLongW(hWnd
, GWL_STYLE
) & WS_SYSMENU
))
351 GetWindowRect(hWnd
, &rect
);
353 rect
.right
= rect
.right
- rect
.left
;
354 rect
.bottom
= rect
.bottom
- rect
.top
;
355 rect
.left
= rect
.top
= 0;
359 case DFCS_CAPTIONMIN
:
361 if ((GetWindowLongW(hWnd
, GWL_EXSTYLE
) & WS_EX_TOOLWINDOW
) == TRUE
)
362 return; /* ToolWindows don't have min/max buttons */
364 SetRect(&rect
, rect
.right
- OffsetX
- (iBmpWidth
* 3) - 5,
365 OffsetY
+ 2, rect
.right
- (iBmpWidth
* 2) - OffsetX
- 5,
366 rect
.top
+ iBmpHeight
+ OffsetY
+ 2);
367 DrawFrameControl(hDC
, &rect
, DFC_CAPTION
,
368 DFCS_CAPTIONMIN
| (bDown
? DFCS_PUSHED
: 0) |
369 (IsMinBoxActive(hWnd
) ? 0 : DFCS_INACTIVE
));
372 case DFCS_CAPTIONMAX
:
374 if ((GetWindowLongW(hWnd
, GWL_EXSTYLE
) & WS_EX_TOOLWINDOW
) == TRUE
)
375 return; /* ToolWindows don't have min/max buttons */
377 SetRect(&rect
, rect
.right
- OffsetX
- (iBmpWidth
* 2) - 5,
378 OffsetY
+ 2, rect
.right
- iBmpWidth
- OffsetX
- 5,
379 rect
.top
+ iBmpHeight
+ OffsetY
+ 2);
380 DrawFrameControl(hDC
, &rect
, DFC_CAPTION
,
381 (IsZoomed(hWnd
) ? DFCS_CAPTIONRESTORE
: DFCS_CAPTIONMAX
) |
382 (bDown
? DFCS_PUSHED
: 0) |
383 (IsMaxBoxActive(hWnd
) ? 0 : DFCS_INACTIVE
));
386 case DFCS_CAPTIONCLOSE
:
388 SetRect(&rect
, rect
.right
- OffsetX
- iBmpWidth
- 3,
389 OffsetY
+ 2, rect
.right
- OffsetX
- 3,
390 rect
.top
+ iBmpHeight
+ OffsetY
+ 2 );
391 DrawFrameControl(hDC
, &rect
, DFC_CAPTION
,
392 (DFCS_CAPTIONCLOSE
| (bDown
? DFCS_PUSHED
: 0) |
393 (IsCloseBoxActive(hWnd
) ? 0 : DFCS_INACTIVE
)));
399 // Enabling this will cause captions to draw smoother, but slower:
400 // #define DOUBLE_BUFFER_CAPTION
401 // NOTE: Double buffering appears to be broken for this at the moment
413 NONCLIENTMETRICSW nclm
;
416 UINT VCenter
= 0, Padding
= 0;
419 HFONT hOldFont
= NULL
;
420 HBRUSH OldBrush
= NULL
;
423 #ifdef DOUBLE_BUFFER_CAPTION
424 HBITMAP MemBMP
= NULL
, OldBMP
= NULL
;
426 MemDC
= CreateCompatibleDC(hDC
);
427 if (! MemDC
) goto cleanup
;
428 MemBMP
= CreateCompatibleBitmap(hDC
, lprc
->right
- lprc
->left
, lprc
->bottom
- lprc
->top
);
429 if (! MemBMP
) goto cleanup
;
430 OldBMP
= SelectObject(MemDC
, MemBMP
);
431 if (! OldBMP
) goto cleanup
;
435 OffsetViewportOrgEx(MemDC
, lprc
->left
, lprc
->top
, NULL
);
438 // If DC_GRADIENT is specified, a Win 98/2000 style caption gradient should
439 // be painted. For now, that flag is ignored:
440 // Windows 98/Me, Windows 2000/XP: When this flag is set, the function uses
441 // COLOR_GRADIENTACTIVECAPTION (if the DC_ACTIVE flag was set) or
442 // COLOR_GRADIENTINACTIVECAPTION for the title-bar color.
444 // Draw the caption background
445 if (uFlags
& DC_INBUTTON
)
447 OldBrush
= SelectObject(MemDC
, GetSysColorBrush(uFlags
& DC_ACTIVE
? COLOR_BTNFACE
: COLOR_BTNSHADOW
) );
448 if (! OldBrush
) goto cleanup
;
449 if (! PatBlt(MemDC
, 0, 0, lprc
->right
- lprc
->left
, lprc
->bottom
- lprc
->top
, PATCOPY
)) goto cleanup
;
453 // DC_GRADIENT check should go here somewhere
454 OldBrush
= SelectObject(MemDC
, GetSysColorBrush(uFlags
& DC_ACTIVE
? COLOR_ACTIVECAPTION
: COLOR_INACTIVECAPTION
) );
455 if (! OldBrush
) goto cleanup
;
456 if (! PatBlt(MemDC
, 0, 0, lprc
->right
- lprc
->left
, lprc
->bottom
- lprc
->top
, PATCOPY
)) goto cleanup
;
459 VCenter
= (lprc
->bottom
- lprc
->top
) / 2;
460 Padding
= VCenter
- (GetSystemMetrics(SM_CYCAPTION
) / 2);
463 r
.right
= r
.left
+ (lprc
->right
- lprc
->left
);
465 r
.bottom
= r
.top
+ (GetSystemMetrics(SM_CYCAPTION
) / 2);
467 if (uFlags
& DC_ICON
)
469 // For some reason the icon isn't centered correctly...
471 UserDrawSysMenuButton(hWnd
, MemDC
, &r
, FALSE
);
478 if ((uFlags
& DC_TEXT
) && (GetWindowTextW( hWnd
, buffer
, sizeof(buffer
)/sizeof(buffer
[0]) )))
480 // Duplicate odd behaviour from Windows:
481 if ((! uFlags
& DC_SMALLCAP
) || (uFlags
& DC_ICON
) || (uFlags
& DC_INBUTTON
) ||
482 (! uFlags
& DC_ACTIVE
))
483 r
.left
+= GetSystemMetrics(SM_CXSIZE
) + Padding
;
485 r
.right
= (lprc
->right
- lprc
->left
);
487 nclm
.cbSize
= sizeof(nclm
);
488 if (! SystemParametersInfoW(SPI_GETNONCLIENTMETRICS
, sizeof(NONCLIENTMETRICSW
), &nclm
, 0)) goto cleanup
;
490 if (uFlags
& DC_INBUTTON
)
491 SetTextColor(MemDC
, SysColours
[ uFlags
& DC_ACTIVE
? COLOR_BTNTEXT
: COLOR_GRAYTEXT
]);
493 SetTextColor(MemDC
, SysColours
[ uFlags
& DC_ACTIVE
? COLOR_CAPTIONTEXT
: COLOR_INACTIVECAPTIONTEXT
]);
495 SetBkMode( MemDC
, TRANSPARENT
);
496 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_EX_TOOLWINDOW
)
497 // if (uFlags & DC_SMALLCAP) // incorrect
498 hFont
= CreateFontIndirectW(&nclm
.lfSmCaptionFont
);
500 hFont
= CreateFontIndirectW(&nclm
.lfCaptionFont
);
502 if (! hFont
) goto cleanup
;
504 hOldFont
= SelectObject(MemDC
, hFont
);
505 if (! hOldFont
) goto cleanup
;
507 DrawTextW(MemDC
, buffer
, wcslen(buffer
), &r
, DT_VCENTER
| DT_END_ELLIPSIS
);
509 // TextOutW(hDC, r.left + (GetSystemMetrics(SM_CXDLGFRAME) * 2), lprc->top + (nclm.lfCaptionFont.lfHeight / 2), buffer, wcslen(buffer));
512 if (uFlags
& DC_BUTTONS
)
514 // Windows XP draws the caption buttons with DC_BUTTONS
515 // r.left += GetSystemMetrics(SM_CXSIZE) + 1;
516 // UserDrawCaptionButton( hWnd, hDC, FALSE, DFCS_CAPTIONCLOSE);
517 // r.right -= GetSystemMetrics(SM_CXSMSIZE) + 1;
518 // UserDrawCaptionButton( hWnd, hDC, FALSE, DFCS_CAPTIONMIN);
519 // UserDrawCaptionButton( hWnd, hDC, FALSE, DFCS_CAPTIONMAX);
522 #ifdef DOUBLE_BUFFER_CAPTION
523 if (! BitBlt(hDC
, lprc
->left
, lprc
->top
, lprc
->right
- lprc
->left
, lprc
->bottom
- lprc
->top
,
524 MemDC
, 0, 0, SRCCOPY
)) goto cleanup
;
532 if (OldBrush
) SelectObject(MemDC
, OldBrush
);
533 if (hOldFont
) SelectObject(MemDC
, hOldFont
);
534 if (hFont
) DeleteObject(hFont
);
535 #ifdef DOUBLE_BUFFER_CAPTION
536 if (OldBMP
) SelectObject(MemDC
, OldBMP
);
537 if (MemBMP
) DeleteObject(MemBMP
);
540 OffsetViewportOrgEx(MemDC
, -lprc
->left
, -lprc
->top
, NULL
);
560 capflags
= DC_ICON
| DC_TEXT
;
561 capflags
|= (active
& DC_ACTIVE
);
563 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_EX_TOOLWINDOW
)
564 capflags
|= DC_SMALLCAP
;
567 // PatBlt(hDC,rect->left + GetSystemMetrics(SM_CXFRAME), rect->top +
568 // GetSystemMetrics(SM_CYFRAME), rect->right - (GetSystemMetrics(SM_CXFRAME) * 2), (rect->top +
569 // GetSystemMetrics(SM_CYCAPTION)) - 1, PATCOPY );
571 r
.left
+= GetSystemMetrics(SM_CXFRAME
);
572 r
.top
+= GetSystemMetrics(SM_CYFRAME
);
573 r
.right
-= GetSystemMetrics(SM_CXFRAME
);
574 r
.bottom
= r
.top
+ GetSystemMetrics(SM_CYCAPTION
);
575 // GetSystemMetrics(SM_CYCAPTION)) - 1, PATCOPY );
577 DrawCaption(hWnd
, hDC
, &r
, capflags
);
579 if (style
& WS_SYSMENU
)
581 // UserDrawSysMenuButton( hWnd, hDC, FALSE);
582 r
.left
+= GetSystemMetrics(SM_CXSIZE
) + 1;
583 UserDrawCaptionButton( hWnd
, hDC
, FALSE
, DFCS_CAPTIONCLOSE
);
584 r
.right
-= GetSystemMetrics(SM_CXSMSIZE
) + 1;
585 UserDrawCaptionButton( hWnd
, hDC
, FALSE
, DFCS_CAPTIONMIN
);
586 UserDrawCaptionButton( hWnd
, hDC
, FALSE
, DFCS_CAPTIONMAX
);
592 UserDrawFrameNC(HWND hWnd
, RECT
* rect
, BOOL dlgFrame
, BOOL active
)
594 HDC hDC
= GetWindowDC(hWnd
);
595 SelectObject( hDC
, GetSysColorBrush(COLOR_WINDOW
) );
596 DrawEdge(hDC
, rect
,EDGE_RAISED
, BF_RECT
| BF_MIDDLE
);
601 SCROLL_DrawScrollBar (HWND hWnd
, HDC hDC
, INT nBar
, BOOL arrows
, BOOL interior
);
604 DefWndDoPaintNC(HWND hWnd
, HRGN clip
)
613 // This won't work because it conflicts with BS_BITMAP :
614 if (GetActiveWindow() == hWnd
) Active
= TRUE
;
615 Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
616 ExStyle
= GetWindowLongW(hWnd
, GWL_EXSTYLE
);
618 hDC
= GetDCEx(hWnd
, (clip
> (HRGN
)1) ? clip
: 0, DCX_USESTYLE
| DCX_WINDOW
|
619 ((clip
> (HRGN
)1) ? (DCX_INTERSECTRGN
| DCX_KEEPCLIPRGN
) : 0));
625 /* FIXME: Test whether we need to draw anything at all. */
627 GetWindowRect(hWnd
, &rect
);
628 rect
.right
= rect
.right
- rect
.left
;
629 rect
.bottom
= rect
.bottom
- rect
.top
;
630 rect
.top
= rect
.left
= 0;
631 SelectObject(hDC
, GetSysColorPen(COLOR_WINDOWFRAME
));
632 if (UserHasThickFrameStyle(Style
, ExStyle
))
634 UserDrawFrameNC(hWnd
, &rect
, FALSE
, Active
);
635 wFrame
= GetSystemMetrics(SM_CXSIZEFRAME
);
637 else if (UserHasDlgFrameStyle(Style
, ExStyle
))
639 UserDrawFrameNC(hWnd
, &rect
, TRUE
, Active
);
640 wFrame
= GetSystemMetrics(SM_CXDLGFRAME
);
642 if (Style
& WS_CAPTION
)
645 r
.bottom
= rect
.top
+ GetSystemMetrics(SM_CYSIZE
);
646 rect
.top
+= GetSystemMetrics(SM_CYSIZE
) +
647 GetSystemMetrics(SM_CYBORDER
);
648 UserDrawCaptionNC(hDC
, &r
, hWnd
, Style
, Active
);
652 if (UserHasMenu(hWnd
, Style
))
655 r
.bottom
= rect
.top
+ GetSystemMetrics(SM_CYMENU
);
658 rect
.top
+= MenuDrawMenuBar(hDC
, &r
, hWnd
, FALSE
);
661 /* Draw scrollbars */
662 if (Style
& WS_VSCROLL
)
663 SCROLL_DrawScrollBar(hWnd
, hDC
, SB_VERT
, TRUE
, TRUE
);
664 if (Style
& WS_HSCROLL
)
665 SCROLL_DrawScrollBar(hWnd
, hDC
, SB_HORZ
, TRUE
, TRUE
);
667 /* FIXME: Draw size box.*/
669 ReleaseDC(hWnd
, hDC
);
674 DefWndPaintNC(HWND hWnd
, HRGN clip
)
676 if (IsWindowVisible(hWnd
))
680 DefWndRedrawIconTitle(hWnd
);
684 DefWndDoPaintNC(hWnd
, clip
);
692 DefWndHitTestNC(HWND hWnd
, POINT Point
)
695 ULONG Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
696 ULONG ExStyle
= GetWindowLongW(hWnd
, GWL_EXSTYLE
);
698 GetWindowRect(hWnd
, &WindowRect
);
699 if (!PtInRect(&WindowRect
, Point
))
703 if (Style
& WS_MINIMIZE
)
707 if (UserHasThickFrameStyle(Style
, ExStyle
))
709 InflateRect(&WindowRect
, -GetSystemMetrics(SM_CXFRAME
),
710 -GetSystemMetrics(SM_CYFRAME
));
711 if (!PtInRect(&WindowRect
, Point
))
713 if (Point
.y
< WindowRect
.top
)
715 if (Point
.x
< (WindowRect
.left
+ GetSystemMetrics(SM_CXSIZE
)))
719 if (Point
.x
>= (WindowRect
.right
- GetSystemMetrics(SM_CXSIZE
)))
725 if (Point
.y
>= WindowRect
.bottom
)
727 if (Point
.x
< (WindowRect
.left
+ GetSystemMetrics(SM_CXSIZE
)))
729 return(HTBOTTOMLEFT
);
731 if (Point
.x
>= (WindowRect
.right
- GetSystemMetrics(SM_CXSIZE
)))
733 return(HTBOTTOMRIGHT
);
737 if (Point
.x
< WindowRect
.left
)
739 if (Point
.y
< (WindowRect
.top
+ GetSystemMetrics(SM_CYSIZE
)))
743 if (Point
.y
>= (WindowRect
.bottom
- GetSystemMetrics(SM_CYSIZE
)))
745 return(HTBOTTOMLEFT
);
749 if (Point
.x
>= WindowRect
.right
)
751 if (Point
.y
< (WindowRect
.top
+ GetSystemMetrics(SM_CYSIZE
)))
755 if (Point
.y
>= (WindowRect
.bottom
- GetSystemMetrics(SM_CYSIZE
)))
757 return(HTBOTTOMRIGHT
);
765 if (UserHasDlgFrameStyle(Style
, ExStyle
))
767 InflateRect(&WindowRect
, -GetSystemMetrics(SM_CXDLGFRAME
),
768 -GetSystemMetrics(SM_CYDLGFRAME
));
770 else if (UserHasThinFrameStyle(Style
, ExStyle
))
772 InflateRect(&WindowRect
, -GetSystemMetrics(SM_CXBORDER
),
773 -GetSystemMetrics(SM_CYBORDER
));
775 if (!PtInRect(&WindowRect
, Point
))
781 if ((Style
& WS_CAPTION
) == WS_CAPTION
)
783 WindowRect
.top
+= (GetSystemMetrics(SM_CYCAPTION
) -
784 GetSystemMetrics(SM_CYBORDER
));
785 if (!PtInRect(&WindowRect
, Point
))
787 if ((Style
& WS_SYSMENU
) && !(ExStyle
& WS_EX_TOOLWINDOW
))
789 WindowRect
.left
+= GetSystemMetrics(SM_CXSIZE
);
790 WindowRect
.right
-= GetSystemMetrics(SM_CXSIZE
) + 1;
792 if (Point
.x
<= WindowRect
.left
)
796 if (WindowRect
.right
<= Point
.x
)
801 if (Style
& WS_MAXIMIZEBOX
|| Style
& WS_MINIMIZEBOX
)
803 WindowRect
.right
-= GetSystemMetrics(SM_CXSIZE
);
805 if (Point
.x
>= WindowRect
.right
)
810 if (Style
& WS_MINIMIZEBOX
)
812 WindowRect
.right
-= GetSystemMetrics(SM_CXSIZE
);
814 if (Point
.x
>= WindowRect
.right
)
822 ScreenToClient(hWnd
, &Point
);
823 GetClientRect(hWnd
, &WindowRect
);
825 if (PtInRect(&WindowRect
, Point
))
830 if (Style
& WS_VSCROLL
)
832 WindowRect
.right
+= GetSystemMetrics(SM_CXVSCROLL
);
833 if (PtInRect(&WindowRect
, Point
))
839 if (Style
& WS_HSCROLL
)
841 WindowRect
.bottom
+= GetSystemMetrics(SM_CYHSCROLL
);
842 if (PtInRect(&WindowRect
, Point
))
844 if ((Style
& WS_VSCROLL
) &&
845 (Point
.x
>= (WindowRect
.right
- GetSystemMetrics(SM_CXVSCROLL
))))
853 if (UserHasMenu(hWnd
, Style
))
855 if (Point
.y
< 0 && Point
.x
>= 0 && Point
.x
<= WindowRect
.right
)
865 DefWndDoButton(HWND hWnd
, WPARAM wParam
)
868 BOOL InBtn
= TRUE
, HasBtn
= FALSE
;
870 WPARAM SCMsg
, CurBtn
= wParam
, OrigBtn
= wParam
;
875 Btn
= DFCS_CAPTIONCLOSE
;
877 HasBtn
= IsCloseBoxActive(hWnd
);
880 Btn
= DFCS_CAPTIONMIN
;
882 HasBtn
= IsMinBoxActive(hWnd
);
885 Btn
= DFCS_CAPTIONMAX
;
887 HasBtn
= IsMaxBoxActive(hWnd
);
897 UserDrawCaptionButton( hWnd
, GetWindowDC(hWnd
), HasBtn
, Btn
);
901 GetMessageW(&Msg
, 0, 0, 0);
915 CurBtn
= DefWndHitTestNC(hWnd
, Msg
.pt
);
916 if(InBtn
!= (CurBtn
== OrigBtn
))
918 UserDrawCaptionButton( hWnd
, GetWindowDC(hWnd
), (CurBtn
== OrigBtn
) , Btn
);
920 InBtn
= CurBtn
== OrigBtn
;
926 UserDrawCaptionButton( hWnd
, GetWindowDC(hWnd
), FALSE
, Btn
);
928 SendMessageA(hWnd
, WM_SYSCOMMAND
, SCMsg
, 0);
933 DefWndDoScrollBarDown(HWND hWnd
, UINT Msg
, WPARAM wParam
, LPARAM lParam
)
936 Point
.x
= SLOWORD(lParam
);
937 Point
.y
= SHIWORD(lParam
);
939 SendMessageA(hWnd
, WM_SYSCOMMAND
, Msg
+ (UINT
)wParam
, lParam
);
943 DefWndHandleLButtonDownNC(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
949 HWND hTopWnd
= GetAncestor(hWnd
, GA_ROOT
);
950 if (SetActiveWindow(hTopWnd
) || GetActiveWindow() == hTopWnd
)
952 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_MOVE
+ HTCAPTION
, lParam
);
958 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_SYSMENU
)
960 if (!(GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
))
962 HDC hDC
= GetWindowDC(hWnd
);
963 // UserDrawSysMenuButton(hWnd, hDC, TRUE);
964 ReleaseDC(hWnd
, hDC
);
966 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_MOUSEMENU
+ HTSYSMENU
,
973 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_MOUSEMENU
+ HTMENU
, lParam
);
978 DefWndDoScrollBarDown(hWnd
, SC_HSCROLL
, HTHSCROLL
, lParam
);
979 //SendMessageA(hWnd, WM_SYSCOMMAND, SC_HSCROLL + HTHSCROLL, lParam);
984 DefWndDoScrollBarDown(hWnd
, SC_VSCROLL
, HTVSCROLL
, lParam
);
985 //SendMessageA(hWnd, WM_SYSCOMMAND, SC_VSCROLL + HTVSCROLL, lParam);
992 DefWndDoButton(hWnd
, wParam
);
1004 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_SIZE
+ wParam
- 2, lParam
);
1013 DefWndHandleLButtonDblClkNC(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
1021 DefWndHandleLButtonUpNC(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
1029 DefWndHandleActiveNC(HWND hWnd
, WPARAM wParam
)
1037 DefWndSetRedraw(HWND hWnd
, WPARAM wParam
)
1044 DefWndHandleSetCursor(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
1046 /* Not for child windows. */
1047 if (hWnd
!= (HWND
)wParam
)
1052 switch(LOWORD(lParam
))
1056 WORD Msg
= HIWORD(lParam
);
1057 if (Msg
== WM_LBUTTONDOWN
|| Msg
== WM_MBUTTONDOWN
||
1058 Msg
== WM_RBUTTONDOWN
)
1067 HICON hCursor
= (HICON
)GetClassLongW(hWnd
, GCL_HCURSOR
);
1079 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZEWE
)));
1085 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZENS
)));
1091 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZENWSE
)));
1097 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZENESW
)));
1100 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_ARROW
)));
1104 DefWndStartSizeMove(HWND hWnd
, WPARAM wParam
, POINT
*capturePoint
)
1110 ULONG Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
1112 GetWindowRect(hWnd
, &rectWindow
);
1114 if ((wParam
& 0xfff0) == SC_MOVE
)
1116 /* Move pointer at the center of the caption */
1118 UserGetInsideRectNC(hWnd
, &rect
);
1119 if (Style
& WS_SYSMENU
)
1120 rect
.left
+= GetSystemMetrics(SM_CXSIZE
) + 1;
1121 if (Style
& WS_MINIMIZEBOX
)
1122 rect
.right
-= GetSystemMetrics(SM_CXSIZE
) + 1;
1123 if (Style
& WS_MAXIMIZEBOX
)
1124 rect
.right
-= GetSystemMetrics(SM_CXSIZE
) + 1;
1125 pt
.x
= rectWindow
.left
+ (rect
.right
- rect
.left
) / 2;
1126 pt
.y
= rectWindow
.top
+ rect
.top
+ GetSystemMetrics(SM_CYSIZE
)/2;
1127 hittest
= HTCAPTION
;
1134 GetMessageW(&msg
, NULL
, 0, 0);
1138 hittest
= DefWndHitTestNC(hWnd
, msg
.pt
);
1139 if ((hittest
< HTLEFT
) || (hittest
> HTBOTTOMRIGHT
))
1151 pt
.x
=(rectWindow
.left
+rectWindow
.right
)/2;
1152 pt
.y
= rectWindow
.top
+ GetSystemMetrics(SM_CYFRAME
) / 2;
1156 pt
.x
=(rectWindow
.left
+rectWindow
.right
)/2;
1157 pt
.y
= rectWindow
.bottom
- GetSystemMetrics(SM_CYFRAME
) / 2;
1161 pt
.x
= rectWindow
.left
+ GetSystemMetrics(SM_CXFRAME
) / 2;
1162 pt
.y
=(rectWindow
.top
+rectWindow
.bottom
)/2;
1166 pt
.x
= rectWindow
.right
- GetSystemMetrics(SM_CXFRAME
) / 2;
1167 pt
.y
=(rectWindow
.top
+rectWindow
.bottom
)/2;
1170 case VK_ESCAPE
: return 0;
1176 SetCursorPos( pt
.x
, pt
.y
);
1177 DefWndHandleSetCursor(hWnd
, (WPARAM
)hWnd
, MAKELONG(hittest
, WM_MOUSEMOVE
));
1181 #define ON_LEFT_BORDER(hit) \
1182 (((hit) == HTLEFT) || ((hit) == HTTOPLEFT) || ((hit) == HTBOTTOMLEFT))
1183 #define ON_RIGHT_BORDER(hit) \
1184 (((hit) == HTRIGHT) || ((hit) == HTTOPRIGHT) || ((hit) == HTBOTTOMRIGHT))
1185 #define ON_TOP_BORDER(hit) \
1186 (((hit) == HTTOP) || ((hit) == HTTOPLEFT) || ((hit) == HTTOPRIGHT))
1187 #define ON_BOTTOM_BORDER(hit) \
1188 (((hit) == HTBOTTOM) || ((hit) == HTBOTTOMLEFT) || ((hit) == HTBOTTOMRIGHT))
1191 UserDrawWindowFrame(HDC hdc
, const RECT
*rect
,
1192 ULONG width
, ULONG height
, DWORD rop
)
1194 HBRUSH hbrush
= SelectObject( hdc
, GetStockObject( GRAY_BRUSH
) );
1195 PatBlt( hdc
, rect
->left
, rect
->top
,
1196 rect
->right
- rect
->left
- width
, height
, rop
);
1197 PatBlt( hdc
, rect
->left
, rect
->top
+ height
, width
,
1198 rect
->bottom
- rect
->top
- height
, rop
);
1199 PatBlt( hdc
, rect
->left
+ width
, rect
->bottom
- 1,
1200 rect
->right
- rect
->left
- width
, -height
, rop
);
1201 PatBlt( hdc
, rect
->right
- 1, rect
->top
, -width
,
1202 rect
->bottom
- rect
->top
- height
, rop
);
1203 SelectObject( hdc
, hbrush
);
1207 UserDrawMovingFrame(HDC hdc
, RECT
*rect
, BOOL thickframe
)
1211 UserDrawWindowFrame(hdc
, rect
, GetSystemMetrics(SM_CXFRAME
),
1212 GetSystemMetrics(SM_CYFRAME
), PATINVERT
);
1214 else DrawFocusRect( hdc
, rect
);
1218 DefWndDoSizeMove(HWND hwnd
, WORD wParam
)
1221 RECT sizingRect
, mouseRect
, origRect
;
1223 LONG hittest
= (LONG
)(wParam
& 0x0f);
1224 HCURSOR hDragCursor
= 0, hOldCursor
= 0;
1225 POINT minTrack
, maxTrack
;
1226 POINT capturePoint
, pt
;
1227 ULONG Style
= GetWindowLongW(hwnd
, GWL_STYLE
);
1228 ULONG ExStyle
= GetWindowLongW(hwnd
, GWL_EXSTYLE
);
1229 BOOL thickframe
= UserHasThickFrameStyle(Style
, ExStyle
);
1230 BOOL iconic
= Style
& WS_MINIMIZE
;
1232 DWORD dwPoint
= GetMessagePos();
1233 BOOL DragFullWindows
= FALSE
;
1236 SystemParametersInfoA(SPI_GETDRAGFULLWINDOWS
, 0, &DragFullWindows
, 0);
1238 pt
.x
= SLOWORD(dwPoint
);
1239 pt
.y
= SHIWORD(dwPoint
);
1242 if (IsZoomed(hwnd
) || !IsWindowVisible(hwnd
))
1247 if ((wParam
& 0xfff0) == SC_MOVE
)
1251 hittest
= DefWndStartSizeMove(hwnd
, wParam
, &capturePoint
);
1264 if (hittest
&& hittest
!= HTSYSMENU
)
1271 hittest
= DefWndStartSizeMove(hwnd
, wParam
, &capturePoint
);
1280 if (Style
& WS_CHILD
)
1282 hWndParent
= GetParent(hwnd
);
1285 /* Get min/max info */
1287 WinPosGetMinMaxInfo(hwnd
, NULL
, NULL
, &minTrack
, &maxTrack
);
1288 GetWindowRect(hwnd
, &sizingRect
);
1289 origRect
= sizingRect
;
1290 if (Style
& WS_CHILD
)
1292 GetClientRect(hWndParent
, &mouseRect
);
1296 SetRect(&mouseRect
, 0, 0, GetSystemMetrics(SM_CXSCREEN
),
1297 GetSystemMetrics(SM_CYSCREEN
));
1299 if (ON_LEFT_BORDER(hittest
))
1301 mouseRect
.left
= max( mouseRect
.left
, sizingRect
.right
-maxTrack
.x
);
1302 mouseRect
.right
= min( mouseRect
.right
, sizingRect
.right
-minTrack
.x
);
1304 else if (ON_RIGHT_BORDER(hittest
))
1306 mouseRect
.left
= max( mouseRect
.left
, sizingRect
.left
+minTrack
.x
);
1307 mouseRect
.right
= min( mouseRect
.right
, sizingRect
.left
+maxTrack
.x
);
1309 if (ON_TOP_BORDER(hittest
))
1311 mouseRect
.top
= max( mouseRect
.top
, sizingRect
.bottom
-maxTrack
.y
);
1312 mouseRect
.bottom
= min( mouseRect
.bottom
,sizingRect
.bottom
-minTrack
.y
);
1314 else if (ON_BOTTOM_BORDER(hittest
))
1316 mouseRect
.top
= max( mouseRect
.top
, sizingRect
.top
+minTrack
.y
);
1317 mouseRect
.bottom
= min( mouseRect
.bottom
, sizingRect
.top
+maxTrack
.y
);
1319 if (Style
& WS_CHILD
)
1321 MapWindowPoints( hWndParent
, 0, (LPPOINT
)&mouseRect
, 2 );
1323 SendMessageA( hwnd
, WM_ENTERSIZEMOVE
, 0, 0 );
1325 if (GetCapture() != hwnd
) SetCapture( hwnd
);
1327 if (Style
& WS_CHILD
)
1329 /* Retrieve a default cache DC (without using the window style) */
1330 hdc
= GetDCEx(hWndParent
, 0, DCX_CACHE
);
1337 if( iconic
) /* create a cursor for dragging */
1339 HICON hIcon
= (HICON
)GetClassLongW(hwnd
, GCL_HICON
);
1340 if(!hIcon
) hIcon
= (HICON
)SendMessageW( hwnd
, WM_QUERYDRAGICON
, 0, 0L);
1341 if( hIcon
) hDragCursor
= CursorIconToCursor( hIcon
, TRUE
);
1342 if( !hDragCursor
) iconic
= FALSE
;
1345 /* invert frame if WIN31_LOOK to indicate mouse click on caption */
1346 if( !iconic
&& !DragFullWindows
)
1348 UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1355 GetMessageW(&msg
, 0, 0, 0);
1357 /* Exit on button-up, Return, or Esc */
1358 if ((msg
.message
== WM_LBUTTONUP
) ||
1359 ((msg
.message
== WM_KEYDOWN
) &&
1360 ((msg
.wParam
== VK_RETURN
) || (msg
.wParam
== VK_ESCAPE
)))) break;
1362 if (msg
.message
== WM_PAINT
)
1364 if(!iconic
&& !DragFullWindows
) UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1365 UpdateWindow( msg
.hwnd
);
1366 if(!iconic
&& !DragFullWindows
) UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1370 if ((msg
.message
!= WM_KEYDOWN
) && (msg
.message
!= WM_MOUSEMOVE
))
1371 continue; /* We are not interested in other messages */
1375 if (msg
.message
== WM_KEYDOWN
) switch(msg
.wParam
)
1377 case VK_UP
: pt
.y
-= 8; break;
1378 case VK_DOWN
: pt
.y
+= 8; break;
1379 case VK_LEFT
: pt
.x
-= 8; break;
1380 case VK_RIGHT
: pt
.x
+= 8; break;
1383 pt
.x
= max( pt
.x
, mouseRect
.left
);
1384 pt
.x
= min( pt
.x
, mouseRect
.right
);
1385 pt
.y
= max( pt
.y
, mouseRect
.top
);
1386 pt
.y
= min( pt
.y
, mouseRect
.bottom
);
1388 dx
= pt
.x
- capturePoint
.x
;
1389 dy
= pt
.y
- capturePoint
.y
;
1397 if( iconic
) /* ok, no system popup tracking */
1399 hOldCursor
= SetCursor(hDragCursor
);
1401 WinPosShowIconTitle( hwnd
, FALSE
);
1405 if (msg
.message
== WM_KEYDOWN
) SetCursorPos( pt
.x
, pt
.y
);
1408 RECT newRect
= sizingRect
;
1409 WPARAM wpSizingHit
= 0;
1411 if (hittest
== HTCAPTION
) OffsetRect( &newRect
, dx
, dy
);
1412 if (ON_LEFT_BORDER(hittest
)) newRect
.left
+= dx
;
1413 else if (ON_RIGHT_BORDER(hittest
)) newRect
.right
+= dx
;
1414 if (ON_TOP_BORDER(hittest
)) newRect
.top
+= dy
;
1415 else if (ON_BOTTOM_BORDER(hittest
)) newRect
.bottom
+= dy
;
1416 if(!iconic
&& !DragFullWindows
) UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1419 /* determine the hit location */
1420 if (hittest
>= HTLEFT
&& hittest
<= HTBOTTOMRIGHT
)
1421 wpSizingHit
= WMSZ_LEFT
+ (hittest
- HTLEFT
);
1422 SendMessageA( hwnd
, WM_SIZING
, wpSizingHit
, (LPARAM
)&newRect
);
1426 if(!DragFullWindows
)
1427 UserDrawMovingFrame( hdc
, &newRect
, thickframe
);
1429 /* To avoid any deadlocks, all the locks on the windows
1430 structures must be suspended before the SetWindowPos */
1431 SetWindowPos( hwnd
, 0, newRect
.left
, newRect
.top
,
1432 newRect
.right
- newRect
.left
,
1433 newRect
.bottom
- newRect
.top
,
1434 ( hittest
== HTCAPTION
) ? SWP_NOSIZE
: 0 );
1437 sizingRect
= newRect
;
1445 if( moved
) /* restore cursors, show icon title later on */
1447 ShowCursor( FALSE
);
1448 SetCursor( hOldCursor
);
1450 DestroyCursor( hDragCursor
);
1452 else if(!DragFullWindows
)
1453 UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1455 if (Style
& WS_CHILD
)
1456 ReleaseDC( hWndParent
, hdc
);
1458 ReleaseDC( 0, hdc
);
1460 SendMessageA( hwnd
, WM_EXITSIZEMOVE
, 0, 0 );
1461 SendMessageA( hwnd
, WM_SETVISIBLE
, !IsIconic(hwnd
), 0L);
1463 /* window moved or resized */
1466 /* if the moving/resizing isn't canceled call SetWindowPos
1467 * with the new position or the new size of the window
1469 if (!((msg
.message
== WM_KEYDOWN
) && (msg
.wParam
== VK_ESCAPE
)) )
1471 /* NOTE: SWP_NOACTIVATE prevents document window activation in Word 6 */
1472 if(!DragFullWindows
)
1473 SetWindowPos( hwnd
, 0, sizingRect
.left
, sizingRect
.top
,
1474 sizingRect
.right
- sizingRect
.left
,
1475 sizingRect
.bottom
- sizingRect
.top
,
1476 ( hittest
== HTCAPTION
) ? SWP_NOSIZE
: 0 );
1478 else { /* restore previous size/position */
1480 SetWindowPos( hwnd
, 0, origRect
.left
, origRect
.top
,
1481 origRect
.right
- origRect
.left
,
1482 origRect
.bottom
- origRect
.top
,
1483 ( hittest
== HTCAPTION
) ? SWP_NOSIZE
: 0 );
1487 if( IsWindow(hwnd
) )
1488 if( Style
& WS_MINIMIZE
)
1490 /* Single click brings up the system menu when iconized */
1494 if( Style
& WS_SYSMENU
)
1495 SendMessageA( hwnd
, WM_SYSCOMMAND
,
1496 SC_MOUSEMENU
+ HTSYSMENU
, MAKELONG(pt
.x
,pt
.y
));
1498 else WinPosShowIconTitle( hwnd
, TRUE
);
1504 DefWndHandleSysCommand(HWND hWnd
, WPARAM wParam
, POINT Pt
)
1506 switch (wParam
& 0xfff0)
1510 DefWndDoSizeMove(hWnd
, wParam
);
1513 SendMessageA(hWnd
, WM_CLOSE
, 0, 0);
1516 MenuTrackMouseMenuBar(hWnd
, wParam
, Pt
);
1519 MenuTrackKbdMenuBar(hWnd
, wParam
, Pt
.x
);
1522 /* FIXME: Implement */
1532 DefWndAdjustRect(RECT
* Rect
, ULONG Style
, BOOL Menu
, ULONG ExStyle
)
1534 if (Style
& WS_ICONIC
)
1539 if (UserHasThickFrameStyle(Style
, ExStyle
))
1541 InflateRect(Rect
, GetSystemMetrics(SM_CXFRAME
),
1542 GetSystemMetrics(SM_CYFRAME
));
1544 else if (UserHasDlgFrameStyle(Style
, ExStyle
))
1546 InflateRect(Rect
, GetSystemMetrics(SM_CXDLGFRAME
),
1547 GetSystemMetrics(SM_CYDLGFRAME
));
1549 else if (UserHasThinFrameStyle(Style
, ExStyle
))
1551 InflateRect(Rect
, GetSystemMetrics(SM_CXBORDER
),
1552 GetSystemMetrics(SM_CYBORDER
));
1554 if (Style
& WS_CAPTION
)
1556 Rect
->top
-= (GetSystemMetrics(SM_CYCAPTION
) -
1557 GetSystemMetrics(SM_CYBORDER
)) + 1;
1561 Rect
->top
-= GetSystemMetrics(SM_CYMENU
) + GetSystemMetrics(SM_CYBORDER
);
1563 if (Style
& WS_VSCROLL
)
1565 Rect
->right
+= GetSystemMetrics(SM_CXVSCROLL
) - 1;
1566 if (UserHasAnyFrameStyle(Style
, ExStyle
))
1571 if (Style
& WS_HSCROLL
)
1573 Rect
->bottom
+= GetSystemMetrics(SM_CYHSCROLL
) - 1;
1574 if (UserHasAnyFrameStyle(Style
, ExStyle
))
1582 DefWndNCCalcSize(HWND hWnd
, RECT
* Rect
)
1585 LONG Style
= GetClassLongW(hWnd
, GCL_STYLE
);
1586 RECT TmpRect
= {0, 0, 0, 0};
1588 if (Style
& CS_VREDRAW
)
1590 Result
|= WVR_VREDRAW
;
1592 if (Style
& CS_HREDRAW
)
1594 Result
|= WVR_HREDRAW
;
1597 if (!(GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
))
1599 DefWndAdjustRect(&TmpRect
, GetWindowLongW(hWnd
, GWL_STYLE
),
1600 FALSE
, GetWindowLongW(hWnd
, GWL_EXSTYLE
));
1601 Rect
->left
-= TmpRect
.left
;
1602 Rect
->top
-= TmpRect
.top
;
1603 Rect
->right
-= TmpRect
.right
;
1604 Rect
->bottom
-= TmpRect
.bottom
;
1605 if (UserHasMenu(hWnd
, GetWindowLongW(hWnd
, GWL_STYLE
)))
1607 Rect
->top
+= MenuGetMenuBarHeight(hWnd
, Rect
->right
- Rect
->left
,
1608 -TmpRect
.left
, -TmpRect
.top
) + 1;
1610 if (Rect
->top
> Rect
->bottom
)
1611 Rect
->bottom
= Rect
->top
;
1612 if (Rect
->left
> Rect
->right
)
1613 Rect
->right
= Rect
->left
;
1620 DefWndHandleWindowPosChanging(HWND hWnd
, WINDOWPOS
* Pos
)
1622 POINT maxSize
, minTrack
;
1623 LONG style
= GetWindowLongA(hWnd
, GWL_STYLE
);
1625 if (Pos
->flags
& SWP_NOSIZE
) return 0;
1626 if ((style
& WS_THICKFRAME
) || ((style
& (WS_POPUP
| WS_CHILD
)) == 0))
1628 WinPosGetMinMaxInfo(hWnd
, &maxSize
, NULL
, &minTrack
, NULL
);
1629 Pos
->cx
= min(Pos
->cx
, maxSize
.x
);
1630 Pos
->cy
= min(Pos
->cy
, maxSize
.y
);
1631 if (!(style
& WS_MINIMIZE
))
1633 if (Pos
->cx
< minTrack
.x
) Pos
->cx
= minTrack
.x
;
1634 if (Pos
->cy
< minTrack
.y
) Pos
->cy
= minTrack
.y
;
1640 /* Undocumented flags. */
1641 #define SWP_NOCLIENTMOVE 0x0800
1642 #define SWP_NOCLIENTSIZE 0x1000
1645 DefWndHandleWindowPosChanged(HWND hWnd
, WINDOWPOS
* Pos
)
1649 GetClientRect(hWnd
, &rect
);
1651 if (!(Pos
->flags
& SWP_NOCLIENTMOVE
))
1652 SendMessageW(hWnd
, WM_MOVE
, 0, MAKELONG(rect
.left
, rect
.top
));
1654 if (!(Pos
->flags
& SWP_NOCLIENTSIZE
))
1656 WPARAM wp
= SIZE_RESTORED
;
1657 if (IsZoomed(hWnd
)) wp
= SIZE_MAXIMIZED
;
1658 else if (IsIconic(hWnd
)) wp
= SIZE_MINIMIZED
;
1659 SendMessageW(hWnd
, WM_SIZE
, wp
,
1660 MAKELONG(rect
.right
- rect
.left
, rect
.bottom
- rect
.top
));
1668 User32DefWindowProc(HWND hWnd
,
1678 return (DefWndPaintNC(hWnd
, (HRGN
)wParam
));
1683 return (DefWndNCCalcSize(hWnd
, (RECT
*)lParam
));
1686 case WM_WINDOWPOSCHANGING
:
1688 return (DefWndHandleWindowPosChanging(hWnd
, (WINDOWPOS
*)lParam
));
1691 case WM_WINDOWPOSCHANGED
:
1693 return (DefWndHandleWindowPosChanged(hWnd
, (WINDOWPOS
*)lParam
));
1699 Point
.x
= SLOWORD(lParam
);
1700 Point
.y
= SHIWORD(lParam
);
1701 return (DefWndHitTestNC(hWnd
, Point
));
1704 case WM_NCLBUTTONDOWN
:
1706 return (DefWndHandleLButtonDownNC(hWnd
, wParam
, lParam
));
1709 case WM_NCLBUTTONUP
:
1711 return (DefWndHandleLButtonUpNC(hWnd
, wParam
, lParam
));
1714 case WM_LBUTTONDBLCLK
:
1715 case WM_NCLBUTTONDBLCLK
:
1717 return (DefWndHandleLButtonDblClkNC(hWnd
, wParam
, lParam
));
1720 case WM_NCRBUTTONDOWN
:
1722 if (wParam
== HTCAPTION
)
1732 if (hWnd
== GetCapture())
1736 Pt
.x
= SLOWORD(lParam
);
1737 Pt
.y
= SHIWORD(lParam
);
1738 ClientToScreen(hWnd
, &Pt
);
1739 lParam
= MAKELPARAM(Pt
.x
, Pt
.y
);
1742 SendMessageW(hWnd
, WM_CONTEXTMENU
, (WPARAM
)hWnd
, lParam
);
1746 SendMessageA(hWnd
, WM_CONTEXTMENU
, (WPARAM
)hWnd
, lParam
);
1751 case WM_CONTEXTMENU
:
1753 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1757 SendMessageW(GetParent(hWnd
), Msg
, wParam
, lParam
);
1761 SendMessageA(GetParent(hWnd
), WM_CONTEXTMENU
, wParam
, lParam
);
1769 Pt
.x
= SLOWORD(lParam
);
1770 Pt
.y
= SHIWORD(lParam
);
1771 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1773 ScreenToClient(GetParent(hWnd
), &Pt
);
1776 HitCode
= DefWndHitTestNC(hWnd
, Pt
);
1778 if (HitCode
== HTCAPTION
|| HitCode
== HTSYSMENU
)
1780 TrackPopupMenu(GetSystemMenu(hWnd
, FALSE
),
1781 TPM_LEFTBUTTON
| TPM_RIGHTBUTTON
,
1782 Pt
.x
, Pt
.y
, 0, hWnd
, NULL
);
1790 return (DefWndHandleActiveNC(hWnd
, wParam
));
1795 /* FIXME: Implement. */
1803 HDC hDC
= BeginPaint(hWnd
, &Ps
);
1807 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
&&
1808 (hIcon
= (HICON
)GetClassLongW(hWnd
, GCL_HICON
)) != NULL
)
1812 GetWindowRect(hWnd
, &WindowRect
);
1813 x
= (WindowRect
.right
- WindowRect
.left
-
1814 GetSystemMetrics(SM_CXICON
)) / 2;
1815 y
= (WindowRect
.bottom
- WindowRect
.top
-
1816 GetSystemMetrics(SM_CYICON
)) / 2;
1817 DrawIcon(hDC
, x
, y
, hIcon
);
1819 if (GetWindowLongW(hWnd
, GWL_EXSTYLE
) & WS_EX_CLIENTEDGE
)
1822 GetClientRect(hWnd
, &WindowRect
);
1823 DrawEdge(hDC
, &WindowRect
, EDGE_SUNKEN
, BF_RECT
);
1825 EndPaint(hWnd
, &Ps
);
1833 hRgn
= CreateRectRgn(0, 0, 0, 0);
1834 if (GetUpdateRgn(hWnd
, hRgn
, FALSE
) != NULLREGION
)
1836 RedrawWindow(hWnd
, NULL
, hRgn
,
1837 RDW_ERASENOW
| RDW_ERASE
| RDW_FRAME
|
1846 DefWndSetRedraw(hWnd
, wParam
);
1852 DestroyWindow(hWnd
);
1856 case WM_MOUSEACTIVATE
:
1858 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1863 Ret
= SendMessageW(GetParent(hWnd
), WM_MOUSEACTIVATE
,
1868 Ret
= SendMessageA(GetParent(hWnd
), WM_MOUSEACTIVATE
,
1876 return ((LOWORD(lParam
) >= HTCLIENT
) ? MA_ACTIVATE
: MA_NOACTIVATE
);
1881 /* Check if the window is minimized. */
1882 if (LOWORD(lParam
) != WA_INACTIVE
&&
1883 !(GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
))
1892 if (GetWindowLongW(hWnd
, GWL_STYLE
& WS_CHILD
))
1896 return (SendMessageW(GetParent(hWnd
), WM_MOUSEWHEEL
,
1901 return (SendMessageA(GetParent(hWnd
), WM_MOUSEWHEEL
,
1909 case WM_ICONERASEBKGND
:
1912 HBRUSH hBrush
= (HBRUSH
)GetClassLongW(hWnd
, GCL_HBRBACKGROUND
);
1918 if (0 == (((DWORD
) hBrush
) & 0xffff0000))
1920 hBrush
= GetSysColorBrush((DWORD
) hBrush
- 1);
1922 GetClipBox((HDC
)wParam
, &Rect
);
1923 FillRect((HDC
)wParam
, &Rect
, hBrush
);
1927 /* FIXME: Implement colour controls. */
1929 case WM_CTLCOLORMSGBOX:
1930 case WM_CTLCOLOREDIT:
1931 case WM_CTLCOLORLISTBOX:
1932 case WM_CTLCOLORBTN:
1933 case WM_CTLCOLORDLG:
1934 case WM_CTLCOLORSTATIC:
1935 case WM_CTLCOLORSCROLLBAR:
1941 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1943 if (LOWORD(lParam
) < HTLEFT
|| LOWORD(lParam
) > HTBOTTOMRIGHT
)
1948 bResult
= SendMessageW(GetParent(hWnd
), WM_SETCURSOR
,
1953 bResult
= SendMessageA(GetParent(hWnd
), WM_SETCURSOR
,
1962 return (DefWndHandleSetCursor(hWnd
, wParam
, lParam
));
1968 Pt
.x
= SLOWORD(lParam
);
1969 Pt
.y
= SHIWORD(lParam
);
1970 return (DefWndHandleSysCommand(hWnd
, wParam
, Pt
));
1973 /* FIXME: Handle key messages. */
1981 /* FIXME: This is also incomplete. */
1984 if (HIWORD(lParam
) & KEYDATA_ALT
)
1986 if (wParam
== VK_F4
) /* Try to close the window */
1988 HWND top
= GetAncestor(hWnd
, GA_ROOT
);
1989 if (!(GetClassLongW(top
, GCL_STYLE
) & CS_NOCLOSE
))
1992 PostMessageW(top
, WM_SYSCOMMAND
, SC_CLOSE
, 0);
1994 PostMessageA(top
, WM_SYSCOMMAND
, SC_CLOSE
, 0);
2007 Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
2008 if (!(Style
& WS_POPUP
))
2010 if ((Style
& WS_VISIBLE
) && wParam
)
2012 if (!(Style
& WS_VISIBLE
) && !wParam
)
2014 if (!GetWindow(hWnd
, GW_OWNER
))
2016 ShowWindow(hWnd
, wParam
? SW_SHOWNA
: SW_HIDE
);
2022 /* FIXME: Check for a desktop. */
2023 if (GetCapture() == hWnd
)
2035 /* FIXME: Implement this. */
2038 case WM_QUERYDROPOBJECT
:
2040 if (GetWindowLongW(hWnd
, GWL_EXSTYLE
) & WS_EX_ACCEPTFILES
)
2047 case WM_QUERYDRAGICON
:
2052 hIcon
= (HICON
)GetClassLongW(hWnd
, GCL_HICON
);
2055 return ((LRESULT
)hIcon
);
2057 for (Len
= 1; Len
< 64; Len
++)
2059 if ((hIcon
= LoadIconW(NULL
, MAKEINTRESOURCEW(Len
))) != NULL
)
2061 return((LRESULT
)hIcon
);
2064 return ((LRESULT
)LoadIconW(0, IDI_APPLICATION
));
2067 /* FIXME: WM_ISACTIVEICON */
2069 case WM_NOTIFYFORMAT
:
2071 if (IsWindowUnicode(hWnd
))
2073 return(NFR_UNICODE
);
2083 INT Index
= (wParam
!= 0) ? GCL_HICON
: GCL_HICONSM
;
2084 HICON hOldIcon
= (HICON
)GetClassLongW(hWnd
, Index
);
2085 SetClassLongW(hWnd
, Index
, lParam
);
2086 SetWindowPos(hWnd
, 0, 0, 0, 0, 0,
2087 SWP_FRAMECHANGED
| SWP_NOSIZE
| SWP_NOMOVE
|
2088 SWP_NOACTIVATE
| SWP_NOZORDER
);
2089 return ((LRESULT
)hOldIcon
);
2094 INT Index
= (wParam
!= 0) ? GCL_HICON
: GCL_HICONSM
;
2095 return (GetClassLongW(hWnd
, Index
));
2102 SendMessageW(GetParent(hWnd
), Msg
, wParam
, lParam
);
2106 SendMessageA(GetParent(hWnd
), Msg
, wParam
, lParam
);
2112 case WM_QUERYENDSESSION
:
2122 DefWindowProcA(HWND hWnd
,
2127 static LPSTR WindowTextAtom
= 0;
2134 CREATESTRUCTA
*Cs
= (CREATESTRUCTA
*)lParam
;
2135 if (HIWORD(Cs
->lpszName
))
2137 if (0 == WindowTextAtom
)
2140 (LPSTR
)(ULONG
)GlobalAddAtomA("USER32!WindowTextAtomA");
2142 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2143 strlen(Cs
->lpszName
) * sizeof(CHAR
));
2144 strcpy(WindowText
, Cs
->lpszName
);
2145 SetPropA(hWnd
, WindowTextAtom
, WindowText
);
2150 case WM_GETTEXTLENGTH
:
2152 if (WindowTextAtom
== 0 ||
2153 (WindowText
= GetPropA(hWnd
, WindowTextAtom
)) == NULL
)
2157 return (strlen(WindowText
));
2162 if (WindowTextAtom
== 0 ||
2163 (WindowText
= GetPropA(hWnd
, WindowTextAtom
)) == NULL
)
2167 *((PSTR
)lParam
) = '\0';
2171 strncpy((LPSTR
)lParam
, WindowText
, wParam
);
2172 return (min(wParam
, strlen(WindowText
)));
2177 if (0 == WindowTextAtom
)
2180 (LPSTR
)(DWORD
)GlobalAddAtomA("USER32!WindowTextAtomA");
2182 if (WindowTextAtom
!= 0 &&
2183 (WindowText
= GetPropA(hWnd
, WindowTextAtom
)) == NULL
)
2185 RtlFreeHeap(RtlGetProcessHeap(), 0, WindowText
);
2187 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2188 (strlen((PSTR
)lParam
) + 1) * sizeof(CHAR
));
2189 strcpy(WindowText
, (PSTR
)lParam
);
2190 SetPropA(hWnd
, WindowTextAtom
, WindowText
);
2191 if (0 != (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CAPTION
))
2193 DefWndPaintNC(hWnd
, (HRGN
) 1);
2199 FIXME: Implement these.
2201 case WM_IME_KEYDOWN:
2203 case WM_IME_STARTCOMPOSITION:
2204 case WM_IME_COMPOSITION:
2205 case WM_IME_ENDCOMPOSITION:
2207 case WM_IME_SETCONTEXT:
2212 if (WindowTextAtom
!= 0 &&
2213 (WindowText
= RemovePropA(hWnd
, WindowTextAtom
)) == NULL
)
2215 RtlFreeHeap(GetProcessHeap(), 0, WindowText
);
2221 return User32DefWindowProc(hWnd
, Msg
, wParam
, lParam
, FALSE
);
2226 DefWindowProcW(HWND hWnd
,
2231 static LPWSTR WindowTextAtom
= 0;
2238 CREATESTRUCTW
* CreateStruct
= (CREATESTRUCTW
*)lParam
;
2239 if (HIWORD(CreateStruct
->lpszName
))
2241 if (0 == WindowTextAtom
)
2244 (LPWSTR
)(DWORD
)GlobalAddAtomW(L
"USER32!WindowTextAtomW");
2246 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2247 wcslen(CreateStruct
->lpszName
) * sizeof(WCHAR
));
2248 wcscpy(WindowText
, CreateStruct
->lpszName
);
2249 SetPropW(hWnd
, WindowTextAtom
, WindowText
);
2254 case WM_GETTEXTLENGTH
:
2256 if (WindowTextAtom
== 0 ||
2257 (WindowText
= GetPropW(hWnd
, WindowTextAtom
)) == NULL
)
2261 return (wcslen(WindowText
));
2266 if (WindowTextAtom
== 0 ||
2267 (WindowText
= GetPropW(hWnd
, WindowTextAtom
)) == NULL
)
2271 ((PWSTR
)lParam
) = '\0';
2275 wcsncpy((PWSTR
)lParam
, WindowText
, wParam
);
2276 return (min(wParam
, wcslen(WindowText
)));
2281 if (WindowTextAtom
== 0)
2284 (LPWSTR
)(DWORD
)GlobalAddAtomW(L
"USER32!WindowTextAtomW");
2286 if (WindowTextAtom
!= 0 &&
2287 (WindowText
= GetPropW(hWnd
, WindowTextAtom
)) == NULL
)
2289 RtlFreeHeap(RtlGetProcessHeap(), 0, WindowText
);
2291 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2292 (wcslen((PWSTR
)lParam
) + 1) * sizeof(WCHAR
));
2293 wcscpy(WindowText
, (PWSTR
)lParam
);
2294 SetPropW(hWnd
, WindowTextAtom
, WindowText
);
2295 if ((GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CAPTION
) == WS_CAPTION
)
2297 DefWndPaintNC(hWnd
, (HRGN
)1);
2304 SendMessageW(hWnd
, WM_CHAR
, wParam
, lParam
);
2308 case WM_IME_SETCONTEXT
:
2316 if (WindowTextAtom
!= 0 &&
2317 (WindowText
= RemovePropW(hWnd
, WindowTextAtom
)) == NULL
)
2319 RtlFreeHeap(RtlGetProcessHeap(), 0, WindowText
);
2325 return User32DefWindowProc(hWnd
, Msg
, wParam
, lParam
, TRUE
);