1 /* $Id: defwnd.c,v 1.85 2003/09/09 09:39:21 gvg 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(255, 255, 255) /* COLOR_WINDOW */,
52 RGB(0, 0, 0) /* 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, Height
;
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 /* Windows behaves like this */
460 Height
= GetSystemMetrics(SM_CYCAPTION
) - 1;
462 VCenter
= (lprc
->bottom
- lprc
->top
) / 2;
463 Padding
= VCenter
- (Height
/ 2);
466 r
.right
= r
.left
+ (lprc
->right
- lprc
->left
);
468 r
.bottom
= r
.top
+ (Height
/ 2);
470 if (uFlags
& DC_ICON
)
472 // For some reason the icon isn't centered correctly...
474 UserDrawSysMenuButton(hWnd
, MemDC
, &r
, FALSE
);
481 if ((uFlags
& DC_TEXT
) && (GetWindowTextW( hWnd
, buffer
, sizeof(buffer
)/sizeof(buffer
[0]) )))
483 // Duplicate odd behaviour from Windows:
484 if ((! uFlags
& DC_SMALLCAP
) || (uFlags
& DC_ICON
) || (uFlags
& DC_INBUTTON
) ||
485 (! uFlags
& DC_ACTIVE
))
486 r
.left
+= GetSystemMetrics(SM_CXSIZE
) + Padding
;
488 r
.right
= (lprc
->right
- lprc
->left
);
490 nclm
.cbSize
= sizeof(nclm
);
491 if (! SystemParametersInfoW(SPI_GETNONCLIENTMETRICS
, sizeof(NONCLIENTMETRICSW
), &nclm
, 0)) goto cleanup
;
493 if (uFlags
& DC_INBUTTON
)
494 SetTextColor(MemDC
, SysColours
[ uFlags
& DC_ACTIVE
? COLOR_BTNTEXT
: COLOR_GRAYTEXT
]);
496 SetTextColor(MemDC
, SysColours
[ uFlags
& DC_ACTIVE
? COLOR_CAPTIONTEXT
: COLOR_INACTIVECAPTIONTEXT
]);
498 SetBkMode( MemDC
, TRANSPARENT
);
499 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_EX_TOOLWINDOW
)
500 // if (uFlags & DC_SMALLCAP) // incorrect
501 hFont
= CreateFontIndirectW(&nclm
.lfSmCaptionFont
);
503 hFont
= CreateFontIndirectW(&nclm
.lfCaptionFont
);
505 if (! hFont
) goto cleanup
;
507 hOldFont
= SelectObject(MemDC
, hFont
);
508 if (! hOldFont
) goto cleanup
;
510 DrawTextW(MemDC
, buffer
, wcslen(buffer
), &r
, DT_VCENTER
| DT_END_ELLIPSIS
);
512 // TextOutW(hDC, r.left + (GetSystemMetrics(SM_CXDLGFRAME) * 2), lprc->top + (nclm.lfCaptionFont.lfHeight / 2), buffer, wcslen(buffer));
515 if (uFlags
& DC_BUTTONS
)
517 // Windows XP draws the caption buttons with DC_BUTTONS
518 // r.left += GetSystemMetrics(SM_CXSIZE) + 1;
519 // UserDrawCaptionButton( hWnd, hDC, FALSE, DFCS_CAPTIONCLOSE);
520 // r.right -= GetSystemMetrics(SM_CXSMSIZE) + 1;
521 // UserDrawCaptionButton( hWnd, hDC, FALSE, DFCS_CAPTIONMIN);
522 // UserDrawCaptionButton( hWnd, hDC, FALSE, DFCS_CAPTIONMAX);
525 #ifdef DOUBLE_BUFFER_CAPTION
526 if (! BitBlt(hDC
, lprc
->left
, lprc
->top
, lprc
->right
- lprc
->left
, lprc
->bottom
- lprc
->top
,
527 MemDC
, 0, 0, SRCCOPY
)) goto cleanup
;
535 if (OldBrush
) SelectObject(MemDC
, OldBrush
);
536 if (hOldFont
) SelectObject(MemDC
, hOldFont
);
537 if (hFont
) DeleteObject(hFont
);
538 #ifdef DOUBLE_BUFFER_CAPTION
539 if (OldBMP
) SelectObject(MemDC
, OldBMP
);
540 if (MemBMP
) DeleteObject(MemBMP
);
543 OffsetViewportOrgEx(MemDC
, -lprc
->left
, -lprc
->top
, NULL
);
565 capflags
= DC_ICON
| DC_TEXT
;
566 capflags
|= (active
& DC_ACTIVE
);
568 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_EX_TOOLWINDOW
)
569 capflags
|= DC_SMALLCAP
;
572 // PatBlt(hDC,rect->left + GetSystemMetrics(SM_CXFRAME), rect->top +
573 // GetSystemMetrics(SM_CYFRAME), rect->right - (GetSystemMetrics(SM_CXFRAME) * 2), (rect->top +
574 // GetSystemMetrics(SM_CYCAPTION)) - 1, PATCOPY );
576 r
.left
+= GetSystemMetrics(SM_CXFRAME
);
577 r
.top
+= GetSystemMetrics(SM_CYFRAME
);
578 r
.right
-= GetSystemMetrics(SM_CXFRAME
);
579 r
.bottom
= r
.top
+ GetSystemMetrics(SM_CYCAPTION
) - 1;
580 // GetSystemMetrics(SM_CYCAPTION)) - 1, PATCOPY );
582 DrawCaption(hWnd
, hDC
, &r
, capflags
);
584 /* draw line below caption */
585 lPen
= GetSysColorPen(COLOR_MENU
);
586 oPen
= SelectObject(hDC
, lPen
);
587 MoveToEx(hDC
, r
.left
, r
.bottom
, &OldPos
);
588 LineTo(hDC
, r
.right
, r
.bottom
);
589 MoveToEx(hDC
, OldPos
.x
, OldPos
.y
, NULL
);
590 SelectObject(hDC
, oPen
);
593 if (style
& WS_SYSMENU
)
595 // UserDrawSysMenuButton( hWnd, hDC, FALSE);
596 r
.left
+= GetSystemMetrics(SM_CXSIZE
) + 1;
597 UserDrawCaptionButton( hWnd
, hDC
, FALSE
, DFCS_CAPTIONCLOSE
);
598 r
.right
-= GetSystemMetrics(SM_CXSMSIZE
) + 1;
599 UserDrawCaptionButton( hWnd
, hDC
, FALSE
, DFCS_CAPTIONMIN
);
600 UserDrawCaptionButton( hWnd
, hDC
, FALSE
, DFCS_CAPTIONMAX
);
606 UserDrawFrameNC(HDC hDC
, RECT
* rect
, BOOL dlgFrame
, BOOL active
)
608 DrawEdge(hDC
, rect
,EDGE_RAISED
, BF_RECT
| BF_MIDDLE
);
613 SCROLL_DrawScrollBar (HWND hWnd
, HDC hDC
, INT nBar
, BOOL arrows
, BOOL interior
);
616 DefWndDoPaintNC(HWND hWnd
, HRGN clip
)
625 if (GetActiveWindow() == hWnd
) Active
= TRUE
;
626 Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
627 ExStyle
= GetWindowLongW(hWnd
, GWL_EXSTYLE
);
629 hDC
= GetDCEx(hWnd
, (clip
> (HRGN
)1) ? clip
: 0, DCX_USESTYLE
| DCX_WINDOW
|
630 ((clip
> (HRGN
)1) ? (DCX_INTERSECTRGN
| DCX_KEEPCLIPRGN
) : 0));
636 /* FIXME: Test whether we need to draw anything at all. */
638 GetWindowRect(hWnd
, &rect
);
639 rect
.right
= rect
.right
- rect
.left
;
640 rect
.bottom
= rect
.bottom
- rect
.top
;
641 rect
.top
= rect
.left
= 0;
642 SelectObject(hDC
, GetSysColorPen(COLOR_WINDOWFRAME
));
643 if (UserHasThickFrameStyle(Style
, ExStyle
))
645 UserDrawFrameNC(hDC
, &rect
, FALSE
, Active
);
646 wFrame
= GetSystemMetrics(SM_CXSIZEFRAME
);
648 else if (UserHasDlgFrameStyle(Style
, ExStyle
))
650 UserDrawFrameNC(hDC
, &rect
, TRUE
, Active
);
651 wFrame
= GetSystemMetrics(SM_CXDLGFRAME
);
653 if (Style
& WS_CAPTION
)
656 r
.bottom
= rect
.top
+ GetSystemMetrics(SM_CYSIZE
);
657 rect
.top
+= GetSystemMetrics(SM_CYSIZE
) +
658 GetSystemMetrics(SM_CYBORDER
);
659 UserDrawCaptionNC(hDC
, &r
, hWnd
, Style
, Active
);
663 if (UserHasMenu(hWnd
, Style
))
666 r
.bottom
= rect
.top
+ GetSystemMetrics(SM_CYMENU
);
669 rect
.top
+= MenuDrawMenuBar(hDC
, &r
, hWnd
, FALSE
);
672 /* Draw scrollbars */
673 if (Style
& WS_VSCROLL
)
674 SCROLL_DrawScrollBar(hWnd
, hDC
, SB_VERT
, TRUE
, TRUE
);
675 if (Style
& WS_HSCROLL
)
676 SCROLL_DrawScrollBar(hWnd
, hDC
, SB_HORZ
, TRUE
, TRUE
);
678 /* FIXME: Draw size box.*/
680 ReleaseDC(hWnd
, hDC
);
685 DefWndPaintNC(HWND hWnd
, HRGN clip
)
687 if (IsWindowVisible(hWnd
))
691 DefWndRedrawIconTitle(hWnd
);
695 DefWndDoPaintNC(hWnd
, clip
);
703 DefWndHitTestNC(HWND hWnd
, POINT Point
)
706 ULONG Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
707 ULONG ExStyle
= GetWindowLongW(hWnd
, GWL_EXSTYLE
);
709 GetWindowRect(hWnd
, &WindowRect
);
710 if (!PtInRect(&WindowRect
, Point
))
714 if (Style
& WS_MINIMIZE
)
718 if (UserHasThickFrameStyle(Style
, ExStyle
))
720 InflateRect(&WindowRect
, -GetSystemMetrics(SM_CXFRAME
),
721 -GetSystemMetrics(SM_CYFRAME
));
722 if (!PtInRect(&WindowRect
, Point
))
724 if (Point
.y
< WindowRect
.top
)
726 if (Point
.x
< (WindowRect
.left
+ GetSystemMetrics(SM_CXSIZE
)))
730 if (Point
.x
>= (WindowRect
.right
- GetSystemMetrics(SM_CXSIZE
)))
736 if (Point
.y
>= WindowRect
.bottom
)
738 if (Point
.x
< (WindowRect
.left
+ GetSystemMetrics(SM_CXSIZE
)))
740 return(HTBOTTOMLEFT
);
742 if (Point
.x
>= (WindowRect
.right
- GetSystemMetrics(SM_CXSIZE
)))
744 return(HTBOTTOMRIGHT
);
748 if (Point
.x
< WindowRect
.left
)
750 if (Point
.y
< (WindowRect
.top
+ GetSystemMetrics(SM_CYSIZE
)))
754 if (Point
.y
>= (WindowRect
.bottom
- GetSystemMetrics(SM_CYSIZE
)))
756 return(HTBOTTOMLEFT
);
760 if (Point
.x
>= WindowRect
.right
)
762 if (Point
.y
< (WindowRect
.top
+ GetSystemMetrics(SM_CYSIZE
)))
766 if (Point
.y
>= (WindowRect
.bottom
- GetSystemMetrics(SM_CYSIZE
)))
768 return(HTBOTTOMRIGHT
);
776 if (UserHasDlgFrameStyle(Style
, ExStyle
))
778 InflateRect(&WindowRect
, -GetSystemMetrics(SM_CXDLGFRAME
),
779 -GetSystemMetrics(SM_CYDLGFRAME
));
781 else if (UserHasThinFrameStyle(Style
, ExStyle
))
783 InflateRect(&WindowRect
, -GetSystemMetrics(SM_CXBORDER
),
784 -GetSystemMetrics(SM_CYBORDER
));
786 if (!PtInRect(&WindowRect
, Point
))
792 if ((Style
& WS_CAPTION
) == WS_CAPTION
)
794 WindowRect
.top
+= (GetSystemMetrics(SM_CYCAPTION
) -
795 GetSystemMetrics(SM_CYBORDER
));
796 if (!PtInRect(&WindowRect
, Point
))
798 if ((Style
& WS_SYSMENU
) && !(ExStyle
& WS_EX_TOOLWINDOW
))
800 WindowRect
.left
+= GetSystemMetrics(SM_CXSIZE
);
801 WindowRect
.right
-= GetSystemMetrics(SM_CXSIZE
) + 1;
803 if (Point
.x
<= WindowRect
.left
)
807 if (WindowRect
.right
<= Point
.x
)
812 if (Style
& WS_MAXIMIZEBOX
|| Style
& WS_MINIMIZEBOX
)
814 WindowRect
.right
-= GetSystemMetrics(SM_CXSIZE
);
816 if (Point
.x
>= WindowRect
.right
)
821 if (Style
& WS_MINIMIZEBOX
)
823 WindowRect
.right
-= GetSystemMetrics(SM_CXSIZE
);
825 if (Point
.x
>= WindowRect
.right
)
833 ScreenToClient(hWnd
, &Point
);
834 GetClientRect(hWnd
, &WindowRect
);
836 if (PtInRect(&WindowRect
, Point
))
841 if (Style
& WS_VSCROLL
)
843 WindowRect
.right
+= GetSystemMetrics(SM_CXVSCROLL
);
844 if (PtInRect(&WindowRect
, Point
))
850 if (Style
& WS_HSCROLL
)
852 WindowRect
.bottom
+= GetSystemMetrics(SM_CYHSCROLL
);
853 if (PtInRect(&WindowRect
, Point
))
855 if ((Style
& WS_VSCROLL
) &&
856 (Point
.x
>= (WindowRect
.right
- GetSystemMetrics(SM_CXVSCROLL
))))
858 return(HTBOTTOMRIGHT
);
864 if (UserHasMenu(hWnd
, Style
))
866 if (Point
.y
< 0 && Point
.x
>= 0 && Point
.x
<= WindowRect
.right
)
876 DefWndDoButton(HWND hWnd
, WPARAM wParam
)
879 BOOL InBtn
= TRUE
, HasBtn
= FALSE
;
881 WPARAM SCMsg
, CurBtn
= wParam
, OrigBtn
= wParam
;
886 Btn
= DFCS_CAPTIONCLOSE
;
888 HasBtn
= IsCloseBoxActive(hWnd
);
891 Btn
= DFCS_CAPTIONMIN
;
893 HasBtn
= IsMinBoxActive(hWnd
);
896 Btn
= DFCS_CAPTIONMAX
;
898 HasBtn
= IsMaxBoxActive(hWnd
);
908 UserDrawCaptionButton( hWnd
, GetWindowDC(hWnd
), HasBtn
, Btn
);
912 GetMessageW(&Msg
, 0, 0, 0);
926 CurBtn
= DefWndHitTestNC(hWnd
, Msg
.pt
);
927 if(InBtn
!= (CurBtn
== OrigBtn
))
929 UserDrawCaptionButton( hWnd
, GetWindowDC(hWnd
), (CurBtn
== OrigBtn
) , Btn
);
931 InBtn
= CurBtn
== OrigBtn
;
937 UserDrawCaptionButton( hWnd
, GetWindowDC(hWnd
), FALSE
, Btn
);
939 SendMessageA(hWnd
, WM_SYSCOMMAND
, SCMsg
, 0);
944 DefWndDoScrollBarDown(HWND hWnd
, UINT Msg
, WPARAM wParam
, LPARAM lParam
)
948 Point
.x
= SLOWORD(lParam
);
949 Point
.y
= SHIWORD(lParam
);
951 hit
= SCROLL_HitTest(hWnd
, (wParam
== HTHSCROLL
) ? SB_HORZ
: SB_VERT
, Point
, FALSE
);
954 DbgPrint("SCROLL_HitTest() == 0x%x\n", hit
);
956 SendMessageA(hWnd
, WM_SYSCOMMAND
, Msg
+ (UINT
)wParam
, lParam
);
960 DefWndHandleLButtonDownNC(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
966 HWND hTopWnd
= GetAncestor(hWnd
, GA_ROOT
);
967 if (SetActiveWindow(hTopWnd
) || GetActiveWindow() == hTopWnd
)
969 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_MOVE
+ HTCAPTION
, lParam
);
975 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_SYSMENU
)
977 if (!(GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
))
979 HDC hDC
= GetWindowDC(hWnd
);
980 // UserDrawSysMenuButton(hWnd, hDC, TRUE);
981 ReleaseDC(hWnd
, hDC
);
983 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_MOUSEMENU
+ HTSYSMENU
,
990 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_MOUSEMENU
+ HTMENU
, lParam
);
995 DefWndDoScrollBarDown(hWnd
, SC_HSCROLL
, HTHSCROLL
, lParam
);
996 //SendMessageA(hWnd, WM_SYSCOMMAND, SC_HSCROLL + HTHSCROLL, lParam);
1001 DefWndDoScrollBarDown(hWnd
, SC_VSCROLL
, HTVSCROLL
, lParam
);
1002 //SendMessageA(hWnd, WM_SYSCOMMAND, SC_VSCROLL + HTVSCROLL, lParam);
1009 DefWndDoButton(hWnd
, wParam
);
1021 SendMessageA(hWnd
, WM_SYSCOMMAND
, SC_SIZE
+ wParam
- 2, lParam
);
1030 DefWndHandleLButtonDblClkNC(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
1038 DefWndHandleLButtonUpNC(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
1046 DefWndHandleActiveNC(HWND hWnd
, WPARAM wParam
)
1054 DefWndSetRedraw(HWND hWnd
, WPARAM wParam
)
1061 DefWndHandleSetCursor(HWND hWnd
, WPARAM wParam
, LPARAM lParam
)
1063 /* Not for child windows. */
1064 if (hWnd
!= (HWND
)wParam
)
1069 switch(LOWORD(lParam
))
1073 WORD Msg
= HIWORD(lParam
);
1074 if (Msg
== WM_LBUTTONDOWN
|| Msg
== WM_MBUTTONDOWN
||
1075 Msg
== WM_RBUTTONDOWN
)
1084 HICON hCursor
= (HICON
)GetClassLongW(hWnd
, GCL_HCURSOR
);
1096 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZEWE
)));
1102 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZENS
)));
1108 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZENWSE
)));
1114 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_SIZENESW
)));
1117 return((LRESULT
)SetCursor(LoadCursorW(0, IDC_ARROW
)));
1121 DefWndStartSizeMove(HWND hWnd
, WPARAM wParam
, POINT
*capturePoint
)
1127 ULONG Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
1129 GetWindowRect(hWnd
, &rectWindow
);
1131 if ((wParam
& 0xfff0) == SC_MOVE
)
1133 /* Move pointer at the center of the caption */
1135 UserGetInsideRectNC(hWnd
, &rect
);
1136 if (Style
& WS_SYSMENU
)
1137 rect
.left
+= GetSystemMetrics(SM_CXSIZE
) + 1;
1138 if (Style
& WS_MINIMIZEBOX
)
1139 rect
.right
-= GetSystemMetrics(SM_CXSIZE
) + 1;
1140 if (Style
& WS_MAXIMIZEBOX
)
1141 rect
.right
-= GetSystemMetrics(SM_CXSIZE
) + 1;
1142 pt
.x
= rectWindow
.left
+ (rect
.right
- rect
.left
) / 2;
1143 pt
.y
= rectWindow
.top
+ rect
.top
+ GetSystemMetrics(SM_CYSIZE
)/2;
1144 hittest
= HTCAPTION
;
1151 GetMessageW(&msg
, NULL
, 0, 0);
1155 hittest
= DefWndHitTestNC(hWnd
, msg
.pt
);
1156 if ((hittest
< HTLEFT
) || (hittest
> HTBOTTOMRIGHT
))
1168 pt
.x
=(rectWindow
.left
+rectWindow
.right
)/2;
1169 pt
.y
= rectWindow
.top
+ GetSystemMetrics(SM_CYFRAME
) / 2;
1173 pt
.x
=(rectWindow
.left
+rectWindow
.right
)/2;
1174 pt
.y
= rectWindow
.bottom
- GetSystemMetrics(SM_CYFRAME
) / 2;
1178 pt
.x
= rectWindow
.left
+ GetSystemMetrics(SM_CXFRAME
) / 2;
1179 pt
.y
=(rectWindow
.top
+rectWindow
.bottom
)/2;
1183 pt
.x
= rectWindow
.right
- GetSystemMetrics(SM_CXFRAME
) / 2;
1184 pt
.y
=(rectWindow
.top
+rectWindow
.bottom
)/2;
1187 case VK_ESCAPE
: return 0;
1193 SetCursorPos( pt
.x
, pt
.y
);
1194 DefWndHandleSetCursor(hWnd
, (WPARAM
)hWnd
, MAKELONG(hittest
, WM_MOUSEMOVE
));
1198 #define ON_LEFT_BORDER(hit) \
1199 (((hit) == HTLEFT) || ((hit) == HTTOPLEFT) || ((hit) == HTBOTTOMLEFT))
1200 #define ON_RIGHT_BORDER(hit) \
1201 (((hit) == HTRIGHT) || ((hit) == HTTOPRIGHT) || ((hit) == HTBOTTOMRIGHT))
1202 #define ON_TOP_BORDER(hit) \
1203 (((hit) == HTTOP) || ((hit) == HTTOPLEFT) || ((hit) == HTTOPRIGHT))
1204 #define ON_BOTTOM_BORDER(hit) \
1205 (((hit) == HTBOTTOM) || ((hit) == HTBOTTOMLEFT) || ((hit) == HTBOTTOMRIGHT))
1208 UserDrawWindowFrame(HDC hdc
, const RECT
*rect
,
1209 ULONG width
, ULONG height
, DWORD rop
)
1211 HBRUSH hbrush
= SelectObject( hdc
, GetStockObject( GRAY_BRUSH
) );
1212 PatBlt( hdc
, rect
->left
, rect
->top
,
1213 rect
->right
- rect
->left
- width
, height
, rop
);
1214 PatBlt( hdc
, rect
->left
, rect
->top
+ height
, width
,
1215 rect
->bottom
- rect
->top
- height
, rop
);
1216 PatBlt( hdc
, rect
->left
+ width
, rect
->bottom
- 1,
1217 rect
->right
- rect
->left
- width
, -height
, rop
);
1218 PatBlt( hdc
, rect
->right
- 1, rect
->top
, -width
,
1219 rect
->bottom
- rect
->top
- height
, rop
);
1220 SelectObject( hdc
, hbrush
);
1224 UserDrawMovingFrame(HDC hdc
, RECT
*rect
, BOOL thickframe
)
1228 UserDrawWindowFrame(hdc
, rect
, GetSystemMetrics(SM_CXFRAME
),
1229 GetSystemMetrics(SM_CYFRAME
), PATINVERT
);
1231 else DrawFocusRect( hdc
, rect
);
1235 DefWndDoSizeMove(HWND hwnd
, WORD wParam
)
1238 RECT sizingRect
, mouseRect
, origRect
;
1240 LONG hittest
= (LONG
)(wParam
& 0x0f);
1241 HCURSOR hDragCursor
= 0, hOldCursor
= 0;
1242 POINT minTrack
, maxTrack
;
1243 POINT capturePoint
, pt
;
1244 ULONG Style
= GetWindowLongW(hwnd
, GWL_STYLE
);
1245 ULONG ExStyle
= GetWindowLongW(hwnd
, GWL_EXSTYLE
);
1246 BOOL thickframe
= UserHasThickFrameStyle(Style
, ExStyle
);
1247 BOOL iconic
= Style
& WS_MINIMIZE
;
1249 DWORD dwPoint
= GetMessagePos();
1250 BOOL DragFullWindows
= FALSE
;
1253 SystemParametersInfoA(SPI_GETDRAGFULLWINDOWS
, 0, &DragFullWindows
, 0);
1255 pt
.x
= SLOWORD(dwPoint
);
1256 pt
.y
= SHIWORD(dwPoint
);
1259 if (IsZoomed(hwnd
) || !IsWindowVisible(hwnd
))
1264 if ((wParam
& 0xfff0) == SC_MOVE
)
1268 hittest
= DefWndStartSizeMove(hwnd
, wParam
, &capturePoint
);
1281 if (hittest
&& hittest
!= HTSYSMENU
)
1288 hittest
= DefWndStartSizeMove(hwnd
, wParam
, &capturePoint
);
1297 if (Style
& WS_CHILD
)
1299 hWndParent
= GetParent(hwnd
);
1302 /* Get min/max info */
1304 WinPosGetMinMaxInfo(hwnd
, NULL
, NULL
, &minTrack
, &maxTrack
);
1305 GetWindowRect(hwnd
, &sizingRect
);
1306 origRect
= sizingRect
;
1307 if (Style
& WS_CHILD
)
1309 GetClientRect(hWndParent
, &mouseRect
);
1313 SetRect(&mouseRect
, 0, 0, GetSystemMetrics(SM_CXSCREEN
),
1314 GetSystemMetrics(SM_CYSCREEN
));
1316 if (ON_LEFT_BORDER(hittest
))
1318 mouseRect
.left
= max( mouseRect
.left
, sizingRect
.right
-maxTrack
.x
);
1319 mouseRect
.right
= min( mouseRect
.right
, sizingRect
.right
-minTrack
.x
);
1321 else if (ON_RIGHT_BORDER(hittest
))
1323 mouseRect
.left
= max( mouseRect
.left
, sizingRect
.left
+minTrack
.x
);
1324 mouseRect
.right
= min( mouseRect
.right
, sizingRect
.left
+maxTrack
.x
);
1326 if (ON_TOP_BORDER(hittest
))
1328 mouseRect
.top
= max( mouseRect
.top
, sizingRect
.bottom
-maxTrack
.y
);
1329 mouseRect
.bottom
= min( mouseRect
.bottom
,sizingRect
.bottom
-minTrack
.y
);
1331 else if (ON_BOTTOM_BORDER(hittest
))
1333 mouseRect
.top
= max( mouseRect
.top
, sizingRect
.top
+minTrack
.y
);
1334 mouseRect
.bottom
= min( mouseRect
.bottom
, sizingRect
.top
+maxTrack
.y
);
1336 if (Style
& WS_CHILD
)
1338 MapWindowPoints( hWndParent
, 0, (LPPOINT
)&mouseRect
, 2 );
1340 SendMessageA( hwnd
, WM_ENTERSIZEMOVE
, 0, 0 );
1342 if (GetCapture() != hwnd
) SetCapture( hwnd
);
1344 if (Style
& WS_CHILD
)
1346 /* Retrieve a default cache DC (without using the window style) */
1347 hdc
= GetDCEx(hWndParent
, 0, DCX_CACHE
);
1354 if( iconic
) /* create a cursor for dragging */
1356 HICON hIcon
= (HICON
)GetClassLongW(hwnd
, GCL_HICON
);
1357 if(!hIcon
) hIcon
= (HICON
)SendMessageW( hwnd
, WM_QUERYDRAGICON
, 0, 0L);
1358 if( hIcon
) hDragCursor
= CursorIconToCursor( hIcon
, TRUE
);
1359 if( !hDragCursor
) iconic
= FALSE
;
1362 /* invert frame if WIN31_LOOK to indicate mouse click on caption */
1363 if( !iconic
&& !DragFullWindows
)
1365 UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1372 GetMessageW(&msg
, 0, 0, 0);
1374 /* Exit on button-up, Return, or Esc */
1375 if ((msg
.message
== WM_LBUTTONUP
) ||
1376 ((msg
.message
== WM_KEYDOWN
) &&
1377 ((msg
.wParam
== VK_RETURN
) || (msg
.wParam
== VK_ESCAPE
)))) break;
1379 if (msg
.message
== WM_PAINT
)
1381 if(!iconic
&& !DragFullWindows
) UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1382 UpdateWindow( msg
.hwnd
);
1383 if(!iconic
&& !DragFullWindows
) UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1387 if ((msg
.message
!= WM_KEYDOWN
) && (msg
.message
!= WM_MOUSEMOVE
))
1388 continue; /* We are not interested in other messages */
1392 if (msg
.message
== WM_KEYDOWN
) switch(msg
.wParam
)
1394 case VK_UP
: pt
.y
-= 8; break;
1395 case VK_DOWN
: pt
.y
+= 8; break;
1396 case VK_LEFT
: pt
.x
-= 8; break;
1397 case VK_RIGHT
: pt
.x
+= 8; break;
1400 pt
.x
= max( pt
.x
, mouseRect
.left
);
1401 pt
.x
= min( pt
.x
, mouseRect
.right
);
1402 pt
.y
= max( pt
.y
, mouseRect
.top
);
1403 pt
.y
= min( pt
.y
, mouseRect
.bottom
);
1405 dx
= pt
.x
- capturePoint
.x
;
1406 dy
= pt
.y
- capturePoint
.y
;
1414 if( iconic
) /* ok, no system popup tracking */
1416 hOldCursor
= SetCursor(hDragCursor
);
1418 WinPosShowIconTitle( hwnd
, FALSE
);
1422 if (msg
.message
== WM_KEYDOWN
) SetCursorPos( pt
.x
, pt
.y
);
1425 RECT newRect
= sizingRect
;
1426 WPARAM wpSizingHit
= 0;
1428 if (hittest
== HTCAPTION
) OffsetRect( &newRect
, dx
, dy
);
1429 if (ON_LEFT_BORDER(hittest
)) newRect
.left
+= dx
;
1430 else if (ON_RIGHT_BORDER(hittest
)) newRect
.right
+= dx
;
1431 if (ON_TOP_BORDER(hittest
)) newRect
.top
+= dy
;
1432 else if (ON_BOTTOM_BORDER(hittest
)) newRect
.bottom
+= dy
;
1433 if(!iconic
&& !DragFullWindows
) UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1436 /* determine the hit location */
1437 if (hittest
>= HTLEFT
&& hittest
<= HTBOTTOMRIGHT
)
1438 wpSizingHit
= WMSZ_LEFT
+ (hittest
- HTLEFT
);
1439 SendMessageA( hwnd
, WM_SIZING
, wpSizingHit
, (LPARAM
)&newRect
);
1443 if(!DragFullWindows
)
1444 UserDrawMovingFrame( hdc
, &newRect
, thickframe
);
1446 /* To avoid any deadlocks, all the locks on the windows
1447 structures must be suspended before the SetWindowPos */
1448 SetWindowPos( hwnd
, 0, newRect
.left
, newRect
.top
,
1449 newRect
.right
- newRect
.left
,
1450 newRect
.bottom
- newRect
.top
,
1451 ( hittest
== HTCAPTION
) ? SWP_NOSIZE
: 0 );
1454 sizingRect
= newRect
;
1462 if( moved
) /* restore cursors, show icon title later on */
1464 ShowCursor( FALSE
);
1465 SetCursor( hOldCursor
);
1467 DestroyCursor( hDragCursor
);
1469 else if(!DragFullWindows
)
1470 UserDrawMovingFrame( hdc
, &sizingRect
, thickframe
);
1472 if (Style
& WS_CHILD
)
1473 ReleaseDC( hWndParent
, hdc
);
1475 ReleaseDC( 0, hdc
);
1477 SendMessageA( hwnd
, WM_EXITSIZEMOVE
, 0, 0 );
1478 SendMessageA( hwnd
, WM_SETVISIBLE
, !IsIconic(hwnd
), 0L);
1480 /* window moved or resized */
1483 /* if the moving/resizing isn't canceled call SetWindowPos
1484 * with the new position or the new size of the window
1486 if (!((msg
.message
== WM_KEYDOWN
) && (msg
.wParam
== VK_ESCAPE
)) )
1488 /* NOTE: SWP_NOACTIVATE prevents document window activation in Word 6 */
1489 if(!DragFullWindows
)
1490 SetWindowPos( hwnd
, 0, sizingRect
.left
, sizingRect
.top
,
1491 sizingRect
.right
- sizingRect
.left
,
1492 sizingRect
.bottom
- sizingRect
.top
,
1493 ( hittest
== HTCAPTION
) ? SWP_NOSIZE
: 0 );
1495 else { /* restore previous size/position */
1497 SetWindowPos( hwnd
, 0, origRect
.left
, origRect
.top
,
1498 origRect
.right
- origRect
.left
,
1499 origRect
.bottom
- origRect
.top
,
1500 ( hittest
== HTCAPTION
) ? SWP_NOSIZE
: 0 );
1504 if( IsWindow(hwnd
) )
1505 if( Style
& WS_MINIMIZE
)
1507 /* Single click brings up the system menu when iconized */
1511 if( Style
& WS_SYSMENU
)
1512 SendMessageA( hwnd
, WM_SYSCOMMAND
,
1513 SC_MOUSEMENU
+ HTSYSMENU
, MAKELONG(pt
.x
,pt
.y
));
1515 else WinPosShowIconTitle( hwnd
, TRUE
);
1521 DefWndHandleSysCommand(HWND hWnd
, WPARAM wParam
, POINT Pt
)
1523 switch (wParam
& 0xfff0)
1527 DefWndDoSizeMove(hWnd
, wParam
);
1530 SendMessageA(hWnd
, WM_CLOSE
, 0, 0);
1533 MenuTrackMouseMenuBar(hWnd
, wParam
, Pt
);
1536 MenuTrackKbdMenuBar(hWnd
, wParam
, Pt
.x
);
1539 /* FIXME: Implement */
1549 DefWndAdjustRect(RECT
* Rect
, ULONG Style
, BOOL Menu
, ULONG ExStyle
)
1551 if (Style
& WS_ICONIC
)
1556 if (UserHasThickFrameStyle(Style
, ExStyle
))
1558 InflateRect(Rect
, GetSystemMetrics(SM_CXFRAME
),
1559 GetSystemMetrics(SM_CYFRAME
));
1561 else if (UserHasDlgFrameStyle(Style
, ExStyle
))
1563 InflateRect(Rect
, GetSystemMetrics(SM_CXDLGFRAME
),
1564 GetSystemMetrics(SM_CYDLGFRAME
));
1566 else if (UserHasThinFrameStyle(Style
, ExStyle
))
1568 InflateRect(Rect
, GetSystemMetrics(SM_CXBORDER
),
1569 GetSystemMetrics(SM_CYBORDER
));
1571 if (Style
& WS_CAPTION
)
1573 Rect
->top
-= (GetSystemMetrics(SM_CYCAPTION
) -
1574 GetSystemMetrics(SM_CYBORDER
)) + 1;
1578 Rect
->top
-= GetSystemMetrics(SM_CYMENU
) + GetSystemMetrics(SM_CYBORDER
);
1580 if (Style
& WS_VSCROLL
)
1582 Rect
->right
+= GetSystemMetrics(SM_CXVSCROLL
) - 1;
1583 if (UserHasAnyFrameStyle(Style
, ExStyle
))
1588 if (Style
& WS_HSCROLL
)
1590 Rect
->bottom
+= GetSystemMetrics(SM_CYHSCROLL
) - 1;
1591 if (UserHasAnyFrameStyle(Style
, ExStyle
))
1599 DefWndNCCalcSize(HWND hWnd
, RECT
* Rect
)
1602 LONG Style
= GetClassLongW(hWnd
, GCL_STYLE
);
1603 RECT TmpRect
= {0, 0, 0, 0};
1605 if (Style
& CS_VREDRAW
)
1607 Result
|= WVR_VREDRAW
;
1609 if (Style
& CS_HREDRAW
)
1611 Result
|= WVR_HREDRAW
;
1614 if (!(GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
))
1616 DefWndAdjustRect(&TmpRect
, GetWindowLongW(hWnd
, GWL_STYLE
),
1617 FALSE
, GetWindowLongW(hWnd
, GWL_EXSTYLE
));
1618 Rect
->left
-= TmpRect
.left
;
1619 Rect
->top
-= TmpRect
.top
;
1620 Rect
->right
-= TmpRect
.right
;
1621 Rect
->bottom
-= TmpRect
.bottom
;
1622 if (UserHasMenu(hWnd
, GetWindowLongW(hWnd
, GWL_STYLE
)))
1624 Rect
->top
+= MenuGetMenuBarHeight(hWnd
, Rect
->right
- Rect
->left
,
1625 -TmpRect
.left
, -TmpRect
.top
) + 1;
1627 if (Rect
->top
> Rect
->bottom
)
1628 Rect
->bottom
= Rect
->top
;
1629 if (Rect
->left
> Rect
->right
)
1630 Rect
->right
= Rect
->left
;
1637 DefWndHandleWindowPosChanging(HWND hWnd
, WINDOWPOS
* Pos
)
1639 POINT maxSize
, minTrack
;
1640 LONG style
= GetWindowLongA(hWnd
, GWL_STYLE
);
1642 if (Pos
->flags
& SWP_NOSIZE
) return 0;
1643 if ((style
& WS_THICKFRAME
) || ((style
& (WS_POPUP
| WS_CHILD
)) == 0))
1645 WinPosGetMinMaxInfo(hWnd
, &maxSize
, NULL
, &minTrack
, NULL
);
1646 Pos
->cx
= min(Pos
->cx
, maxSize
.x
);
1647 Pos
->cy
= min(Pos
->cy
, maxSize
.y
);
1648 if (!(style
& WS_MINIMIZE
))
1650 if (Pos
->cx
< minTrack
.x
) Pos
->cx
= minTrack
.x
;
1651 if (Pos
->cy
< minTrack
.y
) Pos
->cy
= minTrack
.y
;
1657 /* Undocumented flags. */
1658 #define SWP_NOCLIENTMOVE 0x0800
1659 #define SWP_NOCLIENTSIZE 0x1000
1662 DefWndHandleWindowPosChanged(HWND hWnd
, WINDOWPOS
* Pos
)
1666 GetClientRect(hWnd
, &rect
);
1668 if (!(Pos
->flags
& SWP_NOCLIENTMOVE
))
1669 SendMessageW(hWnd
, WM_MOVE
, 0, MAKELONG(rect
.left
, rect
.top
));
1671 if (!(Pos
->flags
& SWP_NOCLIENTSIZE
))
1673 WPARAM wp
= SIZE_RESTORED
;
1674 if (IsZoomed(hWnd
)) wp
= SIZE_MAXIMIZED
;
1675 else if (IsIconic(hWnd
)) wp
= SIZE_MINIMIZED
;
1676 SendMessageW(hWnd
, WM_SIZE
, wp
,
1677 MAKELONG(rect
.right
- rect
.left
, rect
.bottom
- rect
.top
));
1685 User32DefWindowProc(HWND hWnd
,
1695 return (DefWndPaintNC(hWnd
, (HRGN
)wParam
));
1700 return (DefWndNCCalcSize(hWnd
, (RECT
*)lParam
));
1703 case WM_WINDOWPOSCHANGING
:
1705 return (DefWndHandleWindowPosChanging(hWnd
, (WINDOWPOS
*)lParam
));
1708 case WM_WINDOWPOSCHANGED
:
1710 return (DefWndHandleWindowPosChanged(hWnd
, (WINDOWPOS
*)lParam
));
1716 Point
.x
= SLOWORD(lParam
);
1717 Point
.y
= SHIWORD(lParam
);
1718 return (DefWndHitTestNC(hWnd
, Point
));
1721 case WM_NCLBUTTONDOWN
:
1723 return (DefWndHandleLButtonDownNC(hWnd
, wParam
, lParam
));
1726 case WM_NCLBUTTONUP
:
1728 return (DefWndHandleLButtonUpNC(hWnd
, wParam
, lParam
));
1731 case WM_LBUTTONDBLCLK
:
1732 case WM_NCLBUTTONDBLCLK
:
1734 return (DefWndHandleLButtonDblClkNC(hWnd
, wParam
, lParam
));
1737 case WM_NCRBUTTONDOWN
:
1739 if (wParam
== HTCAPTION
)
1749 if (hWnd
== GetCapture())
1753 Pt
.x
= SLOWORD(lParam
);
1754 Pt
.y
= SHIWORD(lParam
);
1755 ClientToScreen(hWnd
, &Pt
);
1756 lParam
= MAKELPARAM(Pt
.x
, Pt
.y
);
1759 SendMessageW(hWnd
, WM_CONTEXTMENU
, (WPARAM
)hWnd
, lParam
);
1763 SendMessageA(hWnd
, WM_CONTEXTMENU
, (WPARAM
)hWnd
, lParam
);
1768 case WM_CONTEXTMENU
:
1770 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1774 SendMessageW(GetParent(hWnd
), Msg
, wParam
, lParam
);
1778 SendMessageA(GetParent(hWnd
), WM_CONTEXTMENU
, wParam
, lParam
);
1786 Pt
.x
= SLOWORD(lParam
);
1787 Pt
.y
= SHIWORD(lParam
);
1788 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1790 ScreenToClient(GetParent(hWnd
), &Pt
);
1793 HitCode
= DefWndHitTestNC(hWnd
, Pt
);
1795 if (HitCode
== HTCAPTION
|| HitCode
== HTSYSMENU
)
1797 TrackPopupMenu(GetSystemMenu(hWnd
, FALSE
),
1798 TPM_LEFTBUTTON
| TPM_RIGHTBUTTON
,
1799 Pt
.x
, Pt
.y
, 0, hWnd
, NULL
);
1807 return (DefWndHandleActiveNC(hWnd
, wParam
));
1812 /* FIXME: Implement. */
1820 HDC hDC
= BeginPaint(hWnd
, &Ps
);
1824 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
&&
1825 (hIcon
= (HICON
)GetClassLongW(hWnd
, GCL_HICON
)) != NULL
)
1829 GetWindowRect(hWnd
, &WindowRect
);
1830 x
= (WindowRect
.right
- WindowRect
.left
-
1831 GetSystemMetrics(SM_CXICON
)) / 2;
1832 y
= (WindowRect
.bottom
- WindowRect
.top
-
1833 GetSystemMetrics(SM_CYICON
)) / 2;
1834 DrawIcon(hDC
, x
, y
, hIcon
);
1836 if (GetWindowLongW(hWnd
, GWL_EXSTYLE
) & WS_EX_CLIENTEDGE
)
1839 GetClientRect(hWnd
, &WindowRect
);
1840 DrawEdge(hDC
, &WindowRect
, EDGE_SUNKEN
, BF_RECT
);
1842 EndPaint(hWnd
, &Ps
);
1850 hRgn
= CreateRectRgn(0, 0, 0, 0);
1851 if (GetUpdateRgn(hWnd
, hRgn
, FALSE
) != NULLREGION
)
1853 RedrawWindow(hWnd
, NULL
, hRgn
,
1854 RDW_ERASENOW
| RDW_ERASE
| RDW_FRAME
|
1863 DefWndSetRedraw(hWnd
, wParam
);
1869 DestroyWindow(hWnd
);
1873 case WM_MOUSEACTIVATE
:
1875 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1880 Ret
= SendMessageW(GetParent(hWnd
), WM_MOUSEACTIVATE
,
1885 Ret
= SendMessageA(GetParent(hWnd
), WM_MOUSEACTIVATE
,
1893 return ((LOWORD(lParam
) >= HTCLIENT
) ? MA_ACTIVATE
: MA_NOACTIVATE
);
1898 /* Check if the window is minimized. */
1899 if (LOWORD(lParam
) != WA_INACTIVE
&&
1900 !(GetWindowLongW(hWnd
, GWL_STYLE
) & WS_MINIMIZE
))
1909 if (GetWindowLongW(hWnd
, GWL_STYLE
& WS_CHILD
))
1913 return (SendMessageW(GetParent(hWnd
), WM_MOUSEWHEEL
,
1918 return (SendMessageA(GetParent(hWnd
), WM_MOUSEWHEEL
,
1926 case WM_ICONERASEBKGND
:
1929 HBRUSH hBrush
= (HBRUSH
)GetClassLongW(hWnd
, GCL_HBRBACKGROUND
);
1935 if (0 == (((DWORD
) hBrush
) & 0xffff0000))
1937 hBrush
= GetSysColorBrush((DWORD
) hBrush
- 1);
1939 GetClipBox((HDC
)wParam
, &Rect
);
1940 FillRect((HDC
)wParam
, &Rect
, hBrush
);
1944 /* FIXME: Implement colour controls. */
1946 case WM_CTLCOLORMSGBOX:
1947 case WM_CTLCOLOREDIT:
1948 case WM_CTLCOLORLISTBOX:
1949 case WM_CTLCOLORBTN:
1950 case WM_CTLCOLORDLG:
1951 case WM_CTLCOLORSTATIC:
1952 case WM_CTLCOLORSCROLLBAR:
1958 if (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CHILD
)
1960 if (LOWORD(lParam
) < HTLEFT
|| LOWORD(lParam
) > HTBOTTOMRIGHT
)
1965 bResult
= SendMessageW(GetParent(hWnd
), WM_SETCURSOR
,
1970 bResult
= SendMessageA(GetParent(hWnd
), WM_SETCURSOR
,
1979 return (DefWndHandleSetCursor(hWnd
, wParam
, lParam
));
1985 Pt
.x
= SLOWORD(lParam
);
1986 Pt
.y
= SHIWORD(lParam
);
1987 return (DefWndHandleSysCommand(hWnd
, wParam
, Pt
));
1990 /* FIXME: Handle key messages. */
1998 /* FIXME: This is also incomplete. */
2001 if (HIWORD(lParam
) & KEYDATA_ALT
)
2003 if (wParam
== VK_F4
) /* Try to close the window */
2005 HWND top
= GetAncestor(hWnd
, GA_ROOT
);
2006 if (!(GetClassLongW(top
, GCL_STYLE
) & CS_NOCLOSE
))
2009 PostMessageW(top
, WM_SYSCOMMAND
, SC_CLOSE
, 0);
2011 PostMessageA(top
, WM_SYSCOMMAND
, SC_CLOSE
, 0);
2024 Style
= GetWindowLongW(hWnd
, GWL_STYLE
);
2025 if (!(Style
& WS_POPUP
))
2027 if ((Style
& WS_VISIBLE
) && wParam
)
2029 if (!(Style
& WS_VISIBLE
) && !wParam
)
2031 if (!GetWindow(hWnd
, GW_OWNER
))
2033 ShowWindow(hWnd
, wParam
? SW_SHOWNA
: SW_HIDE
);
2039 /* FIXME: Check for a desktop. */
2040 if (GetCapture() == hWnd
)
2052 /* FIXME: Implement this. */
2055 case WM_QUERYDROPOBJECT
:
2057 if (GetWindowLongW(hWnd
, GWL_EXSTYLE
) & WS_EX_ACCEPTFILES
)
2064 case WM_QUERYDRAGICON
:
2069 hIcon
= (HICON
)GetClassLongW(hWnd
, GCL_HICON
);
2072 return ((LRESULT
)hIcon
);
2074 for (Len
= 1; Len
< 64; Len
++)
2076 if ((hIcon
= LoadIconW(NULL
, MAKEINTRESOURCEW(Len
))) != NULL
)
2078 return((LRESULT
)hIcon
);
2081 return ((LRESULT
)LoadIconW(0, IDI_APPLICATION
));
2084 /* FIXME: WM_ISACTIVEICON */
2086 case WM_NOTIFYFORMAT
:
2088 if (IsWindowUnicode(hWnd
))
2090 return(NFR_UNICODE
);
2100 INT Index
= (wParam
!= 0) ? GCL_HICON
: GCL_HICONSM
;
2101 HICON hOldIcon
= (HICON
)GetClassLongW(hWnd
, Index
);
2102 SetClassLongW(hWnd
, Index
, lParam
);
2103 SetWindowPos(hWnd
, 0, 0, 0, 0, 0,
2104 SWP_FRAMECHANGED
| SWP_NOSIZE
| SWP_NOMOVE
|
2105 SWP_NOACTIVATE
| SWP_NOZORDER
);
2106 return ((LRESULT
)hOldIcon
);
2111 INT Index
= (wParam
!= 0) ? GCL_HICON
: GCL_HICONSM
;
2112 return (GetClassLongW(hWnd
, Index
));
2119 SendMessageW(GetParent(hWnd
), Msg
, wParam
, lParam
);
2123 SendMessageA(GetParent(hWnd
), Msg
, wParam
, lParam
);
2129 case WM_QUERYENDSESSION
:
2139 DefWindowProcA(HWND hWnd
,
2144 static LPSTR WindowTextAtom
= 0;
2151 CREATESTRUCTA
*Cs
= (CREATESTRUCTA
*)lParam
;
2152 if (HIWORD(Cs
->lpszName
))
2154 if (0 == WindowTextAtom
)
2157 (LPSTR
)(ULONG
)GlobalAddAtomA("USER32!WindowTextAtomA");
2159 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2160 strlen(Cs
->lpszName
) * sizeof(CHAR
));
2161 strcpy(WindowText
, Cs
->lpszName
);
2162 SetPropA(hWnd
, WindowTextAtom
, WindowText
);
2167 case WM_GETTEXTLENGTH
:
2169 if (WindowTextAtom
== 0 ||
2170 (WindowText
= GetPropA(hWnd
, WindowTextAtom
)) == NULL
)
2174 return (strlen(WindowText
));
2179 if (WindowTextAtom
== 0 ||
2180 (WindowText
= GetPropA(hWnd
, WindowTextAtom
)) == NULL
)
2184 *((PSTR
)lParam
) = '\0';
2188 strncpy((LPSTR
)lParam
, WindowText
, wParam
);
2189 return (min(wParam
, strlen(WindowText
)));
2194 if (0 == WindowTextAtom
)
2197 (LPSTR
)(DWORD
)GlobalAddAtomA("USER32!WindowTextAtomA");
2199 if (WindowTextAtom
!= 0 &&
2200 (WindowText
= GetPropA(hWnd
, WindowTextAtom
)) == NULL
)
2202 RtlFreeHeap(RtlGetProcessHeap(), 0, WindowText
);
2204 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2205 (strlen((PSTR
)lParam
) + 1) * sizeof(CHAR
));
2206 strcpy(WindowText
, (PSTR
)lParam
);
2207 SetPropA(hWnd
, WindowTextAtom
, WindowText
);
2208 if (0 != (GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CAPTION
))
2210 DefWndPaintNC(hWnd
, (HRGN
) 1);
2216 FIXME: Implement these.
2218 case WM_IME_KEYDOWN:
2220 case WM_IME_STARTCOMPOSITION:
2221 case WM_IME_COMPOSITION:
2222 case WM_IME_ENDCOMPOSITION:
2224 case WM_IME_SETCONTEXT:
2229 if (WindowTextAtom
!= 0 &&
2230 (WindowText
= RemovePropA(hWnd
, WindowTextAtom
)) == NULL
)
2232 RtlFreeHeap(GetProcessHeap(), 0, WindowText
);
2238 return User32DefWindowProc(hWnd
, Msg
, wParam
, lParam
, FALSE
);
2243 DefWindowProcW(HWND hWnd
,
2248 static LPWSTR WindowTextAtom
= 0;
2255 CREATESTRUCTW
* CreateStruct
= (CREATESTRUCTW
*)lParam
;
2256 if (HIWORD(CreateStruct
->lpszName
))
2258 if (0 == WindowTextAtom
)
2261 (LPWSTR
)(DWORD
)GlobalAddAtomW(L
"USER32!WindowTextAtomW");
2263 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2264 wcslen(CreateStruct
->lpszName
) * sizeof(WCHAR
));
2265 wcscpy(WindowText
, CreateStruct
->lpszName
);
2266 SetPropW(hWnd
, WindowTextAtom
, WindowText
);
2271 case WM_GETTEXTLENGTH
:
2273 if (WindowTextAtom
== 0 ||
2274 (WindowText
= GetPropW(hWnd
, WindowTextAtom
)) == NULL
)
2278 return (wcslen(WindowText
));
2283 if (WindowTextAtom
== 0 ||
2284 (WindowText
= GetPropW(hWnd
, WindowTextAtom
)) == NULL
)
2288 ((PWSTR
)lParam
) = '\0';
2292 wcsncpy((PWSTR
)lParam
, WindowText
, wParam
);
2293 return (min(wParam
, wcslen(WindowText
)));
2298 if (WindowTextAtom
== 0)
2301 (LPWSTR
)(DWORD
)GlobalAddAtomW(L
"USER32!WindowTextAtomW");
2303 if (WindowTextAtom
!= 0 &&
2304 (WindowText
= GetPropW(hWnd
, WindowTextAtom
)) == NULL
)
2306 RtlFreeHeap(RtlGetProcessHeap(), 0, WindowText
);
2308 WindowText
= RtlAllocateHeap(RtlGetProcessHeap(), 0,
2309 (wcslen((PWSTR
)lParam
) + 1) * sizeof(WCHAR
));
2310 wcscpy(WindowText
, (PWSTR
)lParam
);
2311 SetPropW(hWnd
, WindowTextAtom
, WindowText
);
2312 if ((GetWindowLongW(hWnd
, GWL_STYLE
) & WS_CAPTION
) == WS_CAPTION
)
2314 DefWndPaintNC(hWnd
, (HRGN
)1);
2321 SendMessageW(hWnd
, WM_CHAR
, wParam
, lParam
);
2325 case WM_IME_SETCONTEXT
:
2333 if (WindowTextAtom
!= 0 &&
2334 (WindowText
= RemovePropW(hWnd
, WindowTextAtom
)) == NULL
)
2336 RtlFreeHeap(RtlGetProcessHeap(), 0, WindowText
);
2342 return User32DefWindowProc(hWnd
, Msg
, wParam
, lParam
, TRUE
);