2 * Non-client area window functions
4 * Copyright 1994 Alexandre Julliard
7 static char Copyright
[] = "Copyright Alexandre Julliard, 1994";
11 #include "sysmetrics.h"
16 static HBITMAP hbitmapClose
= 0;
17 static HBITMAP hbitmapMinimize
= 0;
18 static HBITMAP hbitmapMinimizeD
= 0;
19 static HBITMAP hbitmapMaximize
= 0;
20 static HBITMAP hbitmapMaximizeD
= 0;
21 static HBITMAP hbitmapRestore
= 0;
22 static HBITMAP hbitmapRestoreD
= 0;
24 extern void NC_TrackSysMenu( HWND hwnd
); /* menu.c */
25 extern void WINPOS_GetMinMaxInfo( HWND hwnd
, POINT
*maxSize
, POINT
*maxPos
,
26 POINT
*minTrack
, POINT
*maxTrack
); /* winpos.c */
28 extern Display
* display
;
31 /* Some useful macros */
32 #define HAS_DLGFRAME(style,exStyle) \
33 (((style) & WS_DLGFRAME) && \
34 (((exStyle) & WS_EX_DLGMODALFRAME) || !((style) & WS_BORDER)))
36 #define HAS_THICKFRAME(style) \
37 (((style) & WS_THICKFRAME) && \
38 !(((style) & (WS_DLGFRAME|WS_BORDER)) == WS_DLGFRAME))
40 #define HAS_MENU(w) (!((w)->dwStyle & WS_CHILD) && ((w)->wIDmenu != 0))
42 /***********************************************************************
45 * Compute the size of the window rectangle from the size of the
48 static void NC_AdjustRect( LPRECT rect
, DWORD style
, BOOL menu
, DWORD exStyle
)
50 if (HAS_DLGFRAME( style
, exStyle
))
51 InflateRect( rect
, SYSMETRICS_CXDLGFRAME
, SYSMETRICS_CYDLGFRAME
);
54 if (HAS_THICKFRAME(style
))
55 InflateRect( rect
, SYSMETRICS_CXFRAME
, SYSMETRICS_CYFRAME
);
56 if (style
& WS_BORDER
)
57 InflateRect( rect
, SYSMETRICS_CXBORDER
, SYSMETRICS_CYBORDER
);
60 if ((style
& WS_CAPTION
) == WS_CAPTION
)
61 rect
->top
-= SYSMETRICS_CYCAPTION
- 1;
63 if (menu
) rect
->top
-= SYSMETRICS_CYMENU
+ 1;
65 if (style
& WS_VSCROLL
) rect
->right
+= SYSMETRICS_CXVSCROLL
;
66 if (style
& WS_HSCROLL
) rect
->bottom
+= SYSMETRICS_CYHSCROLL
;
70 /***********************************************************************
71 * AdjustWindowRect (USER.102)
73 void AdjustWindowRect( LPRECT rect
, DWORD style
, BOOL menu
)
75 AdjustWindowRectEx( rect
, style
, menu
, 0 );
79 /***********************************************************************
80 * AdjustWindowRectEx (USER.454)
82 void AdjustWindowRectEx( LPRECT rect
, DWORD style
, BOOL menu
, DWORD exStyle
)
84 /* Correct the window style */
86 if (!(style
& (WS_POPUP
| WS_CHILD
))) /* Overlapped window */
88 if (exStyle
& WS_EX_DLGMODALFRAME
) style
&= ~WS_THICKFRAME
;
90 #ifdef DEBUG_NONCLIENT
91 printf( "AdjustWindowRectEx: (%d,%d)-(%d,%d) %08x %d %08x\n",
92 rect
->left
, rect
->top
, rect
->right
, rect
->bottom
, style
, menu
, exStyle
);
95 NC_AdjustRect( rect
, style
, menu
, exStyle
);
99 /***********************************************************************
100 * NC_HandleNCCalcSize
102 * Handle a WM_NCCALCSIZE message. Called from DefWindowProc().
104 LONG
NC_HandleNCCalcSize( HWND hwnd
, NCCALCSIZE_PARAMS
*params
)
106 RECT tmpRect
= { 0, 0, 0, 0 };
107 WND
*wndPtr
= WIN_FindWndPtr( hwnd
);
109 if (!wndPtr
) return 0;
111 NC_AdjustRect( &tmpRect
, wndPtr
->dwStyle
,
112 HAS_MENU(wndPtr
), wndPtr
->dwExStyle
);
114 params
->rgrc
[0].left
-= tmpRect
.left
;
115 params
->rgrc
[0].top
-= tmpRect
.top
;
116 params
->rgrc
[0].right
-= tmpRect
.right
;
117 params
->rgrc
[0].bottom
-= tmpRect
.bottom
;
122 /***********************************************************************
125 * Get the 'inside' rectangle of a window, i.e. the whole window rectangle
126 * but without the borders (if any).
127 * The rectangle is in window coordinates (for drawing with GetWindowDC()).
129 static void NC_GetInsideRect( HWND hwnd
, RECT
*rect
)
131 WND
* wndPtr
= WIN_FindWndPtr( hwnd
);
133 rect
->top
= rect
->left
= 0;
134 rect
->right
= wndPtr
->rectWindow
.right
- wndPtr
->rectWindow
.left
;
135 rect
->bottom
= wndPtr
->rectWindow
.bottom
- wndPtr
->rectWindow
.top
;
137 /* Remove frame from rectangle */
138 if (HAS_DLGFRAME( wndPtr
->dwStyle
, wndPtr
->dwExStyle
))
140 InflateRect( rect
, -SYSMETRICS_CXDLGFRAME
, -SYSMETRICS_CYDLGFRAME
);
141 if (wndPtr
->dwExStyle
& WS_EX_DLGMODALFRAME
) InflateRect( rect
, -1, 0);
145 if (HAS_THICKFRAME( wndPtr
->dwStyle
))
146 InflateRect( rect
, -SYSMETRICS_CXFRAME
, -SYSMETRICS_CYFRAME
);
147 if (wndPtr
->dwStyle
& WS_BORDER
)
148 InflateRect( rect
, -SYSMETRICS_CXBORDER
, -SYSMETRICS_CYBORDER
);
153 /***********************************************************************
154 * NC_InternalNCHitTest
156 * Perform the hit test calculation, but whithout testing the capture
159 static LONG
NC_InternalNCHitTest( HWND hwnd
, POINT pt
)
162 WND
*wndPtr
= WIN_FindWndPtr( hwnd
);
163 if (!wndPtr
) return HTERROR
;
165 GetWindowRect( hwnd
, &rect
);
166 if (!PtInRect( &rect
, pt
)) return HTNOWHERE
;
167 ScreenToClient( hwnd
, &pt
);
168 GetClientRect( hwnd
, &rect
);
170 if (PtInRect( &rect
, pt
)) return HTCLIENT
;
172 /* Check vertical scroll bar */
173 if (wndPtr
->dwStyle
& WS_VSCROLL
)
175 rect
.right
+= SYSMETRICS_CXVSCROLL
;
176 if (PtInRect( &rect
, pt
)) return HTVSCROLL
;
179 /* Check horizontal scroll bar */
180 if (wndPtr
->dwStyle
& WS_HSCROLL
)
182 rect
.bottom
+= SYSMETRICS_CYHSCROLL
;
183 if (PtInRect( &rect
, pt
))
186 if ((wndPtr
->dwStyle
& WS_VSCROLL
) &&
187 (pt
.x
>= rect
.right
- SYSMETRICS_CXVSCROLL
)) return HTSIZE
;
193 if (HAS_MENU(wndPtr
))
195 rect
.top
-= SYSMETRICS_CYMENU
+ 1;
196 if (PtInRect( &rect
, pt
)) return HTMENU
;
200 if ((wndPtr
->dwStyle
& WS_CAPTION
) == WS_CAPTION
)
202 rect
.top
-= SYSMETRICS_CYCAPTION
- 1;
203 if (PtInRect( &rect
, pt
))
205 /* Check system menu */
206 if ((wndPtr
->dwStyle
& WS_SYSMENU
) && (pt
.x
<= SYSMETRICS_CXSIZE
))
208 /* Check maximize box */
209 if (wndPtr
->dwStyle
& WS_MAXIMIZEBOX
)
210 rect
.right
-= SYSMETRICS_CXSIZE
+ 1;
211 if (pt
.x
>= rect
.right
) return HTMAXBUTTON
;
212 /* Check minimize box */
213 if (wndPtr
->dwStyle
& WS_MINIMIZEBOX
)
214 rect
.right
-= SYSMETRICS_CXSIZE
+ 1;
215 if (pt
.x
>= rect
.right
) return HTMINBUTTON
;
220 /* Check non-sizing border */
221 if (!HAS_THICKFRAME( wndPtr
->dwStyle
)) return HTBORDER
;
223 /* Check top sizing border */
226 if (pt
.x
< rect
.left
+SYSMETRICS_CXSIZE
) return HTTOPLEFT
;
227 if (pt
.x
>= rect
.right
-SYSMETRICS_CXSIZE
) return HTTOPRIGHT
;
231 /* Check bottom sizing border */
232 if (pt
.y
>= rect
.bottom
)
234 if (pt
.x
< rect
.left
+SYSMETRICS_CXSIZE
) return HTBOTTOMLEFT
;
235 if (pt
.x
>= rect
.right
-SYSMETRICS_CXSIZE
) return HTBOTTOMRIGHT
;
239 /* Check left sizing border */
240 if (pt
.x
< rect
.left
)
242 if (pt
.y
< rect
.top
+SYSMETRICS_CYSIZE
) return HTTOPLEFT
;
243 if (pt
.y
>= rect
.bottom
-SYSMETRICS_CYSIZE
) return HTBOTTOMLEFT
;
247 /* Check right sizing border */
248 if (pt
.x
>= rect
.right
)
250 if (pt
.y
< rect
.top
+SYSMETRICS_CYSIZE
) return HTTOPRIGHT
;
251 if (pt
.y
>= rect
.bottom
-SYSMETRICS_CYSIZE
) return HTBOTTOMRIGHT
;
255 /* Should never get here */
260 /***********************************************************************
263 * Handle a WM_NCHITTEST message. Called from DefWindowProc().
265 LONG
NC_HandleNCHitTest( HWND hwnd
, POINT pt
)
267 #ifdef DEBUG_NONCLIENT
268 printf( "NC_HandleNCHitTest: hwnd=%x pt=%d,%d\n", hwnd
, pt
.x
, pt
.y
);
270 if (hwnd
== GetCapture()) return HTCLIENT
;
271 return NC_InternalNCHitTest( hwnd
, pt
);
275 /***********************************************************************
278 static void NC_DrawSysButton( HWND hwnd
, HDC hdc
, BOOL down
)
281 HDC hdcMem
= CreateCompatibleDC( hdc
);
284 NC_GetInsideRect( hwnd
, &rect
);
285 SelectObject( hdcMem
, hbitmapClose
);
286 BitBlt( hdc
, rect
.left
, rect
.top
, SYSMETRICS_CXSIZE
,
287 SYSMETRICS_CYSIZE
, hdcMem
, 1, 1, down
? NOTSRCCOPY
: SRCCOPY
);
293 /***********************************************************************
296 static void NC_DrawMaxButton( HWND hwnd
, HDC hdc
, BOOL down
)
299 HDC hdcMem
= CreateCompatibleDC( hdc
);
302 NC_GetInsideRect( hwnd
, &rect
);
304 SelectObject( hdcMem
, down
? hbitmapRestoreD
: hbitmapRestore
);
305 else SelectObject( hdcMem
, down
? hbitmapMaximizeD
: hbitmapMaximize
);
306 BitBlt( hdc
, rect
.right
- SYSMETRICS_CXSIZE
- 1, rect
.top
- 1,
307 SYSMETRICS_CXSIZE
+2, SYSMETRICS_CYSIZE
+2, hdcMem
, 0, 0, SRCCOPY
);
313 /***********************************************************************
316 static void NC_DrawMinButton( HWND hwnd
, HDC hdc
, BOOL down
)
319 WND
*wndPtr
= WIN_FindWndPtr( hwnd
);
320 HDC hdcMem
= CreateCompatibleDC( hdc
);
323 NC_GetInsideRect( hwnd
, &rect
);
324 if (wndPtr
->dwStyle
& WS_MAXIMIZEBOX
)
325 rect
.right
-= SYSMETRICS_CXSIZE
+ 1;
326 if (down
) SelectObject( hdcMem
, hbitmapMinimizeD
);
327 else SelectObject( hdcMem
, hbitmapMinimize
);
328 BitBlt( hdc
, rect
.right
- SYSMETRICS_CXSIZE
- 1, rect
.top
- 1,
329 SYSMETRICS_CXSIZE
+2, SYSMETRICS_CYSIZE
+2, hdcMem
, 0, 0, SRCCOPY
);
335 /***********************************************************************
338 * Draw a window frame inside the given rectangle, and update the rectangle.
339 * The correct pen and brush must be selected in the DC.
341 static void NC_DrawFrame( HDC hdc
, RECT
*rect
, BOOL dlgFrame
)
343 short width
, height
, tmp
;
347 width
= SYSMETRICS_CXDLGFRAME
- 1;
348 height
= SYSMETRICS_CYDLGFRAME
- 1;
352 width
= SYSMETRICS_CXFRAME
- 1;
353 height
= SYSMETRICS_CYFRAME
- 1;
357 PatBlt( hdc
, rect
->left
, rect
->top
,
358 rect
->right
- rect
->left
, height
, PATCOPY
);
359 PatBlt( hdc
, rect
->left
, rect
->top
,
360 width
, rect
->bottom
- rect
->top
, PATCOPY
);
361 PatBlt( hdc
, rect
->left
, rect
->bottom
,
362 rect
->right
- rect
->left
, -height
, PATCOPY
);
363 PatBlt( hdc
, rect
->right
, rect
->top
,
364 -width
, rect
->bottom
- rect
->top
, PATCOPY
);
368 InflateRect( rect
, -width
, -height
);
372 /* Draw inner rectangle */
373 MoveTo( hdc
, rect
->left
+width
, rect
->top
+height
);
374 LineTo( hdc
, rect
->right
-width
-1, rect
->top
+height
);
375 LineTo( hdc
, rect
->right
-width
-1, rect
->bottom
-height
-1 );
376 LineTo( hdc
, rect
->left
+width
, rect
->bottom
-height
-1 );
377 LineTo( hdc
, rect
->left
+width
, rect
->top
+height
);
379 /* Draw the decorations */
380 tmp
= rect
->top
+ SYSMETRICS_CYFRAME
+ SYSMETRICS_CYSIZE
;
381 MoveTo( hdc
, rect
->left
, tmp
);
382 LineTo( hdc
, rect
->left
+width
, tmp
);
383 MoveTo( hdc
, rect
->right
-width
-1, tmp
);
384 LineTo( hdc
, rect
->right
-1, tmp
);
386 tmp
= rect
->bottom
- 1 - SYSMETRICS_CYFRAME
- SYSMETRICS_CYSIZE
;
387 MoveTo( hdc
, rect
->left
, tmp
);
388 LineTo( hdc
, rect
->left
+width
, tmp
);
389 MoveTo( hdc
, rect
->right
-width
-1, tmp
);
390 LineTo( hdc
, rect
->right
-1, tmp
);
392 tmp
= rect
->left
+ SYSMETRICS_CXFRAME
+ SYSMETRICS_CXSIZE
;
393 MoveTo( hdc
, tmp
, rect
->top
);
394 LineTo( hdc
, tmp
, rect
->top
+height
);
395 MoveTo( hdc
, tmp
, rect
->bottom
-height
-1 );
396 LineTo( hdc
, tmp
, rect
->bottom
-1 );
398 tmp
= rect
->right
- 1 - SYSMETRICS_CXFRAME
- SYSMETRICS_CYSIZE
;
399 MoveTo( hdc
, tmp
, rect
->top
);
400 LineTo( hdc
, tmp
, rect
->top
+height
);
401 MoveTo( hdc
, tmp
, rect
->bottom
-height
-1 );
402 LineTo( hdc
, tmp
, rect
->bottom
-1 );
404 InflateRect( rect
, -width
-1, -height
-1 );
408 /***********************************************************************
411 * Draw the frame used when moving or resizing window.
413 static void NC_DrawMovingFrame( HDC hdc
, RECT
*rect
, BOOL thickframe
)
417 SelectObject( hdc
, GetStockObject( GRAY_BRUSH
) );
418 PatBlt( hdc
, rect
->left
, rect
->top
,
419 rect
->right
- rect
->left
- SYSMETRICS_CXFRAME
,
420 SYSMETRICS_CYFRAME
, PATINVERT
);
421 PatBlt( hdc
, rect
->left
, rect
->top
+ SYSMETRICS_CYFRAME
,
423 rect
->bottom
- rect
->top
- SYSMETRICS_CYFRAME
, PATINVERT
);
424 PatBlt( hdc
, rect
->left
+ SYSMETRICS_CXFRAME
, rect
->bottom
,
425 rect
->right
- rect
->left
- SYSMETRICS_CXFRAME
,
426 -SYSMETRICS_CYFRAME
, PATINVERT
);
427 PatBlt( hdc
, rect
->right
, rect
->top
, -SYSMETRICS_CXFRAME
,
428 rect
->bottom
- rect
->top
- SYSMETRICS_CYFRAME
, PATINVERT
);
430 else DrawFocusRect( hdc
, rect
);
434 /***********************************************************************
437 * Draw the window caption.
438 * The correct pen for the window frame must be selected in the DC.
440 static void NC_DrawCaption( HDC hdc
, RECT
*rect
, HWND hwnd
,
441 DWORD style
, BOOL active
)
444 HBRUSH hbrushCaption
;
445 WND
* wndPtr
= WIN_FindWndPtr( hwnd
);
450 if (!(hbitmapClose
= LoadBitmap( 0, MAKEINTRESOURCE(OBM_CLOSE
) )))
452 hbitmapMinimize
= LoadBitmap( 0, MAKEINTRESOURCE(OBM_REDUCE
) );
453 hbitmapMinimizeD
= LoadBitmap( 0, MAKEINTRESOURCE(OBM_REDUCED
) );
454 hbitmapMaximize
= LoadBitmap( 0, MAKEINTRESOURCE(OBM_ZOOM
) );
455 hbitmapMaximizeD
= LoadBitmap( 0, MAKEINTRESOURCE(OBM_ZOOMD
) );
456 hbitmapRestore
= LoadBitmap( 0, MAKEINTRESOURCE(OBM_RESTORE
) );
457 hbitmapRestoreD
= LoadBitmap( 0, MAKEINTRESOURCE(OBM_RESTORED
) );
460 if (wndPtr
->dwExStyle
& WS_EX_DLGMODALFRAME
)
462 HBRUSH hbrushWindow
= CreateSolidBrush(GetSysColor(COLOR_WINDOW
));
463 HBRUSH hbrushOld
= SelectObject( hdc
, hbrushWindow
);
464 PatBlt( hdc
, r
.left
, r
.top
, 1, r
.bottom
-r
.top
+1,PATCOPY
);
465 PatBlt( hdc
, r
.right
-1, r
.top
, 1, r
.bottom
-r
.top
+1, PATCOPY
);
466 PatBlt( hdc
, r
.left
, r
.top
-1, r
.right
-r
.left
, 1, PATCOPY
);
469 SelectObject( hdc
, hbrushOld
);
470 DeleteObject( hbrushWindow
);
474 hbrushCaption
= CreateSolidBrush( GetSysColor(COLOR_ACTIVECAPTION
) );
475 else hbrushCaption
= CreateSolidBrush( GetSysColor(COLOR_INACTIVECAPTION
));
477 MoveTo( hdc
, r
.left
, r
.bottom
);
478 LineTo( hdc
, r
.right
-1, r
.bottom
);
480 if (style
& WS_SYSMENU
)
482 NC_DrawSysButton( hwnd
, hdc
, FALSE
);
483 r
.left
+= SYSMETRICS_CXSIZE
+ 1;
484 MoveTo( hdc
, r
.left
- 1, r
.top
);
485 LineTo( hdc
, r
.left
- 1, r
.bottom
);
487 if (style
& WS_MAXIMIZEBOX
)
489 NC_DrawMaxButton( hwnd
, hdc
, FALSE
);
490 r
.right
-= SYSMETRICS_CXSIZE
+ 1;
492 if (style
& WS_MINIMIZEBOX
)
494 NC_DrawMinButton( hwnd
, hdc
, FALSE
);
495 r
.right
-= SYSMETRICS_CXSIZE
+ 1;
498 FillRect( hdc
, &r
, hbrushCaption
);
499 DeleteObject( hbrushCaption
);
501 if (GetWindowText( hwnd
, buffer
, 256 ))
503 if (active
) SetTextColor( hdc
, GetSysColor( COLOR_CAPTIONTEXT
) );
504 else SetTextColor( hdc
, GetSysColor( COLOR_INACTIVECAPTIONTEXT
) );
505 SetBkMode( hdc
, TRANSPARENT
);
506 DrawText( hdc
, buffer
, -1, &r
, DT_SINGLELINE
| DT_CENTER
| DT_VCENTER
);
511 /***********************************************************************
514 * Paint the non-client area.
516 static void NC_DoNCPaint( HWND hwnd
, HRGN hrgn
, BOOL active
)
520 HBRUSH hbrushBorder
= 0;
523 WND
*wndPtr
= WIN_FindWndPtr( hwnd
);
525 #ifdef DEBUG_NONCLIENT
526 printf( "NC_HandleNCPaint: %d %d\n", hwnd
, hrgn
);
529 if (!wndPtr
|| !hrgn
) return;
530 if (!(wndPtr
->dwStyle
& (WS_BORDER
| WS_DLGFRAME
| WS_THICKFRAME
)))
531 return; /* Nothing to do! */
533 if (hrgn
== 1) hdc
= GetDCEx( hwnd
, 0, DCX_CACHE
| DCX_WINDOW
);
534 else hdc
= GetDCEx( hwnd
, hrgn
, DCX_CACHE
| DCX_WINDOW
| DCX_INTERSECTRGN
);
536 if (ExcludeVisRect( hdc
, wndPtr
->rectClient
.left
-wndPtr
->rectWindow
.left
,
537 wndPtr
->rectClient
.top
-wndPtr
->rectWindow
.top
,
538 wndPtr
->rectClient
.right
-wndPtr
->rectWindow
.left
,
539 wndPtr
->rectClient
.bottom
-wndPtr
->rectWindow
.top
)
542 ReleaseDC( hwnd
, hdc
);
546 rect
.top
= rect
.left
= 0;
547 rect
.right
= wndPtr
->rectWindow
.right
- wndPtr
->rectWindow
.left
;
548 rect
.bottom
= wndPtr
->rectWindow
.bottom
- wndPtr
->rectWindow
.top
;
550 hpenFrame
= CreatePen( PS_SOLID
, 1, GetSysColor(COLOR_WINDOWFRAME
) );
551 SelectObject( hdc
, hpenFrame
);
553 hbrushBorder
= CreateSolidBrush( GetSysColor(COLOR_ACTIVEBORDER
) );
554 else hbrushBorder
= CreateSolidBrush( GetSysColor(COLOR_INACTIVEBORDER
) );
555 SelectObject( hdc
, hbrushBorder
);
557 if ((wndPtr
->dwStyle
& WS_BORDER
) || (wndPtr
->dwStyle
& WS_DLGFRAME
))
560 LineTo( hdc
, rect
.right
-1, 0 );
561 LineTo( hdc
, rect
.right
-1, rect
.bottom
-1 );
562 LineTo( hdc
, 0, rect
.bottom
-1 );
564 InflateRect( &rect
, -1, -1 );
567 if (HAS_DLGFRAME( wndPtr
->dwStyle
, wndPtr
->dwExStyle
))
568 NC_DrawFrame( hdc
, &rect
, TRUE
);
569 else if (wndPtr
->dwStyle
& WS_THICKFRAME
)
570 NC_DrawFrame(hdc
, &rect
, FALSE
);
572 if ((wndPtr
->dwStyle
& WS_CAPTION
) == WS_CAPTION
)
575 rect
.top
+= SYSMETRICS_CYSIZE
+ 1;
576 r
.bottom
= rect
.top
- 1;
577 NC_DrawCaption( hdc
, &r
, hwnd
, wndPtr
->dwStyle
, active
);
580 if (wndPtr
->dwStyle
& (WS_VSCROLL
| WS_HSCROLL
))
582 if (wndPtr
->dwStyle
& WS_VSCROLL
) {
583 int bottom
= rect
.bottom
;
584 if (wndPtr
->dwStyle
& WS_HSCROLL
) bottom
-= SYSMETRICS_CYHSCROLL
;
585 SetRect(&rect2
, rect
.right
- SYSMETRICS_CXVSCROLL
, rect
.top
,
587 StdDrawScrollBar(hwnd
, hdc
, SB_VERT
, &rect2
, (LPHEADSCROLL
)wndPtr
->VScroll
);
589 if (wndPtr
->dwStyle
& WS_HSCROLL
) {
590 int right
= rect
.right
;
591 if (wndPtr
->dwStyle
& WS_VSCROLL
) right
-= SYSMETRICS_CYVSCROLL
;
592 SetRect(&rect2
, rect
.left
, rect
.bottom
- SYSMETRICS_CYHSCROLL
,
594 StdDrawScrollBar(hwnd
, hdc
, SB_HORZ
, &rect2
, (LPHEADSCROLL
)wndPtr
->HScroll
);
597 HBRUSH hbrushScroll = CreateSolidBrush( GetSysColor(COLOR_SCROLLBAR) );
598 HBRUSH hbrushOld = SelectObject( hdc, hbrushScroll );
599 if (wndPtr->dwStyle & WS_VSCROLL)
600 PatBlt( hdc, rect.right - SYSMETRICS_CXVSCROLL, rect.top,
601 SYSMETRICS_CXVSCROLL, rect.bottom-rect.top, PATCOPY );
602 if (wndPtr->dwStyle & WS_HSCROLL)
603 PatBlt( hdc, rect.left, rect.bottom - SYSMETRICS_CYHSCROLL,
604 rect.right-rect.left, SYSMETRICS_CYHSCROLL, PATCOPY );
605 SelectObject( hdc, hbrushOld );
606 DeleteObject( hbrushScroll );
610 ReleaseDC( hwnd
, hdc
);
611 if (hbrushBorder
) DeleteObject( hbrushBorder
);
612 if (hpenFrame
) DeleteObject( hpenFrame
);
616 /***********************************************************************
619 * Handle a WM_NCPAINT message. Called from DefWindowProc().
621 LONG
NC_HandleNCPaint( HWND hwnd
, HRGN hrgn
)
623 NC_DoNCPaint( hwnd
, hrgn
, (hwnd
== GetActiveWindow()) );
628 /***********************************************************************
629 * NC_HandleNCActivate
631 * Handle a WM_NCACTIVATE message. Called from DefWindowProc().
633 LONG
NC_HandleNCActivate( HWND hwnd
, WORD wParam
)
635 NC_DoNCPaint( hwnd
, (HRGN
)1, wParam
);
640 /***********************************************************************
643 * Perform SC_MOVE and SC_SIZE commands.
645 static void NC_DoSizeMove( HWND hwnd
, WORD wParam
, POINT pt
)
652 POINT minTrack
, maxTrack
, capturePoint
= pt
;
653 WND
* wndPtr
= WIN_FindWndPtr( hwnd
);
655 if (IsZoomed(hwnd
) || IsIconic(hwnd
) || !IsWindowVisible(hwnd
)) return;
656 hittest
= wParam
& 0x0f;
657 thickframe
= HAS_THICKFRAME( wndPtr
->dwStyle
);
659 if ((wParam
& 0xfff0) == SC_MOVE
)
661 if (!(wndPtr
->dwStyle
& WS_CAPTION
)) return;
664 /* Move pointer at the center of the caption */
667 NC_GetInsideRect( hwnd
, &rect
);
668 if (wndPtr
->dwStyle
& WS_SYSMENU
)
669 rect
.left
+= SYSMETRICS_CXSIZE
+ 1;
670 if (wndPtr
->dwStyle
& WS_MINIMIZEBOX
)
671 rect
.right
-= SYSMETRICS_CXSIZE
+ 1;
672 if (wndPtr
->dwStyle
& WS_MAXIMIZEBOX
)
673 rect
.right
-= SYSMETRICS_CXSIZE
+ 1;
674 point
.x
= wndPtr
->rectWindow
.left
+ (rect
.right
- rect
.left
) / 2;
675 point
.y
= wndPtr
->rectWindow
.top
+ rect
.top
+ SYSMETRICS_CYSIZE
/2;
676 if (wndPtr
->dwStyle
& WS_CHILD
)
677 ClientToScreen( wndPtr
->hwndParent
, &point
);
678 SetCursorPos( point
.x
, point
.y
);
680 capturePoint
= point
;
685 if (!thickframe
) return;
686 if (hittest
) hittest
+= HTLEFT
-1;
689 WINPOS_GetMinMaxInfo( hwnd
, NULL
, NULL
, &minTrack
, &maxTrack
);
690 SendMessage( hwnd
, WM_ENTERSIZEMOVE
, 0, 0 );
692 if (wndPtr
->dwStyle
& WS_CHILD
) hdc
= GetDC( wndPtr
->hwndParent
);
694 { /* Grab the server only when moving top-level windows */
696 XGrabServer( display
);
699 sizingRect
= wndPtr
->rectWindow
;
700 NC_DrawMovingFrame( hdc
, &sizingRect
, thickframe
);
706 MSG_GetHardwareMessage( &msg
);
708 /* Exit on button-up, Return, or Esc */
709 if ((msg
.message
== WM_LBUTTONUP
) ||
710 ((msg
.message
== WM_KEYDOWN
) &&
711 ((msg
.wParam
== VK_RETURN
) || (msg
.wParam
== VK_ESCAPE
)))) break;
713 if (wndPtr
->dwStyle
& WS_CHILD
)
714 ScreenToClient( wndPtr
->hwndParent
, &msg
.pt
);
719 dx
= msg
.pt
.x
- capturePoint
.x
;
720 dy
= msg
.pt
.y
- capturePoint
.y
;
726 case VK_UP
: msg
.pt
.y
-= 8; break;
727 case VK_DOWN
: msg
.pt
.y
+= 8; break;
728 case VK_LEFT
: msg
.pt
.x
-= 8; break;
729 case VK_RIGHT
: msg
.pt
.x
+= 8; break;
731 SetCursorPos( msg
.pt
.x
, msg
.pt
.y
);
737 RECT newRect
= sizingRect
;
741 OffsetRect( &newRect
, dx
, dy
);
761 newRect
.bottom
+= dy
;
765 newRect
.bottom
+= dy
;
769 newRect
.bottom
+= dy
;
772 NC_DrawMovingFrame( hdc
, &sizingRect
, thickframe
);
773 NC_DrawMovingFrame( hdc
, &newRect
, thickframe
);
774 capturePoint
= msg
.pt
;
775 sizingRect
= newRect
;
779 NC_DrawMovingFrame( hdc
, &sizingRect
, thickframe
);
781 if (wndPtr
->dwStyle
& WS_CHILD
) ReleaseDC( wndPtr
->hwndParent
, hdc
);
785 XUngrabServer( display
);
787 SendMessage( hwnd
, WM_EXITSIZEMOVE
, 0, 0 );
789 /* If Esc key, don't move the window */
790 if ((msg
.message
== WM_KEYDOWN
) && (msg
.wParam
== VK_ESCAPE
)) return;
792 if (hittest
!= HTCAPTION
)
793 SetWindowPos( hwnd
, 0, sizingRect
.left
, sizingRect
.top
,
794 sizingRect
.right
- sizingRect
.left
,
795 sizingRect
.bottom
- sizingRect
.top
,
796 SWP_NOACTIVATE
| SWP_NOZORDER
);
797 else SetWindowPos( hwnd
, 0, sizingRect
.left
, sizingRect
.top
, 0, 0,
798 SWP_NOACTIVATE
| SWP_NOSIZE
| SWP_NOZORDER
);
802 /***********************************************************************
805 * Track a mouse button press on the minimize or maximize box.
807 static void NC_TrackMinMaxBox( HWND hwnd
, WORD wParam
)
810 HDC hdc
= GetWindowDC( hwnd
);
814 if (wParam
== HTMINBUTTON
) NC_DrawMinButton( hwnd
, hdc
, TRUE
);
815 else NC_DrawMaxButton( hwnd
, hdc
, TRUE
);
819 BOOL oldstate
= pressed
;
820 MSG_GetHardwareMessage( &msg
);
822 pressed
= (NC_InternalNCHitTest( hwnd
, msg
.pt
) == wParam
);
823 if (pressed
!= oldstate
)
825 if (wParam
== HTMINBUTTON
) NC_DrawMinButton( hwnd
, hdc
, pressed
);
826 else NC_DrawMaxButton( hwnd
, hdc
, pressed
);
828 } while (msg
.message
!= WM_LBUTTONUP
);
830 if (wParam
== HTMINBUTTON
) NC_DrawMinButton( hwnd
, hdc
, FALSE
);
831 else NC_DrawMaxButton( hwnd
, hdc
, FALSE
);
834 ReleaseDC( hwnd
, hdc
);
835 if (!pressed
) return;
837 if (wParam
== HTMINBUTTON
)
838 SendMessage( hwnd
, WM_SYSCOMMAND
, SC_MINIMIZE
, *(LONG
*)&msg
.pt
);
840 SendMessage( hwnd
, WM_SYSCOMMAND
,
841 IsZoomed(hwnd
) ? SC_RESTORE
: SC_MAXIMIZE
, *(LONG
*)&msg
.pt
);
845 /***********************************************************************
846 * NC_HandleNCLButtonDown
848 * Handle a WM_NCLBUTTONDOWN message. Called from DefWindowProc().
850 LONG
NC_HandleNCLButtonDown( HWND hwnd
, WORD wParam
, LONG lParam
)
852 HDC hdc
= GetWindowDC( hwnd
);
854 switch(wParam
) /* Hit test */
857 SendMessage( hwnd
, WM_SYSCOMMAND
, SC_MOVE
+ HTCAPTION
, lParam
);
861 NC_DrawSysButton( hwnd
, hdc
, TRUE
);
862 NC_TrackSysMenu(hwnd
);
869 SendMessage( hwnd
, WM_SYSCOMMAND
, SC_HSCROLL
, lParam
);
873 SendMessage( hwnd
, WM_SYSCOMMAND
, SC_VSCROLL
, lParam
);
878 NC_TrackMinMaxBox( hwnd
, wParam
);
889 SendMessage( hwnd
, WM_SYSCOMMAND
, SC_SIZE
+ wParam
- HTLEFT
+1, lParam
);
896 ReleaseDC( hwnd
, hdc
);
901 /***********************************************************************
902 * NC_HandleNCLButtonDblClk
904 * Handle a WM_NCLBUTTONDBLCLK message. Called from DefWindowProc().
906 LONG
NC_HandleNCLButtonDblClk( HWND hwnd
, WORD wParam
, LONG lParam
)
908 switch(wParam
) /* Hit test */
911 SendMessage( hwnd
, WM_SYSCOMMAND
, SC_MAXIMIZE
, lParam
);
915 SendMessage( hwnd
, WM_SYSCOMMAND
, SC_CLOSE
, lParam
);
922 /***********************************************************************
923 * NC_HandleSysCommand
925 * Handle a WM_SYSCOMMAND message. Called from DefWindowProc().
927 LONG
NC_HandleSysCommand( HWND hwnd
, WORD wParam
, POINT pt
)
929 WND
*wndPtr
= WIN_FindWndPtr( hwnd
);
931 #ifdef DEBUG_NONCLIENT
932 printf( "Handling WM_SYSCOMMAND %x %d,%d\n", wParam
, pt
.x
, pt
.y
);
935 if (wndPtr
->dwStyle
& WS_CHILD
) ScreenToClient( wndPtr
->hwndParent
, &pt
);
937 switch (wParam
& 0xfff0)
941 NC_DoSizeMove( hwnd
, wParam
, pt
);
945 ShowWindow( hwnd
, SW_MINIMIZE
);
949 ShowWindow( hwnd
, SW_MAXIMIZE
);
953 ShowWindow( hwnd
, SW_RESTORE
);
961 return SendMessage( hwnd
, WM_CLOSE
, 0, 0 );
980 /***********************************************************************
983 * Handle a WM_SETCURSOR message. Called from DefWindowProc().
985 LONG
NC_HandleSetCursor( HWND hwnd
, WORD wParam
, LONG lParam
)
987 if (hwnd
!= wParam
) return 0; /* Don't set the cursor for child windows */
989 switch(LOWORD(lParam
))
993 WORD msg
= HIWORD( lParam
);
994 if ((msg
== WM_LBUTTONDOWN
) || (msg
== WM_MBUTTONDOWN
) ||
995 (msg
== WM_RBUTTONDOWN
))
1002 WND
*wndPtr
= WIN_FindWndPtr( hwnd
);
1003 if (wndPtr
&& wndPtr
->hCursor
)
1005 SetCursor( wndPtr
->hCursor
);
1013 SetCursor( LoadCursor( 0, IDC_SIZEWE
) );
1018 SetCursor( LoadCursor( 0, IDC_SIZENS
) );
1023 SetCursor( LoadCursor( 0, IDC_SIZENWSE
) );
1028 SetCursor( LoadCursor( 0, IDC_SIZENESW
) );
1032 /* Default cursor: arrow */
1033 SetCursor( LoadCursor( 0, IDC_ARROW
) );