]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/msw/renderer.cpp
use first unused colour for the mask instead of hardcoded #ff00ff (patch 1746895)
[wxWidgets.git] / src / msw / renderer.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2// Name: src/msw/renderer.cpp
3// Purpose: implementation of wxRendererNative for Windows
4// Author: Vadim Zeitlin
5// Modified by:
6// Created: 20.07.2003
7// RCS-ID: $Id$
8// Copyright: (c) 2003 Vadim Zeitlin <vadim@wxwindows.org>
9// License: wxWindows licence
10///////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20// for compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#ifndef WX_PRECOMP
28 #include "wx/string.h"
29 #include "wx/window.h"
30 #include "wx/dc.h"
31 #include "wx/settings.h"
32#endif //WX_PRECOMP
33
34#include "wx/splitter.h"
35#include "wx/renderer.h"
36#include "wx/msw/private.h"
37#include "wx/msw/uxtheme.h"
38
39// tmschema.h is in Win32 Platform SDK and might not be available with earlier
40// compilers
41#ifndef CP_DROPDOWNBUTTON
42 #define BP_PUSHBUTTON 1
43 #define BP_CHECKBOX 3
44 #define CBS_UNCHECKEDNORMAL 1
45 #define CBS_CHECKEDNORMAL (CBS_UNCHECKEDNORMAL + 4)
46 #define CBS_MIXEDNORMAL (CBS_CHECKEDNORMAL + 4)
47
48 #define PBS_NORMAL 1
49 #define PBS_HOT 2
50 #define PBS_PRESSED 3
51 #define PBS_DISABLED 4
52 #define PBS_DEFAULTED 5
53
54 #define CP_DROPDOWNBUTTON 1
55
56 #define CBXS_NORMAL 1
57 #define CBXS_HOT 2
58 #define CBXS_PRESSED 3
59 #define CBXS_DISABLED 4
60
61 #define TVP_GLYPH 2
62
63 #define GLPS_CLOSED 1
64 #define GLPS_OPENED 2
65
66 #define HP_HEADERITEM 1
67
68 #define HIS_NORMAL 1
69 #define HIS_HOT 2
70 #define HIS_PRESSED 3
71
72 #define TMT_HEIGHT 2417
73
74 #define HP_HEADERSORTARROW 4
75 #define HSAS_SORTEDUP 1
76 #define HSAS_SORTEDDOWN 2
77#endif
78
79#if defined(__WXWINCE__) && !defined(DFCS_FLAT)
80 #define DFCS_FLAT 0
81#endif
82
83#ifndef DFCS_HOT
84 #define DFCS_HOT 0x1000
85#endif
86
87// ----------------------------------------------------------------------------
88// wxRendererMSW: wxRendererNative implementation for "old" Win32 systems
89// ----------------------------------------------------------------------------
90
91class WXDLLEXPORT wxRendererMSW : public wxDelegateRendererNative
92{
93public:
94 wxRendererMSW() { }
95
96 static wxRendererNative& Get();
97
98 virtual void DrawComboBoxDropButton(wxWindow *win,
99 wxDC& dc,
100 const wxRect& rect,
101 int flags = 0);
102
103 virtual void DrawCheckBox(wxWindow *win,
104 wxDC& dc,
105 const wxRect& rect,
106 int flags = 0);
107
108 virtual void DrawPushButton(wxWindow *win,
109 wxDC& dc,
110 const wxRect& rect,
111 int flags = 0);
112
113private:
114 DECLARE_NO_COPY_CLASS(wxRendererMSW)
115};
116
117// ----------------------------------------------------------------------------
118// wxRendererXP: wxRendererNative implementation for Windows XP and later
119// ----------------------------------------------------------------------------
120
121#if wxUSE_UXTHEME
122
123class WXDLLEXPORT wxRendererXP : public wxDelegateRendererNative
124{
125public:
126 wxRendererXP() : wxDelegateRendererNative(wxRendererMSW::Get()) { }
127
128 static wxRendererNative& Get();
129
130 virtual int DrawHeaderButton(wxWindow *win,
131 wxDC& dc,
132 const wxRect& rect,
133 int flags = 0,
134 wxHeaderSortIconType sortArrow = wxHDR_SORT_ICON_NONE,
135 wxHeaderButtonParams* params = NULL);
136 virtual int GetHeaderButtonHeight(wxWindow *win);
137
138 virtual void DrawTreeItemButton(wxWindow *win,
139 wxDC& dc,
140 const wxRect& rect,
141 int flags = 0);
142 virtual void DrawSplitterBorder(wxWindow *win,
143 wxDC& dc,
144 const wxRect& rect,
145 int flags = 0);
146 virtual void DrawSplitterSash(wxWindow *win,
147 wxDC& dc,
148 const wxSize& size,
149 wxCoord position,
150 wxOrientation orient,
151 int flags = 0);
152 virtual void DrawComboBoxDropButton(wxWindow *win,
153 wxDC& dc,
154 const wxRect& rect,
155 int flags = 0);
156 virtual void DrawCheckBox(wxWindow *win,
157 wxDC& dc,
158 const wxRect& rect,
159 int flags = 0);
160
161 virtual void DrawPushButton(wxWindow *win,
162 wxDC& dc,
163 const wxRect& rect,
164 int flags = 0);
165
166 virtual void DrawItemSelectionRect(wxWindow *win,
167 wxDC& dc,
168 const wxRect& rect,
169 int flags = 0 );
170
171
172 virtual wxSplitterRenderParams GetSplitterParams(const wxWindow *win);
173private:
174 DECLARE_NO_COPY_CLASS(wxRendererXP)
175};
176
177#endif // wxUSE_UXTHEME
178
179// ============================================================================
180// wxRendererNative and wxRendererMSW implementation
181// ============================================================================
182
183/* static */
184wxRendererNative& wxRendererNative::GetDefault()
185{
186#if wxUSE_UXTHEME
187 wxUxThemeEngine *themeEngine = wxUxThemeEngine::Get();
188 if ( themeEngine && themeEngine->IsAppThemed() )
189 return wxRendererXP::Get();
190#endif // wxUSE_UXTHEME
191
192 return wxRendererMSW::Get();
193}
194
195/* static */
196wxRendererNative& wxRendererMSW::Get()
197{
198 static wxRendererMSW s_rendererMSW;
199
200 return s_rendererMSW;
201}
202
203void
204wxRendererMSW::DrawComboBoxDropButton(wxWindow * WXUNUSED(win),
205 wxDC& dc,
206 const wxRect& rect,
207 int flags)
208{
209 RECT r;
210 wxCopyRectToRECT(rect, r);
211
212 int style = DFCS_SCROLLCOMBOBOX;
213 if ( flags & wxCONTROL_DISABLED )
214 style |= DFCS_INACTIVE;
215 if ( flags & wxCONTROL_PRESSED )
216 style |= DFCS_PUSHED | DFCS_FLAT;
217
218 ::DrawFrameControl(GetHdcOf(dc), &r, DFC_SCROLL, style);
219}
220
221void
222wxRendererMSW::DrawCheckBox(wxWindow * WXUNUSED(win),
223 wxDC& dc,
224 const wxRect& rect,
225 int flags)
226{
227 RECT r;
228 wxCopyRectToRECT(rect, r);
229
230 int style = DFCS_BUTTONCHECK;
231 if ( flags & wxCONTROL_CHECKED )
232 style |= DFCS_CHECKED;
233 if ( flags & wxCONTROL_DISABLED )
234 style |= DFCS_INACTIVE;
235 if ( flags & wxCONTROL_FLAT )
236 style |= DFCS_MONO;
237 if ( flags & wxCONTROL_PRESSED )
238 style |= DFCS_PUSHED;
239 if ( flags & wxCONTROL_CURRENT )
240 style |= DFCS_HOT;
241
242 ::DrawFrameControl(GetHdcOf(dc), &r, DFC_BUTTON, style);
243}
244
245void
246wxRendererMSW::DrawPushButton(wxWindow * WXUNUSED(win),
247 wxDC& dc,
248 const wxRect& rectOrig,
249 int flags)
250{
251 wxRect rect(rectOrig);
252
253 int style = DFCS_BUTTONPUSH;
254 if ( flags & wxCONTROL_DISABLED )
255 style |= DFCS_INACTIVE;
256 if ( flags & wxCONTROL_PRESSED )
257 style |= DFCS_PUSHED | DFCS_FLAT;
258 if ( flags & wxCONTROL_ISDEFAULT )
259 {
260 // DrawFrameControl() doesn't seem to support default buttons so we
261 // have to draw the border ourselves
262 wxDCPenChanger pen(dc, *wxBLACK_PEN);
263 wxDCBrushChanger brush(dc, *wxTRANSPARENT_BRUSH);
264 dc.DrawRectangle(rect);
265 rect.Deflate(1);
266 }
267
268 RECT rc;
269 wxCopyRectToRECT(rect, rc);
270
271 ::DrawFrameControl(GetHdcOf(dc), &rc, DFC_BUTTON, style);
272}
273
274// ============================================================================
275// wxRendererXP implementation
276// ============================================================================
277
278#if wxUSE_UXTHEME
279
280/* static */
281wxRendererNative& wxRendererXP::Get()
282{
283 static wxRendererXP s_rendererXP;
284
285 return s_rendererXP;
286}
287
288// NOTE: There is no guarantee that the button drawn fills the entire rect (XP
289// default theme, for example), so the caller should have cleared button's
290// background before this call. This is quite likely a wxMSW-specific thing.
291void
292wxRendererXP::DrawComboBoxDropButton(wxWindow * win,
293 wxDC& dc,
294 const wxRect& rect,
295 int flags)
296{
297 wxUxThemeHandle hTheme(win, L"COMBOBOX");
298 if ( !hTheme )
299 {
300 m_rendererNative.DrawComboBoxDropButton(win, dc, rect, flags);
301 return;
302 }
303
304 RECT r;
305 wxCopyRectToRECT(rect, r);
306
307 int state;
308 if ( flags & wxCONTROL_PRESSED )
309 state = CBXS_PRESSED;
310 else if ( flags & wxCONTROL_CURRENT )
311 state = CBXS_HOT;
312 else if ( flags & wxCONTROL_DISABLED )
313 state = CBXS_DISABLED;
314 else
315 state = CBXS_NORMAL;
316
317 wxUxThemeEngine::Get()->DrawThemeBackground
318 (
319 hTheme,
320 GetHdcOf(dc),
321 CP_DROPDOWNBUTTON,
322 state,
323 &r,
324 NULL
325 );
326
327}
328
329int
330wxRendererXP::DrawHeaderButton(wxWindow *win,
331 wxDC& dc,
332 const wxRect& rect,
333 int flags,
334 wxHeaderSortIconType sortArrow,
335 wxHeaderButtonParams* params)
336{
337 wxUxThemeHandle hTheme(win, L"HEADER");
338 if ( !hTheme )
339 {
340 return m_rendererNative.DrawHeaderButton(win, dc, rect, flags, sortArrow, params);
341 }
342
343 RECT r;
344 wxCopyRectToRECT(rect, r);
345
346 int state;
347 if ( flags & wxCONTROL_PRESSED )
348 state = HIS_PRESSED;
349 else if ( flags & wxCONTROL_CURRENT )
350 state = HIS_HOT;
351 else
352 state = HIS_NORMAL;
353 wxUxThemeEngine::Get()->DrawThemeBackground
354 (
355 hTheme,
356 GetHdcOf(dc),
357 HP_HEADERITEM,
358 state,
359 &r,
360 NULL
361 );
362
363 // NOTE: Using the theme to draw HP_HEADERSORTARROW doesn't do anything.
364 // Why? If this can be fixed then draw the sort arrows using the theme
365 // and then clear those flags before calling DrawHeaderButtonContents.
366
367 // Add any extras that are specified in flags and params
368 return DrawHeaderButtonContents(win, dc, rect, flags, sortArrow, params);
369}
370
371
372int
373wxRendererXP::GetHeaderButtonHeight(wxWindow *win)
374{
375 wxUxThemeHandle hTheme(win, L"HEADER");
376 if ( !hTheme )
377 {
378 return m_rendererNative.GetHeaderButtonHeight(win);
379 }
380
381 HRESULT hr;
382 int value = -1;
383
384 hr = wxUxThemeEngine::Get()->GetThemeMetric( hTheme,
385 NULL,
386 HP_HEADERITEM,
387 HIS_NORMAL,
388 TMT_HEIGHT,
389 &value );
390 if ( hr == S_OK )
391 return value;
392 else
393 return 20;
394}
395
396
397void
398wxRendererXP::DrawTreeItemButton(wxWindow *win,
399 wxDC& dc,
400 const wxRect& rect,
401 int flags)
402{
403 wxUxThemeHandle hTheme(win, L"TREEVIEW");
404 if ( !hTheme )
405 {
406 m_rendererNative.DrawTreeItemButton(win, dc, rect, flags);
407 return;
408 }
409
410 RECT r;
411 wxCopyRectToRECT(rect, r);
412
413 int state = flags & wxCONTROL_EXPANDED ? GLPS_OPENED : GLPS_CLOSED;
414 wxUxThemeEngine::Get()->DrawThemeBackground
415 (
416 hTheme,
417 GetHdcOf(dc),
418 TVP_GLYPH,
419 state,
420 &r,
421 NULL
422 );
423}
424
425void
426wxRendererXP::DrawCheckBox(wxWindow *win,
427 wxDC& dc,
428 const wxRect& rect,
429 int flags)
430{
431 wxUxThemeHandle hTheme(win, L"BUTTON");
432 if ( !hTheme )
433 {
434 m_rendererNative.DrawCheckBox(win, dc, rect, flags);
435 return;
436 }
437
438 RECT r;
439 wxCopyRectToRECT(rect, r);
440
441 int state;
442 if ( flags & wxCONTROL_CHECKED )
443 state = CBS_CHECKEDNORMAL;
444 else if ( flags & wxCONTROL_UNDETERMINED )
445 state = CBS_MIXEDNORMAL;
446 else
447 state = CBS_UNCHECKEDNORMAL;
448
449 // CBS_XXX is followed by CBX_XXXHOT, then CBS_XXXPRESSED and DISABLED
450 enum
451 {
452 CBS_HOT_OFFSET = 1,
453 CBS_PRESSED_OFFSET = 2,
454 CBS_DISABLED_OFFSET = 3
455 };
456
457 if ( flags & wxCONTROL_DISABLED )
458 state += CBS_DISABLED_OFFSET;
459 else if ( flags & wxCONTROL_PRESSED )
460 state += CBS_PRESSED_OFFSET;
461 else if ( flags & wxCONTROL_CURRENT )
462 state += CBS_HOT_OFFSET;
463
464 wxUxThemeEngine::Get()->DrawThemeBackground
465 (
466 hTheme,
467 GetHdcOf(dc),
468 BP_CHECKBOX,
469 state,
470 &r,
471 NULL
472 );
473}
474
475void
476wxRendererXP::DrawPushButton(wxWindow * win,
477 wxDC& dc,
478 const wxRect& rect,
479 int flags)
480{
481 wxUxThemeHandle hTheme(win, L"BUTTON");
482 if ( !hTheme )
483 {
484 m_rendererNative.DrawPushButton(win, dc, rect, flags);
485 return;
486 }
487
488 RECT r;
489 wxCopyRectToRECT(rect, r);
490
491 int state;
492 if ( flags & wxCONTROL_PRESSED )
493 state = PBS_PRESSED;
494 else if ( flags & wxCONTROL_CURRENT )
495 state = PBS_HOT;
496 else if ( flags & wxCONTROL_DISABLED )
497 state = PBS_DISABLED;
498 else if ( flags & wxCONTROL_ISDEFAULT )
499 state = PBS_DEFAULTED;
500 else
501 state = PBS_NORMAL;
502
503 wxUxThemeEngine::Get()->DrawThemeBackground
504 (
505 hTheme,
506 GetHdcOf(dc),
507 BP_PUSHBUTTON,
508 state,
509 &r,
510 NULL
511 );
512
513}
514
515void
516wxRendererXP::DrawItemSelectionRect(wxWindow * WXUNUSED(win),
517 wxDC& dc,
518 const wxRect& rect,
519 int flags)
520{
521 wxBrush brush;
522 if ( flags & wxCONTROL_SELECTED )
523 {
524 if ( flags & wxCONTROL_FOCUSED )
525 {
526 brush = wxBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT));
527 }
528 else // !focused
529 {
530 brush = wxBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNSHADOW));
531 }
532 }
533 else // !selected
534 {
535 brush = *wxTRANSPARENT_BRUSH;
536 }
537
538 dc.SetBrush(brush);
539
540 // unlike for wxRendererGeneric, on windows we _never_ want to draw
541 // the outline of the rectangle:
542 dc.SetPen(*wxTRANSPARENT_PEN);
543
544 dc.DrawRectangle( rect );
545}
546
547
548
549// ----------------------------------------------------------------------------
550// splitter drawing
551// ----------------------------------------------------------------------------
552
553// the width of the sash: this is the same as used by Explorer...
554static const wxCoord SASH_WIDTH = 4;
555
556wxSplitterRenderParams
557wxRendererXP::GetSplitterParams(const wxWindow * win)
558{
559 if ( win->HasFlag(wxSP_NO_XP_THEME) )
560 return m_rendererNative.GetSplitterParams(win);
561 else
562 return wxSplitterRenderParams(SASH_WIDTH, 0, false);
563}
564
565void
566wxRendererXP::DrawSplitterBorder(wxWindow * win,
567 wxDC& dc,
568 const wxRect& rect,
569 int flags)
570{
571 if ( win->HasFlag(wxSP_NO_XP_THEME) )
572 {
573 m_rendererNative.DrawSplitterBorder(win, dc, rect, flags);
574 }
575}
576
577void
578wxRendererXP::DrawSplitterSash(wxWindow *win,
579 wxDC& dc,
580 const wxSize& size,
581 wxCoord position,
582 wxOrientation orient,
583 int flags)
584{
585 if ( !win->HasFlag(wxSP_NO_XP_THEME) )
586 {
587 dc.SetPen(*wxTRANSPARENT_PEN);
588 dc.SetBrush(wxBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_BTNFACE)));
589 if ( orient == wxVERTICAL )
590 {
591 dc.DrawRectangle(position, 0, SASH_WIDTH, size.y);
592 }
593 else // wxHORIZONTAL
594 {
595 dc.DrawRectangle(0, position, size.x, SASH_WIDTH);
596 }
597
598 return;
599 }
600
601 m_rendererNative.DrawSplitterSash(win, dc, size, position, orient, flags);
602}
603
604#endif // wxUSE_UXTHEME