Fix for mingw32 compilation (replaced #ifdef NM_CUSTOMDRAW with a test for _WIN32_IE
[wxWidgets.git] / src / msw / minifram.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: minifram.cpp
3 // Purpose: wxMiniFrame
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 04/01/98
7 // RCS-ID: $Id$
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows license
10 /////////////////////////////////////////////////////////////////////////////
11
12 #ifdef __GNUG__
13 #pragma implementation "minifram.h"
14 #endif
15
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
18
19 #ifdef __BORLANDC__
20 #pragma hdrstop
21 #endif
22
23 #include "wx/window.h"
24 #include "wx/msw/private.h"
25
26 #ifndef WX_PRECOMP
27 #include "wx/setup.h"
28 #include "wx/event.h"
29 #include "wx/app.h"
30 #include "wx/utils.h"
31 #endif
32
33
34 #ifdef __WIN32__
35
36 #include "wx/minifram.h"
37
38 IMPLEMENT_DYNAMIC_CLASS(wxMiniFrame, wxFrame)
39
40
41 #else
42
43
44 #if wxUSE_ITSY_BITSY
45
46 #include "wx/minifram.h"
47
48 #ifndef __TWIN32__
49 #ifdef __GNUWIN32__
50 #ifndef wxUSE_NORLANDER_HEADERS
51 #include "wx/msw/gnuwin32/extra.h"
52 #endif
53 #endif
54 #endif
55
56 IMPLEMENT_DYNAMIC_CLASS(wxMiniFrame, wxFrame)
57
58 long wxMiniFrame::MSWDefWindowProc(WXUINT nMsg, WXWPARAM wParam, WXLPARAM lParam)
59 {
60 if ((GetWindowStyleFlag() & wxTINY_CAPTION_HORIZ) ||
61 (GetWindowStyleFlag() & wxTINY_CAPTION_VERT))
62 return ::ibDefWindowProc((HWND) GetHWND(), nMsg, wParam, lParam);
63 else if ( m_oldWndProc )
64 return ::CallWindowProc(CASTWNDPROC m_oldWndProc, (HWND) GetHWND(), (UINT) nMsg, (WPARAM) wParam, (LPARAM) lParam);
65 else
66 return ::DefWindowProc((HWND) GetHWND(), nMsg, wParam, lParam);
67 }
68
69 wxMiniFrame::~wxMiniFrame(void)
70 {
71 }
72
73 /////////////////////////////////////////////////////////////////////////
74 //
75 // Project: ItsyBitsy window support module
76 // Module: itsybits.c
77 //
78 //
79 // ItsyBitsy is a support module that allows you to create windows
80 // that look and act very much like a popup window witha system
81 // menu and caption bar, except everything is scaled to about 2/3
82 // scale.
83 //
84 // For documentation on how to use ItsyBits, read the document
85 // ITSYBITS.DOC.
86 //
87 // Revisions:
88 // 9/27/91 Charlie Kindel (cek/ckindel)
89 // Wrote and documented it.
90 //
91 // 1/14/93 cek
92 // 2/23/93 cek Added minimize/maximize buttons.
93 // 3/18/93 cek Fixed system menu bug where system menu
94 // popped back up if you clicked on the
95 // icon again while it was up.
96 // 3/24/93 cek More comments. Fixed DS_MODALDIALOG style
97 // problem. Use auto precompiled headers
98 // in MSVC.
99 //
100 //////////////////////////////////////////////////////////////////////////
101
102 #include "wx/window.h"
103 #include "wx/msw/private.h"
104
105 #include <string.h>
106
107 #if !defined( __WATCOMC__ ) && !defined( __MWERKS__ ) && !defined(__SALFORDC__)
108 #include <memory.h>
109 #endif
110
111 #include <stdlib.h>
112
113 #ifndef _RGB_H_
114 #define _RGB_H_
115
116 // Some mildly useful macros for the standard 16 colors
117 #define RGBBLACK RGB(0,0,0)
118 #define RGBRED RGB(128,0,0)
119 #define RGBGREEN RGB(0,128,0)
120 #define RGBBLUE RGB(0,0,128)
121
122 #define RGBBROWN RGB(128,128,0)
123 #define RGBMAGENTA RGB(128,0,128)
124 #define RGBCYAN RGB(0,128,128)
125 #define RGBLTGRAY RGB(192,192,192)
126
127 #define RGBGRAY RGB(128,128,128)
128 #define RGBLTRED RGB(255,0,0)
129 #define RGBLTGREEN RGB(0,255,0)
130 #define RGBLTBLUE RGB(0,0,255)
131
132 #define RGBYELLOW RGB(255,255,0)
133 #define RGBLTMAGENTA RGB(255,0,255)
134 #define RGBLTCYAN RGB(0,255,255)
135 #define RGBWHITE RGB(255,255,255)
136 #endif
137
138 #ifndef GlobalAllocPtr
139 #define GlobalPtrHandle(lp) \
140 ((HGLOBAL)GlobalHandle(lp))
141
142 #define GlobalLockPtr(lp) \
143 ((BOOL)GlobalLock(GlobalPtrHandle(lp)))
144 #define GlobalUnlockPtr(lp) \
145 GlobalUnlock(GlobalPtrHandle(lp))
146
147 #define GlobalAllocPtr(flags, cb) \
148 (GlobalLock(GlobalAlloc((flags), (cb))))
149 #define GlobalReAllocPtr(lp, cbNew, flags) \
150 (GlobalUnlockPtr(lp), GlobalLock(GlobalReAlloc(GlobalPtrHandle(lp) , (cbNew), (flags))))
151 #define GlobalFreePtr(lp) \
152 (GlobalUnlockPtr(lp), (BOOL)GlobalFree(GlobalPtrHandle(lp)))
153 #endif
154
155 #if defined(__BORLANDC__) || defined(__WATCOMC__)
156 #define max(a,b) (((a) > (b)) ? (a) : (b))
157 #define min(a,b) (((a) < (b)) ? (a) : (b))
158 #endif
159
160 // CAPTIONXY is the default size of the system menu icon. This
161 // determines the height/width of the caption.
162 //
163 // The value that results from the following formula works out
164 // nicely for the veritcal caption on VGA, 8514 (Large Fonts),
165 // 8514 (Small Fonts), XGA (Small Fonts), XGA (Large Fonts),
166 // and TIGA (Small Fonts). It may not be good on other displays.
167 //
168 // The problem is that TT fonts turn into bitmap fonts when they
169 // are sized below a certain threshold. The idea is to make the
170 // size of the caption just big enough to get the smallest TT
171 // (scalable) font to fit.
172 //
173 #define CAPTIONXY (GetSystemMetrics( SM_CYCAPTION ) / 2 + 1)
174
175 #define TestWinStyle( hWnd, dwStyleBit ) \
176 (((DWORD)GetWindowLong( hWnd, GWL_STYLE ) & (DWORD)dwStyleBit) ? TRUE : FALSE )
177 #define HASCAPTION( hwnd ) (TestWinStyle( hwnd, IBS_VERTCAPTION ) ||\
178 TestWinStyle( hwnd, IBS_HORZCAPTION ))
179
180 #define SETCAPTIONSIZE(h,i) (UINT)SetProp(h,wxT("ibSize"),(HANDLE)i)
181 #define GETCAPTIONSIZE(h) (UINT)GetProp(h,wxT("ibSize"))
182 #define FREECAPTIONSIZE(h) RemoveProp(h,wxT("ibSize"))
183
184 #define SETMENUWASUPFLAG(h,i) (UINT)SetProp(h,wxT("ibFlag"),(HANDLE)i)
185 #define GETMENUWASUPFLAG(h) (UINT)GetProp(h,wxT("ibFlag"))
186 #define FREEMENUWASUPFLAG(h) RemoveProp(h,wxT("ibFlag"))
187
188 /////////////////////////////////////////////////////////////////////
189 // Little known fact:
190 // ExtTextOut() is the fastest way to draw a filled rectangle
191 // in Windows (if you don't want dithered colors or borders).
192 //
193 // Unfortunately there is a bug in the Windows 3.0 8514 driver
194 // in using ExtTextOut() to a memory DC. If you are drawing
195 // to an off screen bitmap, then blitting that bitmap to the
196 // display, do not #define wxUSE_EXTTEXTOUT below.
197 //
198 // The following macro (DRAWFASTRECT) draws a filled rectangle
199 // with no border and a solid color. It uses the current back-
200 // ground color as the fill color.
201 //////////////////////////////////////////////////////////////////////
202 #define wxUSE_EXTTEXTOUT
203 #ifdef wxUSE_EXTTEXTOUT
204 #define DRAWFASTRECT(hdc,lprc) ExtTextOut(hdc,0,0,ETO_OPAQUE,lprc,NULL,0,NULL)
205 #else
206 #define DRAWFASTRECT(hdc,lprc) {\
207 HBRUSH hbr = CreateSolidBrush( GetBkColor( hdc ) ) ;\
208 hbr = SelectObject(hdc, hbr) ;\
209 PatBlt(hdc,(lprc)->left,(lprc)->top,(lprc)->right-(lprc)->left,(lprc)->bottom-(lprc)->top,PATCOPY) ;\
210 hbr = SelectObject(hdc, hbr) ;\
211 DeleteObject( hbr ) ;\
212 }
213 #endif
214
215 // The DrawArrow function takes the following to indicate what
216 // kind of arrow to draw.
217 //
218 #define ARROW_UP 0
219 #define ARROW_DOWN 1
220 #define ARROW_RESTORE 2
221
222 BOOL PASCAL DepressMinMaxButton( HWND hWnd, UINT uiHT, LPRECT ) ;
223 BOOL PASCAL DoMenu( HWND hWnd ) ;
224 void PASCAL SetupSystemMenu( HWND hWnd, HMENU hMenu ) ;
225 BOOL PASCAL GetCaptionRect( HWND hWnd, LPRECT lprc ) ;
226 BOOL PASCAL GetIconRect( HWND hWnd, LPRECT lprc ) ;
227 BOOL PASCAL GetButtonRect( HWND hWnd, UINT nPos, LPRECT lprc ) ;
228 BOOL PASCAL GetMinButtonRect( HWND hWnd, LPRECT lprc ) ;
229 BOOL PASCAL GetMaxButtonRect( HWND hWnd, LPRECT lprc ) ;
230 BOOL PASCAL DrawCaption( HDC hDC, HWND hWnd, LPRECT lprc,
231 BOOL fVert, BOOL fSysMenu,
232 BOOL fMin, BOOL fMax, BOOL fActive ) ;
233 void PASCAL DrawSysMenu( HDC hDC, HWND hWnd, BOOL fInvert ) ;
234 void PASCAL DrawButton( HDC hDC, HWND hWnd, BOOL fMin, BOOL fDepressed ) ;
235 void PASCAL DrawArrow( HDC hdc, LPRECT lprc, UINT uiStyle ) ;
236
237 // Global vars
238 //
239 static BOOL fWin31 ;
240
241 ///////////////////////////////////////////////////////////////////////
242 // External/Public functions
243 ///////////////////////////////////////////////////////////////////////
244
245 /////////////////////////////////////////////////////////////////
246 // UINT WINAPI ibGetCaptionSize( HWND hWnd )
247 //
248 // Description:
249 //
250 // Gets the size of the caption (height if horz, width if
251 // vertical).
252 //
253 // Comments:
254 //
255 ///////////////////////////////////////////////////////////////
256 UINT WINAPI ibGetCaptionSize( HWND hWnd )
257 {
258 return GETCAPTIONSIZE( hWnd ) + 1 ;
259 } // ibSetCaptionSize()
260
261 /////////////////////////////////////////////////////////////////
262 // UINT WINAPI ibSetCaptionSize( HWND hWnd, UINT nSize )
263 //
264 // Description:
265 //
266 // Changes the size of the caption (height if horz, width if
267 // vertical).
268 //
269 // Comments:
270 //
271 //////////////////////////////////////////////////////////////////
272 UINT WINAPI ibSetCaptionSize( HWND hWnd, UINT nSize )
273 {
274 UINT ui ;
275
276 if (nSize <= 0)
277 return 0 ;
278
279 nSize-- ;
280 ui = SETCAPTIONSIZE( hWnd, nSize ) ;
281
282 // Once we change the window style, we need a WM_NCCALCRECT
283 // to be generated.
284 //
285 // SWP_FRAMECHANGED is not documented in the 3.1 SDK docs,
286 // but *is* in WINDOWS.H.
287 //
288 SetWindowPos( hWnd, NULL, 0, 0, 0, 0, SWP_FRAMECHANGED |
289 SWP_NOSIZE | SWP_NOMOVE |
290 SWP_NOACTIVATE | SWP_NOZORDER) ;
291
292 return ui ;
293
294 } // ibSetCaptionSize()
295
296 /////////////////////////////////////////////////////////////////
297 // LRESULT WINAPI ibDefWindowProc( HWND hWnd, UINT uiMsg, WPARAM wParam, LPARAM lParam )
298 //
299 // Description:
300 //
301 // This function should be called instead of DefWindowProc() for
302 // windows that want to have itsybitsy captions.
303 //
304 // Comments:
305 //
306 //////////////////////////////////////////////////////////////////
307 LRESULT WINAPI ibDefWindowProc( HWND hWnd, UINT uiMsg, WPARAM wParam, LPARAM lParam )
308 {
309 LRESULT lRet ;
310 UINT nCapSize ;
311
312 switch( uiMsg )
313 {
314 case WM_SYSCHAR:
315 // If ALT-SPACE
316 // was hit then pop up the menu
317 //
318 if (HASCAPTION( hWnd ) && (wParam == VK_SPACE))
319 {
320 DoMenu( hWnd ) ;
321 break ;
322 }
323
324 // FALL THROUGH!!!!
325 //
326
327 case WM_SYSKEYDOWN:
328 case WM_SYSKEYUP:
329 case WM_KEYDOWN:
330 case WM_KEYUP:
331 {
332 DWORD dw = GetWindowLong( hWnd, GWL_STYLE ) ;
333
334 // Fool DefWindowProc into thinking we do not have
335 // a system menu. Otherwise it will try to
336 // pop up its own.
337 //
338 SetWindowLong( hWnd, GWL_STYLE, dw &~WS_SYSMENU ) ;
339 lRet = DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
340 SetWindowLong( hWnd, GWL_STYLE, dw ) ;
341 return lRet ;
342 }
343 break ;
344
345 case WM_COMMAND:
346 // The menu that is poped up for the system menu with
347 // TrackPopupMenu() sends it's notifications as WM_COMMAND
348 // messages. We need to translate these into
349 // WM_SYSCOMMAND messages. All standard WM_SYSCOMMAND
350 // ids are greater than 0xF000.
351 //
352 // This could be a possible cause of confusion if the
353 // itsybitsy window had children that used SC_MOVE or SC_CLOSE
354 // as their IDs. Take note and be careful.
355 //
356 // Also, because ibDefWindowProc looks at WM_COMMAND messages,
357 // you will need to be careful to call ibDefWindowProc() for
358 // any wm_command messages that you would normally ignore.
359 // Otherwise the system menu won't work.
360 //
361 if (wParam >= 0xF000)
362 // Call PostMessage() here instead of SendMessage!
363 // Here's why:
364 // Our menu was created by TrackPopupMenu(). TPM() does
365 // not return until after the menu has been destroyed
366 // (and thus the command associated with the menu selection
367 // sent). Therefore when we get here, we are still
368 // *inside* TPM(). If we Send WM_SYSCOMMAND, SC_CLOSE
369 // to the window, the window will be destroyed before
370 // TPM() returns to our code within DoMenu() below. Wel...
371 // We do stuff with the window handle after DoMenu()
372 // returns (namely GetProp()). Since the window has
373 // been destroyed,this is bad.
374 PostMessage( hWnd, WM_SYSCOMMAND, wParam, lParam ) ;
375
376 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
377
378 case WM_GETMINMAXINFO:
379 {
380 nCapSize = GETCAPTIONSIZE( hWnd ) ;
381 if (HASCAPTION( hWnd ) && TestWinStyle( hWnd, WS_THICKFRAME ))
382 {
383 LPPOINT lppt = (LPPOINT)lParam ;
384 RECT rcMenu ;
385 RECT rcMin ;
386 RECT rcMax ;
387 int nX ;
388 int cx, cy ; // window frame/border width
389
390 if (TestWinStyle( hWnd, WS_THICKFRAME ))
391 {
392 cx = GetSystemMetrics( SM_CXFRAME ) ;
393 cy = GetSystemMetrics( SM_CYFRAME ) ;
394 }
395 else if (TestWinStyle(hWnd, WS_BORDER ))
396 {
397 cx = GetSystemMetrics( SM_CXBORDER ) ;
398 cy = GetSystemMetrics( SM_CYBORDER ) ;
399 }
400 else
401 {
402 // VZ: I don't know what should be here, but the vars must
403 // be inited!
404 wxFAIL_MSG(wxT("don't know how to initialize cx, cy"));
405
406 cx = cy = 0;
407 }
408
409 GetIconRect( hWnd, &rcMenu ) ;
410 GetMinButtonRect( hWnd, &rcMin ) ;
411 GetMaxButtonRect( hWnd, &rcMax ) ;
412 nX = (rcMenu.right-rcMenu.left) +
413 (rcMin.right-rcMin.left) +
414 (rcMin.right-rcMin.left) ;
415
416
417 if (TestWinStyle( hWnd, IBS_VERTCAPTION ) )
418 {
419 lppt[3].x = nCapSize + cx * 2 - 1 ;
420 lppt[3].y = nX + (2* nCapSize) ;
421 }
422 else
423 {
424 lppt[3].x = nX + (2* nCapSize) ;
425 lppt[3].y = nCapSize + cy * 2 - 1 ;
426 }
427 }
428 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
429 }
430 break ;
431
432 /////////////////////////////////////////////////////////////////////
433 // Non-client area messages. These are used to allow the
434 // minature caption bar to be handled correctly.
435 //
436 case WM_NCCREATE:
437 {
438 DWORD dwStyle ;
439
440 // We have two things that we need to store somewhere:
441 // 1) The caption height (width).
442 // and 2) A flag indicating whether the sysmenu was
443 // just up or not.
444 //
445 // CAPTIONXY is a macro that calls GetSystemMetrics.
446 //
447 SETCAPTIONSIZE( hWnd, CAPTIONXY ) ;
448
449 // Set our flag that tells us whether the system menu was
450 // 'just up'.
451 //
452 SETMENUWASUPFLAG( hWnd, FALSE ) ;
453
454 // Are we in 3.1? If so we have some neat features
455 // we can use like rotated TrueType fonts.
456 //
457 fWin31 = (BOOL)(LOWORD( GetVersion() ) >= 0x030A) ;
458
459 // If IBS_????CAPTION was specified and the WS_DLGFRAME (or
460 // WS_DLGFRAME 'cause it == WS_CAPTION | WS_BORDER)
461 // was specified the creator made a mistake. Things get really
462 // ugly if DefWindowProc sees WS_DLGFRAME, so we strip
463 // the WS_DLGFRAME part off!
464 //
465 dwStyle = GetWindowLong( hWnd, GWL_STYLE ) ;
466 if ((dwStyle & IBS_VERTCAPTION || dwStyle & IBS_HORZCAPTION) &&
467 dwStyle & WS_DLGFRAME)
468 {
469 dwStyle &= (DWORD)~WS_DLGFRAME ;
470 SetWindowLong( hWnd, GWL_STYLE, dwStyle ) ;
471 }
472 }
473 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
474
475 case WM_NCDESTROY:
476 // We store the caption size in a window prop. so we
477 // must remove props.
478 //
479 FREECAPTIONSIZE( hWnd ) ;
480 FREEMENUWASUPFLAG( hWnd ) ;
481 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
482
483 case WM_NCCALCSIZE:
484 // This is sent when the window manager wants to find out
485 // how big our client area is to be. If we have a mini-caption
486 // then we trap this message and calculate the cleint area rect,
487 // which is the client area rect calculated by DefWindowProc()
488 // minus the width/height of the mini-caption bar
489 //
490 lRet = DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
491 if (!IsIconic( hWnd ) && HASCAPTION( hWnd ))
492 {
493 nCapSize = GETCAPTIONSIZE( hWnd ) ;
494
495 if (TestWinStyle( hWnd, IBS_VERTCAPTION ) )
496 ((LPRECT)lParam)->left += nCapSize ;
497 else
498 ((LPRECT)lParam)->top += nCapSize ;
499 }
500 return lRet ;
501
502 case WM_NCHITTEST:
503 // This message is sent whenever the mouse moves over us.
504 // We will depend on DefWindowProc for everything unless
505 // there is a mini-caption, in which case we will
506 // return HTCAPTION or HTSYSMENU. When the user clicks
507 // or double clicks, NC_LBUTTON/ message are sent with
508 // wParam equal to what we return here.
509 // This means that this is an ideal place to figure out
510 // where we are!
511 //
512 // let defwindowproc handle the standard borders etc...
513 //
514 lRet = DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
515 if (!IsIconic( hWnd ) && HASCAPTION( hWnd ) && lRet == HTNOWHERE)
516 {
517 RECT rc ;
518 RECT rcMenu ;
519 RECT rcMinButton ;
520 RECT rcMaxButton ;
521 POINT pt ;
522
523 nCapSize = GETCAPTIONSIZE( hWnd ) ;
524
525 // if DefWindowProc returned HTCAPTION then we have to
526 // refine the area and return HTSYSMENU if appropriate
527 //
528 pt.x = LOWORD( lParam ) ;
529 pt.y = HIWORD( lParam ) ;
530
531 GetCaptionRect( hWnd, &rc ) ; // window coords
532 if (PtInRect( &rc, pt ))
533 {
534 lRet = HTCAPTION ;
535
536 // rely on the fact that Get???Rect() return an invalid
537 // (empty) rectangle if the menu/buttons don't exist
538 //
539 GetIconRect( hWnd, &rcMenu ) ;
540 GetMinButtonRect( hWnd, &rcMinButton ) ;
541 GetMaxButtonRect( hWnd, &rcMaxButton ) ;
542
543 if (PtInRect( &rcMenu, pt ))
544 lRet = HTSYSMENU ;
545
546 if (PtInRect( &rcMinButton, pt ))
547 lRet = HTMINBUTTON ;
548 else
549 if (PtInRect( &rcMaxButton, pt ))
550 lRet = HTMAXBUTTON ;
551 }
552 }
553 if (lRet != HTSYSMENU)
554 SETMENUWASUPFLAG( hWnd, FALSE ) ;
555 return lRet ;
556
557 case WM_NCLBUTTONDBLCLK:
558 // Windows recieve WM_NC?BUTTONDBLCLK messages whether they
559 // have CS_DBLCLKS or not. We watch for one of these
560 // to see if the user double clicked on the system menu (to
561 // close the window) or on the caption (to maximize the window).
562 //
563 if (!IsIconic( hWnd ) && HASCAPTION( hWnd ) && wParam == HTSYSMENU)
564 {
565 SendMessage( hWnd, WM_CLOSE, 0, 0L ) ;
566 break ;
567 }
568 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
569
570 case WM_NCLBUTTONDOWN:
571 {
572 RECT rc ;
573
574 // If we're iconic or we don't have a caption then
575 // DefWindowProc will do the job just fine.
576 //
577 if (IsIconic( hWnd ) || !HASCAPTION( hWnd ))
578 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
579
580 // Here's were we handle the system menu, the min and max buttons.
581 // If you wanted to change the behavior of the min/max buttons
582 // do something like swap tool palettes or something, you
583 // would change the SendMessage() calls below.
584 //
585 switch (wParam)
586 {
587 case HTSYSMENU:
588 if (GETMENUWASUPFLAG( hWnd ) == FALSE && DoMenu( hWnd ))
589 SETMENUWASUPFLAG( hWnd, TRUE ) ;
590 else
591 SETMENUWASUPFLAG( hWnd, FALSE ) ;
592 break ;
593
594 case HTMINBUTTON:
595 GetMinButtonRect( hWnd, &rc ) ;
596 // Note that DepressMinMaxButton() goes into
597 // a PeekMessage() loop waiting for the mouse
598 // to come back up.
599 //
600 if (DepressMinMaxButton( hWnd, wParam, &rc ))
601 SendMessage( hWnd, WM_SYSCOMMAND, SC_MINIMIZE, lParam ) ;
602 break ;
603
604 case HTMAXBUTTON:
605 GetMaxButtonRect( hWnd, &rc ) ;
606 // Note that DepressMinMaxButton() goes into
607 // a PeekMessage() loop waiting for the mouse
608 // to come back up.
609 //
610 if (DepressMinMaxButton( hWnd, wParam, &rc ))
611 {
612 if (IsZoomed(hWnd))
613 SendMessage( hWnd, WM_SYSCOMMAND, SC_RESTORE, lParam ) ;
614 else
615 SendMessage( hWnd, WM_SYSCOMMAND, SC_MAXIMIZE, lParam ) ;
616 }
617 break ;
618
619 default:
620 // Well, it appears as though the user clicked somewhere other
621 // than the buttons. We let DefWindowProc do it's magic.
622 // This is where things like dragging and sizing the
623 // window happen.
624 //
625 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
626 }
627 }
628 break ;
629
630 case WM_NCPAINT:
631 case WM_NCACTIVATE:
632 if (IsIconic( hWnd ))
633 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
634
635 // Paint the non-client area here. We will call DefWindowProc
636 // after we are done so it can paint the borders and so
637 // forth...
638 //
639 lRet = DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
640 if (HASCAPTION( hWnd ))
641 {
642 RECT rcCap ;
643 RECT rc ;
644 HDC hDC = GetWindowDC( hWnd ) ;
645 BOOL fActive ;
646
647 GetCaptionRect( hWnd, &rcCap ) ; // Convert to window coords
648 GetWindowRect( hWnd, &rc ) ;
649 OffsetRect( &rcCap, -rc.left, -rc.top ) ;
650
651 if (uiMsg == WM_NCPAINT)
652 fActive = (hWnd == GetActiveWindow()) ;
653 else
654 fActive = wParam ;
655
656 DrawCaption( hDC, hWnd, &rcCap,
657 TestWinStyle(hWnd, IBS_VERTCAPTION),
658 TestWinStyle(hWnd, WS_SYSMENU),
659 TestWinStyle(hWnd, WS_MINIMIZEBOX),
660 TestWinStyle(hWnd, WS_MAXIMIZEBOX),
661 fActive ) ;
662
663 ReleaseDC( hWnd, hDC ) ;
664 }
665 return lRet;
666 break;
667
668 default:
669 return DefWindowProc( hWnd, uiMsg, wParam, lParam ) ;
670 }
671
672 return 0L ;
673
674 } // ibDefWindowProc()
675
676 // ibAdjustWindowRect( HWND hWnd, LPRECT lprc )
677 //
678 // Does the same thing as the USER function AdjustWindowRect(),
679 // but knows about itsybitsy windows. AdjustWindowRect() is
680 // bogus for stuff like this.
681 //
682 void WINAPI ibAdjustWindowRect( HWND hWnd, LPRECT lprc )
683 {
684 short cx = 0, cy = 0 ;
685 UINT nCapSize ;
686
687 nCapSize = GETCAPTIONSIZE( hWnd ) ;
688
689 // First check Windows's styles, then our own.
690 //
691 if (TestWinStyle( hWnd, WS_THICKFRAME ))
692 {
693 cx = GetSystemMetrics( SM_CXFRAME ) ;
694 cy = GetSystemMetrics( SM_CYFRAME ) ;
695 }
696 else
697 if (TestWinStyle(hWnd, DS_MODALFRAME ))
698 {
699 cx = GetSystemMetrics( SM_CXDLGFRAME ) + GetSystemMetrics( SM_CXBORDER ) ;
700 cy = GetSystemMetrics( SM_CYDLGFRAME ) + GetSystemMetrics( SM_CYBORDER ) ;
701 }
702 else
703 if (TestWinStyle(hWnd, WS_BORDER ))
704 {
705 cx = GetSystemMetrics( SM_CXBORDER ) ;
706 cy = GetSystemMetrics( SM_CYBORDER ) ;
707 }
708
709 InflateRect( lprc, cx, cy ) ;
710
711 if (TestWinStyle( hWnd, IBS_VERTCAPTION ))
712 lprc->left -= nCapSize ;
713 else
714 if (TestWinStyle( hWnd, IBS_HORZCAPTION ))
715 lprc->top -= nCapSize ;
716
717 } // ibAdjustWindowRect()
718
719
720 ///////////////////////////////////////////////////////////////////////
721 // Internal functions
722 ///////////////////////////////////////////////////////////////////////
723
724 // DepressMinMaxButton()
725 //
726 // This function is called when the user has pressed either the min or
727 // max button (i.e. WM_NCLBUTTONDOWN). We go into a Peekmessage() loop,
728 // waiting for the mouse to come back up. This allows us to make the
729 // button change up/down state like a real button does.
730 //
731 // lprc points to the rectangle that describes the button the
732 // user has clicked on.
733 //
734 BOOL PASCAL DepressMinMaxButton( HWND hWnd, UINT uiHT, LPRECT lprc )
735 {
736 BOOL fDepressed = TRUE ;
737 MSG msg ;
738
739 // Draw button in down state
740 DrawButton( NULL, hWnd, uiHT == HTMINBUTTON, fDepressed ) ;
741 SetCapture( hWnd ) ;
742
743 while (TRUE)
744 {
745 if (PeekMessage((LPMSG)&msg, NULL, WM_MOUSEFIRST, WM_MOUSELAST, PM_REMOVE))
746 {
747 switch (msg.message)
748 {
749 case WM_LBUTTONUP:
750 if (fDepressed)
751 DrawButton( NULL, hWnd, uiHT == HTMINBUTTON, !fDepressed ) ;
752 ReleaseCapture();
753 return PtInRect( lprc, msg.pt ) ;
754
755 case WM_MOUSEMOVE:
756 if (PtInRect( lprc, msg.pt ))
757 {
758 if (!fDepressed)
759 DrawButton( NULL, hWnd, uiHT == HTMINBUTTON, fDepressed = TRUE ) ;
760 }
761 else
762 {
763 if (fDepressed)
764 DrawButton( NULL, hWnd, uiHT == HTMINBUTTON, fDepressed = FALSE ) ;
765 }
766 break;
767 }
768 }
769 }
770
771 } // DepressMinMaxButton()
772
773 // DrawCaption( HDC hDC, HWND hWnd, LPRECT lprc,
774 // BOOL fVert, BOOL fSysMenu, BOOL fActive )
775 //
776 // This function draws an itsy bitsy caption bar with or
777 // without system menu to the dc specified by hDC. The
778 // caption is drawn to fit within the lprc RECT and is
779 // drawn//withOut/ borders.
780 //
781 BOOL PASCAL DrawCaption( HDC hDC, HWND hWnd, LPRECT lprc,
782 BOOL fVert, BOOL fSysMenu, BOOL fMin,
783 BOOL fMax, BOOL fActive )
784 {
785 RECT rc ;
786 RECT rcCap ;
787 COLORREF rgbCaptionBG ;
788 COLORREF rgbText ;
789 COLORREF rgbWindowFrame ;
790 HBRUSH hbrCaption ;
791 UINT ui ;
792 UINT nCapSize ;
793
794 nCapSize = GETCAPTIONSIZE( hWnd ) ;
795
796 // Get the colors.
797 //
798 rgbWindowFrame = GetSysColor( COLOR_WINDOWFRAME ) ;
799
800 // if we have focus use the active caption color
801 // otherwise use the inactive caption color
802 //
803 if (fActive)
804 {
805 rgbText = GetSysColor( COLOR_CAPTIONTEXT ) ;
806 rgbCaptionBG = GetSysColor( COLOR_ACTIVECAPTION ) ;
807 }
808 else
809 {
810 if (fWin31)
811 rgbText = GetSysColor( COLOR_INACTIVECAPTIONTEXT ) ;
812 else
813 rgbText = GetSysColor( COLOR_CAPTIONTEXT ) ;
814
815 rgbCaptionBG = GetSysColor( COLOR_INACTIVECAPTION ) ;
816 }
817
818 SetBkMode( hDC, TRANSPARENT ) ;
819 SelectObject( hDC, GetStockObject( NULL_BRUSH ) ) ;
820 SelectObject( hDC, GetStockObject( NULL_PEN ) ) ;
821
822 rcCap = *lprc ;
823
824 if (fSysMenu)
825 {
826 if (fVert)
827 rcCap.top += nCapSize ;
828 else
829 rcCap.left += nCapSize ;
830 }
831
832 if (fMax)
833 {
834 if (fVert)
835 rcCap.bottom -= nCapSize ;
836 else
837 rcCap.right -= nCapSize ;
838 }
839
840 if (fMin)
841 {
842 if (fVert)
843 rcCap.bottom -= nCapSize ;
844 else
845 rcCap.right -= nCapSize ;
846 }
847
848 if (fVert)
849 {
850 rc.left = lprc->right - 1 ;
851 rc.right = lprc->right ;
852 rc.top = lprc->top ;
853 rc.bottom = lprc->bottom ;
854 }
855 else
856 {
857 rc.left = lprc->left ;
858 rc.right = lprc->right ;
859 rc.bottom = lprc->bottom ;
860 rc.top = rc.bottom - 1 ;
861 }
862
863 SetBkColor( hDC, rgbWindowFrame ) ;
864 DRAWFASTRECT( hDC, &rc ) ;
865
866 hbrCaption = CreateSolidBrush( rgbCaptionBG ) ;
867 hbrCaption = (HBRUSH) SelectObject( hDC, hbrCaption ) ;
868 SelectObject( hDC, (HPEN) GetStockObject( NULL_PEN ) ) ;
869 if (fVert)
870 Rectangle( hDC, rcCap.left, rcCap.top, rcCap.right, rcCap.bottom + 1 ) ;
871 else
872 Rectangle( hDC, rcCap.left, rcCap.top, rcCap.right+1, rcCap.bottom ) ;
873 hbrCaption = (HBRUSH) SelectObject( hDC, hbrCaption ) ;
874 DeleteObject( hbrCaption ) ;
875
876
877 // Draw caption text here. Only do it in 3.1 'cause 3.1 gives
878 // us 'small fonts'.
879 //
880 ui = GetWindowTextLength( hWnd ) ;
881 if (fWin31)
882 {
883 HFONT hFont ;
884 LPTSTR lpsz ;
885 LOGFONT lf ;
886 TEXTMETRIC tm ;
887 int cx ;
888 int cy ;
889 SIZE Size ;
890
891 lpsz = (wxChar*)GlobalAllocPtr( GHND, (ui + 2) * sizeof(wxChar) );
892 if (lpsz)
893 {
894 UINT nBkMode ;
895
896 GetWindowText( hWnd, lpsz, ui + 1 ) ;
897 nBkMode = SetBkMode( hDC, TRANSPARENT ) ;
898 rgbText = SetTextColor( hDC, rgbText ) ;
899
900 memset( &lf, '\0', sizeof(LOGFONT) ) ;
901
902 lf.lfHeight = -(int)(nCapSize - 3) ;
903 lf.lfCharSet = ANSI_CHARSET ;
904 lf.lfQuality = DEFAULT_QUALITY ;
905 lf.lfClipPrecision = CLIP_LH_ANGLES | CLIP_STROKE_PRECIS ;
906 if (nCapSize >= 20)
907 {
908 lf.lfWeight = FW_BOLD ;
909 }
910
911 if (fVert)
912 {
913 // Can only rotate true type fonts (well, ok, we could
914 // try and use "modern").
915 wxStrcpy( lf.lfFaceName, wxT("Arial") ) ;
916 lf.lfPitchAndFamily = FF_SWISS | 0x04;
917 lf.lfEscapement = 900 ;
918
919 // Note: The Win 3.1 documentation for CreateFont() say's
920 // that the lfOrientation member is ignored. It appears,
921 // that on Windows 16 3.1 this is true, but when running
922 // as a 16 bit WinApp on WindowsNT 3.1 the lfOrientation
923 // must be set or the text does not rotate!
924 //
925 lf.lfOrientation = 900 ;
926
927 hFont = CreateFontIndirect( &lf ) ;
928 hFont = (HFONT) SelectObject( hDC, hFont ) ;
929
930 GetTextExtentPoint( hDC, lpsz, ui, &Size ) ;
931 cx = rcCap.bottom - ((rcCap.bottom - rcCap.top - Size.cx) / 2) ;
932 cy = rcCap.left - 1 + ((rcCap.right - rcCap.left - Size.cy) / 2) ;
933
934 // Make sure we got a rotatable font back.
935 //
936 GetTextMetrics( hDC, &tm ) ;
937 if (tm.tmPitchAndFamily & TMPF_VECTOR ||
938 tm.tmPitchAndFamily & TMPF_TRUETYPE)
939 {
940 ExtTextOut( hDC,
941 cy,
942 min( (long)cx, rcCap.bottom),
943 ETO_CLIPPED, &rcCap,
944 lpsz, ui, NULL ) ;
945 }
946
947 hFont = (HFONT) SelectObject( hDC, hFont ) ;
948 DeleteObject( hFont ) ;
949 }
950 else
951 {
952 // Use small fonts always for the horizontal. Cause it looks
953 // more like "System" than Arial.
954 //
955 lf.lfPitchAndFamily = FF_SWISS ;
956
957 hFont = CreateFontIndirect( &lf ) ;
958 hFont = (HFONT) SelectObject( hDC, hFont ) ;
959
960 GetTextExtentPoint( hDC, lpsz, ui, &Size ) ;
961 cx = rcCap.left + ((rcCap.right - rcCap.left - Size.cx) / 2) ;
962 cy = rcCap.top + ((rcCap.bottom - rcCap.top - Size.cy) / 2) ;
963
964 // Figger out how big the string is
965 //
966 ExtTextOut( hDC,
967 max( (long)cx, rcCap.left ),
968 cy,
969 ETO_CLIPPED, &rcCap,
970 lpsz, ui, NULL ) ;
971
972 hFont = (HFONT) SelectObject( hDC, hFont ) ;
973 DeleteObject( hFont ) ;
974 }
975
976 // Unsetup the DC
977 //
978 rgbText = SetTextColor( hDC, rgbText ) ;
979 SetBkMode( hDC, nBkMode ) ;
980
981 #ifdef __WIN16__
982 GlobalFreePtr( (unsigned int)(DWORD) lpsz ) ;
983 #else
984 GlobalFreePtr( lpsz ) ;
985 #endif
986 }
987 }
988
989 if (fSysMenu)
990 DrawSysMenu( hDC, hWnd, FALSE ) ;
991
992 if (fMin)
993 DrawButton( hDC, hWnd, TRUE, FALSE ) ;
994
995 if (fMax)
996 DrawButton( hDC, hWnd, FALSE, FALSE ) ;
997
998 return TRUE ;
999
1000 } // DrawCaption()
1001
1002
1003 // DrawSysMenu( HDC hDC, hWnd, BOOL fInvert )
1004 //
1005 // Draws the little system menu icon.
1006 //
1007 void PASCAL DrawSysMenu( HDC hDC, HWND hWnd, BOOL fInvert )
1008 {
1009 RECT rcIcon ;
1010 RECT rcTemp ;
1011 RECT rc ;
1012 COLORREF rgbIconFace ;
1013 COLORREF rgbWindowFrame ;
1014 BOOL fDC ;
1015 UINT nCapSize ;
1016
1017 nCapSize = GETCAPTIONSIZE( hWnd ) ;
1018
1019 if (!hDC)
1020 {
1021 fDC = TRUE ;
1022 hDC = GetWindowDC( hWnd ) ;
1023 }
1024 else
1025 fDC = FALSE ;
1026
1027 if (hDC)
1028 {
1029 rgbIconFace = GetNearestColor( hDC, RGBLTGRAY ) ;
1030 rgbWindowFrame = GetSysColor( COLOR_WINDOWFRAME ) ;
1031
1032 GetIconRect( hWnd, &rcIcon ) ;
1033 GetWindowRect( hWnd, &rc ) ;
1034
1035 OffsetRect( &rcIcon, -rc.left, -rc.top ) ;
1036
1037 rcTemp = rcIcon ;
1038
1039 if (TestWinStyle( hWnd, IBS_VERTCAPTION ))
1040 {
1041 rc = rcIcon ; // separator line
1042 rc.top = ++rc.bottom - 1 ;
1043 }
1044 else
1045 {
1046 rc = rcIcon ; // separator line
1047 rc.left = ++rc.right - 1 ;
1048 }
1049
1050 // Fill
1051 SetBkColor( hDC, rgbIconFace ) ;
1052 DRAWFASTRECT( hDC, &rcTemp ) ;
1053
1054 // Draw separator line
1055 SetBkColor( hDC, rgbWindowFrame ) ;
1056 DRAWFASTRECT( hDC, &rc ) ;
1057
1058 if (nCapSize > 4)
1059 {
1060 // Draw the little horizontal doo-hickey
1061 //
1062 rcTemp.top = rcIcon.top + ((nCapSize-1) / 2) ;
1063 rcTemp.bottom = rcTemp.top + 3 ;
1064 rcTemp.left = rcTemp.left + 3 ;
1065 rcTemp.right = rcTemp.right - 1 ;
1066
1067 SetBkColor( hDC, RGBGRAY ) ;
1068 DRAWFASTRECT( hDC, &rcTemp ) ;
1069
1070 rc = rcTemp ;
1071 OffsetRect( &rc, -1, -1 ) ;
1072 SetBkColor( hDC, RGBBLACK ) ;
1073 DRAWFASTRECT( hDC, &rc ) ;
1074
1075 InflateRect( &rc, -1, -1 ) ;
1076 SetBkColor( hDC, RGBWHITE ) ;
1077 DRAWFASTRECT( hDC, &rc ) ;
1078 }
1079
1080 if (fInvert)
1081 InvertRect( hDC, &rcIcon ) ;
1082
1083 if (fDC)
1084 ReleaseDC( hWnd, hDC ) ;
1085 }
1086
1087 } // DrawSysMenu()
1088
1089 // DoMenu( HWND hWnd )
1090 //
1091 // Pops up the system menu.
1092 //
1093 BOOL PASCAL DoMenu( HWND hWnd )
1094 {
1095 HDC hDC ;
1096 RECT rcIcon ;
1097 RECT rc ;
1098 POINT pt ;
1099 HMENU hMenu ;
1100 DWORD dw ;
1101
1102 if (!TestWinStyle(hWnd, WS_SYSMENU))
1103 return FALSE ;
1104
1105 hDC = GetWindowDC( hWnd );
1106 if (hDC)
1107 {
1108 // Invert the icon
1109 //
1110 DrawSysMenu( hDC, hWnd, TRUE ) ;
1111
1112 // Pop up the menu
1113 //
1114 if (TestWinStyle( hWnd, IBS_VERTCAPTION ))
1115 {
1116 pt.x = -1 ;
1117 pt.y = 0 ;
1118 }
1119 else
1120 {
1121 pt.x = 0 ;
1122 pt.y = -1 ;
1123 }
1124
1125 GetIconRect( hWnd, &rcIcon ) ;
1126 GetWindowRect( hWnd, &rc ) ;
1127 OffsetRect( &rcIcon, -rc.left, -rc.top ) ;
1128
1129 ClientToScreen( hWnd, &pt ) ;
1130 ClientToScreen( hWnd, (LPPOINT)&rc.right ) ;
1131
1132 dw = GetWindowLong( hWnd, GWL_STYLE ) ;
1133 SetWindowLong( hWnd, GWL_STYLE, dw | WS_SYSMENU ) ;
1134
1135 hMenu = GetSystemMenu( hWnd, FALSE ) ;
1136 SetupSystemMenu( hWnd, hMenu ) ;
1137
1138 SetWindowLong( hWnd, GWL_STYLE, dw ) ;
1139
1140 TrackPopupMenu( hMenu, 0, //TPM_LEFTALIGN,
1141 pt.x,
1142 pt.y,
1143 0,
1144 hWnd,
1145 &rc ) ;
1146
1147 DrawSysMenu( hDC, hWnd, FALSE ) ;
1148 ReleaseDC( hWnd, hDC ) ;
1149 }
1150 return TRUE ;
1151
1152 } // DoMenu()
1153
1154 // SetupSystemMenu( HWND hWnd, HMENU hMenu )
1155 //
1156 // Enables/Disables the appropriate menu items on the
1157 // menu passed for the window passed.
1158 //
1159 void PASCAL SetupSystemMenu( HWND hWnd, HMENU hMenu )
1160 {
1161 UINT wMove ;
1162 UINT wSize ;
1163 UINT wMinBox ;
1164 UINT wMaxBox ;
1165 UINT wRestore ;
1166
1167 // Assume all should be grayed.
1168 //
1169 wSize = wMove = wMinBox = wMaxBox = wRestore = MF_GRAYED ;
1170
1171 if (TestWinStyle( hWnd, WS_MAXIMIZEBOX ) || IsIconic( hWnd ))
1172 wMaxBox = MF_ENABLED ;
1173
1174 if (TestWinStyle( hWnd, WS_MINIMIZEBOX ))
1175 wMinBox = MF_ENABLED ;
1176
1177 if (IsZoomed( hWnd ))
1178 wRestore = MF_ENABLED ;
1179
1180 if (TestWinStyle( hWnd, WS_THICKFRAME ) &&
1181 !(IsIconic( hWnd ) || IsZoomed( hWnd )))
1182 wSize = MF_ENABLED ;
1183
1184 if (!IsZoomed( hWnd ) &&
1185 !IsIconic( hWnd ) &&
1186 TestWinStyle( hWnd, WS_CAPTION ) )
1187 wMove = MF_ENABLED ;
1188
1189 EnableMenuItem( hMenu, SC_MOVE, wMove ) ;
1190 EnableMenuItem( hMenu, SC_SIZE, wSize ) ;
1191 EnableMenuItem( hMenu, SC_MINIMIZE, wMinBox ) ;
1192 EnableMenuItem( hMenu, SC_MAXIMIZE, wMaxBox ) ;
1193 EnableMenuItem( hMenu, SC_RESTORE, wRestore ) ;
1194
1195 } // SetupSystemMenu()
1196
1197 // GetCaptionRect( HWND hWnd, LPRECT lprc )
1198 //
1199 // calcluales the rectangle of the mini-caption in screen coords.
1200 //
1201 BOOL PASCAL GetCaptionRect( HWND hWnd, LPRECT lprc )
1202 {
1203 UINT nCapSize ;
1204
1205 nCapSize = GETCAPTIONSIZE( hWnd ) ;
1206
1207 if (!HASCAPTION( hWnd ))
1208 {
1209 SetRectEmpty( lprc ) ;
1210 return FALSE ;
1211 }
1212
1213 GetWindowRect( hWnd, lprc ) ;
1214
1215 // the window might have other non-client components like
1216 // borders
1217 //
1218 if (TestWinStyle( hWnd, WS_THICKFRAME ))
1219 {
1220 lprc->left += GetSystemMetrics( SM_CXFRAME ) ;
1221 lprc->top += GetSystemMetrics( SM_CYFRAME ) ;
1222 lprc->right -= GetSystemMetrics( SM_CXFRAME ) ;
1223 lprc->bottom -= GetSystemMetrics( SM_CYFRAME ) ;
1224 }
1225 else
1226 if (TestWinStyle( hWnd, DS_MODALFRAME )) // if it's a dialog box
1227 {
1228 lprc->left += GetSystemMetrics( SM_CXDLGFRAME ) + GetSystemMetrics( SM_CXBORDER ) ;
1229 lprc->top += GetSystemMetrics( SM_CYDLGFRAME ) + GetSystemMetrics( SM_CYBORDER ) ;
1230 lprc->right -= GetSystemMetrics( SM_CXDLGFRAME ) + GetSystemMetrics( SM_CXBORDER ) ;
1231 lprc->bottom -= GetSystemMetrics( SM_CYDLGFRAME ) + GetSystemMetrics( SM_CYBORDER ) ;
1232 }
1233 else
1234 if (TestWinStyle( hWnd, WS_BORDER ))
1235 {
1236 lprc->left += GetSystemMetrics( SM_CXBORDER ) ;
1237 lprc->top += GetSystemMetrics( SM_CYBORDER ) ;
1238 lprc->right -= GetSystemMetrics( SM_CXBORDER ) ;
1239 lprc->bottom -= GetSystemMetrics( SM_CYBORDER ) ;
1240 }
1241
1242 if (TestWinStyle( hWnd, IBS_VERTCAPTION ))
1243 lprc->right = lprc->left + nCapSize ;
1244 else
1245 lprc->bottom = lprc->top + nCapSize ;
1246
1247 return TRUE ;
1248 } // GetCaptionRect()
1249
1250 // GetIconRect( HWND hWnd, LPRECT lprc )
1251 //
1252 // Calculates the rect of the icon in screen coordinates.
1253 //
1254 BOOL PASCAL GetIconRect( HWND hWnd, LPRECT lprc )
1255 {
1256 UINT nCapSize ;
1257 BOOL fMenu, fVert ;
1258
1259 fMenu= TestWinStyle( hWnd, WS_SYSMENU ) ;
1260 fVert = TestWinStyle( hWnd, IBS_VERTCAPTION ) ;
1261
1262 if (!GetCaptionRect( hWnd, lprc )) // window coords
1263 return FALSE ;
1264
1265 if (!fMenu)
1266 {
1267 SetRectEmpty( lprc ) ;
1268 return FALSE ;
1269 }
1270
1271 nCapSize = GETCAPTIONSIZE( hWnd ) ;
1272
1273 if (fVert)
1274 lprc->bottom = lprc->top + nCapSize ;
1275 else
1276 lprc->right = lprc->left + nCapSize ;
1277
1278 lprc->bottom-- ;
1279 lprc->right-- ;
1280
1281 return TRUE ;
1282
1283 } // GetIconRect()
1284
1285 // GetMinButtonRect()
1286 //
1287 // Calculates the rect of the minimize button in screen
1288 // coordinates.
1289 //
1290 // For horizontal captions, we have the following situation ('Y' is minimize
1291 // and '^' is maximize or restore):
1292 //
1293 // +---------------------------------+
1294 // | - | | Y | ^ |
1295 // +---------------------------------+
1296 // | |.......| <-- This is the width (nSize)
1297 //
1298 // For vertical captions, we have the following:
1299 //
1300 // | |
1301 // | |
1302 // | |
1303 // | |
1304 // | |
1305 // | |
1306 // |--|--
1307 // | Y| .
1308 // |--| . <-- This is the height of the rectangle (nSize)
1309 // | ^| .
1310 // +--+--
1311 //
1312 // In order to figure out where the minimize button goes, we first need
1313 // to know if there is a maximize button. If so, use GetMaxButtonRect()
1314 // to place...
1315 //
1316 BOOL PASCAL GetMinButtonRect( HWND hWnd, LPRECT lprc )
1317 {
1318 if (!TestWinStyle( hWnd, WS_MINIMIZEBOX ))
1319 {
1320 SetRectEmpty( lprc ) ;
1321 return FALSE ;
1322 }
1323
1324 // The minimize button can be in either position 1 or 2. If there
1325 // is a maximize button, it's in position 2.
1326 //
1327 if (TestWinStyle( hWnd, WS_MAXIMIZEBOX ))
1328 return GetButtonRect( hWnd, 2, lprc ) ;
1329 else
1330 return GetButtonRect( hWnd, 1, lprc ) ;
1331 }
1332
1333 // GetMaxButtonRect()
1334 //
1335 // Calculates the rect of the maximize button in screen
1336 // coordinates.
1337 //
1338 // The maximize button, if present, is always to the far right
1339 // or bottom.
1340 //
1341 BOOL PASCAL GetMaxButtonRect( HWND hWnd, LPRECT lprc )
1342 {
1343 //The maximize button can only be in position 1.
1344 //
1345 if (TestWinStyle( hWnd, WS_MAXIMIZEBOX ))
1346 return GetButtonRect( hWnd, 1, lprc ) ;
1347 else
1348 {
1349 SetRectEmpty( lprc ) ;
1350 return FALSE ;
1351 }
1352 }
1353
1354 // Get the rect where a button would go.
1355 //
1356 // This function does not care if it's a min or max, just whether
1357 // it is the first from the right/bottom or second from the right/bottom
1358 // and so on..
1359 //
1360 BOOL PASCAL GetButtonRect( HWND hWnd, UINT nPos, LPRECT lprc )
1361 {
1362 UINT nSize = 0 ;
1363
1364 if (!GetCaptionRect( hWnd, lprc )) //window coords
1365 return FALSE ;
1366
1367 nSize = GETCAPTIONSIZE( hWnd ) ;
1368
1369 if (TestWinStyle( hWnd, IBS_VERTCAPTION ))
1370 {
1371 lprc->bottom -= nSize * (nPos-1) ;
1372 lprc->top = lprc->bottom - nSize + 1 ;
1373 }
1374 else
1375 {
1376 lprc->right -= nSize * (nPos-1) ;
1377 lprc->left = lprc->right - nSize + 1 ;
1378 }
1379
1380 return TRUE ;
1381 } // GetButtonRect()
1382
1383 // DrawButton( HDC hDC, HWND hWnd, BOOL fMin, BOOL fDepressed )
1384 //
1385 // Draws either the min, max, or restore buttons. If fMin is FALSE then it
1386 // will draw either the Max or Restore button. If fDepressed is TRUE it will
1387 // draw the button in a down state.
1388 //
1389 void PASCAL DrawButton( HDC hDC, HWND hWnd, BOOL fMin, BOOL fDepressed)
1390 {
1391 RECT rcButton ;
1392 RECT rc ;
1393 COLORREF rgbWindowFrame ;
1394 BOOL fDC ;
1395 UINT nCapSize ;
1396 UINT nOffset ;
1397 int n ;
1398
1399 nCapSize = GETCAPTIONSIZE( hWnd ) ;
1400
1401 // If you look at the standard Windows' min/max buttons, you will notice
1402 // that they have two pixels of 'shadow' to the bottom and right. Since
1403 // our buttons can be really, really small, we only want one pixel of
1404 // shadow when they are small. I arbitrarily decided that if the
1405 // caption size is greater than or equal to 20 we will use two
1406 // pixels. That's what this THREASHOLD stuff does.
1407 //
1408 #define THRESHOLD 20
1409 nOffset = (nCapSize >= THRESHOLD) ? 2 : 1 ;
1410
1411 if (!hDC)
1412 {
1413 fDC = TRUE ;
1414 hDC = GetWindowDC( hWnd ) ;
1415 }
1416 else
1417 fDC = FALSE ;
1418
1419 if (hDC)
1420 {
1421 rgbWindowFrame = GetSysColor( COLOR_WINDOWFRAME ) ;
1422
1423 if (fMin)
1424 GetMinButtonRect( hWnd, &rcButton ) ;
1425 else
1426 GetMaxButtonRect( hWnd, &rcButton ) ;
1427
1428 GetWindowRect( hWnd, &rc ) ;
1429 OffsetRect( &rcButton, -rc.left, -rc.top ) ;
1430
1431 rc = rcButton ;
1432 if (TestWinStyle( hWnd, IBS_VERTCAPTION ))
1433 {
1434 rc = rcButton ; //separator line
1435 rc.bottom = --rc.top + 1 ;
1436 rcButton.right-- ;
1437 }
1438 else
1439 {
1440 rc = rcButton ; //separator line
1441 rc.right = --rc.left + 1 ;
1442 rcButton.bottom-- ;
1443 }
1444
1445 //Draw separator line
1446 SetBkColor( hDC, rgbWindowFrame ) ;
1447 DRAWFASTRECT( hDC, &rc ) ;
1448
1449 //Fill
1450 SetBkColor( hDC, RGBLTGRAY ) ;
1451 DRAWFASTRECT( hDC, &rcButton ) ;
1452
1453 if (!fDepressed)
1454 {
1455 //The normal min/max buttons have one pixel on the top and left
1456 //sides for the highlight, and two pixels on the bottom and
1457 //right side for the shadow.
1458 //
1459 //When our caption is 'small' we only use one pixel on all
1460 //sides.
1461 //
1462 SetBkColor( hDC, RGBWHITE ) ;
1463 //Draw left side
1464 rc = rcButton ;
1465 rc.right = rc.left + 1 ;
1466 DRAWFASTRECT( hDC, &rc ) ;
1467
1468 //Draw Top
1469 rc = rcButton ;
1470 rc.bottom = rc.top + 1 ;
1471 DRAWFASTRECT( hDC, &rc ) ;
1472
1473 SetBkColor( hDC, RGBGRAY ) ;
1474 //Draw right side
1475 rc = rcButton ;
1476 rc.left = rc.right - 1 ;
1477 DRAWFASTRECT( hDC, &rc ) ;
1478 if (nCapSize > THRESHOLD)
1479 {
1480 rc.left-- ;
1481 rc.top++ ;
1482 DRAWFASTRECT( hDC, &rc ) ;
1483 }
1484
1485 //Draw bottom
1486 rc = rcButton ;
1487 rc.top = rc.bottom - 1 ;
1488 DRAWFASTRECT( hDC, &rc ) ;
1489 if (nCapSize > THRESHOLD)
1490 {
1491 rc.top-- ;
1492 rc.left++ ;
1493 DRAWFASTRECT( hDC, &rc ) ;
1494 }
1495
1496 rcButton.left++ ;
1497 rcButton.top++ ;
1498 rcButton.right -= nOffset ;
1499 rcButton.bottom -= nOffset ;
1500 }
1501 else
1502 {
1503 //Draw depressed state
1504
1505 SetBkColor( hDC, RGBGRAY ) ;
1506 //Draw left side
1507 rc = rcButton ;
1508 rc.right = rc.left + nOffset ;
1509 DRAWFASTRECT( hDC, &rc ) ;
1510
1511 //Draw Top
1512 rc = rcButton ;
1513 rc.bottom = rc.top + nOffset ;
1514 DRAWFASTRECT( hDC, &rc ) ;
1515
1516 rcButton.left += 2 * nOffset ;
1517 rcButton.top += 2 * nOffset ;
1518 }
1519
1520 // Now draw the arrows. We do not want the
1521 // arrows to grow too large when we have a bigger than
1522 // normal caption, so we restrict their size.
1523 //
1524 // rcButton now represents where we can place our
1525 // arrows.
1526 //
1527 // The maximum size of our arrows (i.e. the width of rcButton)
1528 // has been empirically determined to be SM_CYCAPTION / 2
1529 //
1530 n = ((GetSystemMetrics( SM_CYCAPTION )) / 2) -
1531 (rcButton.right - rcButton.left) ;
1532 if (n < 1)
1533 InflateRect( &rcButton, n/2-1, n/2-1 ) ;
1534
1535 if (fMin)
1536 DrawArrow( hDC, &rcButton, ARROW_DOWN ) ;
1537 else
1538 if (IsZoomed( hWnd ))
1539 {
1540 DrawArrow( hDC, &rcButton, ARROW_RESTORE ) ;
1541 }
1542 else
1543 DrawArrow( hDC, &rcButton, ARROW_UP ) ;
1544
1545 if (fDC)
1546 ReleaseDC( hWnd, hDC ) ;
1547 }
1548
1549 } // DrawButton()
1550
1551
1552 // DrawArrow
1553 //
1554 // Draws either a up or down arrow. The arrow is bound by the rectangle
1555 //
1556 void PASCAL DrawArrow( HDC hdc, LPRECT lprc, UINT uiStyle )
1557 {
1558 int row ;
1559 int xTip ;
1560 int yTip ;
1561 RECT rc ;
1562 int nMax = (lprc->bottom - lprc->top) >> 1 ;
1563
1564 SetBkColor( hdc, RGBBLACK ) ;
1565
1566 // We draw the arrow by drawing a series of horizontal lines
1567 //
1568 xTip = lprc->left + ((lprc->right - lprc->left+1) >> 1) ;
1569 switch (uiStyle)
1570 {
1571 case ARROW_UP:
1572 yTip = lprc->top + ((lprc->bottom - lprc->top-1) >> 2) ;
1573 for (row = 1 ; row <= nMax ; row++ )
1574 {
1575 rc.left = xTip - row ;
1576 rc.right = xTip + row - 1 ;
1577 rc.top = yTip + row ;
1578 rc.bottom = rc.top + 1 ;
1579 DRAWFASTRECT( hdc, &rc ) ;
1580 }
1581 break ;
1582
1583 case ARROW_DOWN:
1584 yTip = lprc->bottom - ((lprc->bottom - lprc->top-1) >> 2) ;
1585 for ( row = nMax ; row > 0 ; row-- )
1586 {
1587 rc.left = xTip - row ;
1588 rc.right = xTip + row - 1 ;
1589 rc.top = yTip - row ;
1590 rc.bottom = rc.top + 1 ;
1591 DRAWFASTRECT( hdc, &rc ) ;
1592 }
1593 break ;
1594
1595 case ARROW_RESTORE:
1596 default:
1597 yTip = lprc->top + ((lprc->bottom - lprc->top-1) >> 3) - 2;
1598 for (row = 1 ; row <= nMax ; row++ )
1599 {
1600 rc.left = xTip - row ;
1601 rc.right = xTip + row - 1 ;
1602 rc.top = yTip + row ;
1603 rc.bottom = rc.top + 1 ;
1604 DRAWFASTRECT( hdc, &rc ) ;
1605 }
1606
1607 yTip += (nMax+1) * 2 ;
1608 for ( row = nMax ; row > 0 ; row-- )
1609 {
1610 rc.left = xTip - row ;
1611 rc.right = xTip + row - 1 ;
1612 rc.top = yTip - row ;
1613 rc.bottom = rc.top + 1 ;
1614 DRAWFASTRECT( hdc, &rc ) ;
1615 }
1616 break ;
1617 }
1618
1619 } // DrawArrow()
1620
1621 #endif // wxUSE_ITSY_BITSY
1622
1623 #endif // __WIN32__