// Author: Vadim Zeitlin
// Modified by:
// Created: 20.07.2003
-// RCS-ID: $Id$
// Copyright: (c) 2003 Vadim Zeitlin <vadim@wxwindows.org>
-// License: wxWindows licence
+// Licence: wxWindows licence
///////////////////////////////////////////////////////////////////////////////
// ============================================================================
#include "wx/settings.h"
#endif //WX_PRECOMP
+#include "wx/dcgraph.h"
#include "wx/scopeguard.h"
#include "wx/splitter.h"
#include "wx/renderer.h"
#include "wx/msw/private.h"
-#include "wx/msw/dc.h"
#include "wx/msw/uxtheme.h"
-#if wxUSE_GRAPHICS_CONTEXT
-// TODO remove this dependency (gdiplus needs the macros)
-#ifndef max
-#define max(a,b) (((a) > (b)) ? (a) : (b))
-#endif
-
-#ifndef min
-#define min(a,b) (((a) < (b)) ? (a) : (b))
-#endif
-
-#include "wx/dcgraph.h"
-#include "gdiplus.h"
-using namespace Gdiplus;
-#endif // wxUSE_GRAPHICS_CONTEXT
-
// tmschema.h is in Win32 Platform SDK and might not be available with earlier
// compilers
#ifndef CP_DROPDOWNBUTTON
#define TMT_TEXTCOLOR 3803
#define TMT_BORDERCOLOR 3801
#define TMT_EDGEFILLCOLOR 3808
-#endif
-
-
-// ----------------------------------------------------------------------------
-// If the DC is a wxGCDC then pull out the HDC from the GraphicsContext when
-// it is needed, and handle the Release when done.
-class GraphicsHDC
-{
-public:
- GraphicsHDC(wxDC* dc)
- {
-#if wxUSE_GRAPHICS_CONTEXT
- m_graphics = NULL;
- wxGCDC* gcdc = wxDynamicCast(dc, wxGCDC);
- if (gcdc) {
- m_graphics = (Graphics*)gcdc->GetGraphicsContext()->GetNativeContext();
- m_hdc = m_graphics->GetHDC();
- }
- else
+ #define WP_MINBUTTON 15
+ #define WP_MAXBUTTON 17
+ #define WP_CLOSEBUTTON 18
+ #define WP_RESTOREBUTTON 21
+ #define WP_HELPBUTTON 23
#endif
- m_hdc = GetHdcOf(*((wxMSWDCImpl*)dc->GetImpl()));
- }
-
- ~GraphicsHDC()
- {
-#if wxUSE_GRAPHICS_CONTEXT
- if (m_graphics)
- m_graphics->ReleaseHDC(m_hdc);
-#endif
- }
-
- operator HDC() const { return m_hdc; }
-
-private:
- HDC m_hdc;
-#if wxUSE_GRAPHICS_CONTEXT
- Graphics* m_graphics;
-#endif
-};
#if defined(__WXWINCE__)
#ifndef DFCS_FLAT
wxDC& dc,
const wxRect& rect,
int flags = 0);
+
+ void DrawChoice(wxWindow* win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags = 0);
+
+ void DrawComboBox(wxWindow* win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags = 0);
+
+ virtual void DrawComboBoxDropButton(wxWindow *win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags = 0) = 0;
+
+ virtual void DrawTextCtrl(wxWindow* win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags = 0) = 0;
};
// ----------------------------------------------------------------------------
const wxRect& rect,
int flags = 0);
- virtual void DrawChoice(wxWindow* win,
- wxDC& dc,
- const wxRect& rect,
- int flags = 0);
-
- virtual void DrawComboBox(wxWindow* win,
- wxDC& dc,
- const wxRect& rect,
- int flags = 0);
-
virtual void DrawTextCtrl(wxWindow* win,
wxDC& dc,
const wxRect& rect,
DoDrawButton(DFCS_BUTTONRADIO, win, dc, rect, flags);
}
+ virtual void DrawTitleBarBitmap(wxWindow *win,
+ wxDC& dc,
+ const wxRect& rect,
+ wxTitleBarButton button,
+ int flags = 0);
+
virtual wxSize GetCheckBoxSize(wxWindow *win);
virtual int GetHeaderButtonHeight(wxWindow *win);
+ virtual int GetHeaderButtonMargin(wxWindow *win);
+
private:
+ // wrapper of DrawFrameControl()
+ void DoDrawFrameControl(UINT type,
+ UINT kind,
+ wxWindow *win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags);
+
// common part of Draw{PushButton,CheckBox,RadioBitmap}(): wraps
// DrawFrameControl(DFC_BUTTON)
void DoDrawButton(UINT kind,
wxWindow *win,
wxDC& dc,
const wxRect& rect,
- int flags);
+ int flags)
+ {
+ DoDrawFrameControl(DFC_BUTTON, kind, win, dc, rect, flags);
+ }
wxDECLARE_NO_COPY_CLASS(wxRendererMSW);
};
m_rendererNative.DrawPushButton(win, dc, rect, flags);
}
+ virtual void DrawTextCtrl(wxWindow* win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags = 0);
+
virtual void DrawRadioBitmap(wxWindow *win,
wxDC& dc,
const wxRect& rect,
m_rendererNative.DrawRadioBitmap(win, dc, rect, flags);
}
+ virtual void DrawTitleBarBitmap(wxWindow *win,
+ wxDC& dc,
+ const wxRect& rect,
+ wxTitleBarButton button,
+ int flags = 0);
+
virtual wxSplitterRenderParams GetSplitterParams(const wxWindow *win);
private:
+ // wrapper around DrawThemeBackground() translating flags to NORMAL/HOT/
+ // PUSHED/DISABLED states (and so suitable for drawing anything
+ // button-like)
+ void DoDrawButtonLike(HTHEME htheme,
+ int part,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags);
+
// common part of DrawCheckBox(), DrawPushButton() and DrawRadioBitmap()
bool DoDrawXPButton(int kind,
wxWindow *win,
RECT rc;
wxCopyRectToRECT(rect, rc);
- ::DrawFocusRect(GraphicsHDC(&dc), &rc);
+ ::DrawFocusRect(GetHdcOf(dc.GetTempHDC()), &rc);
}
void wxRendererMSWBase::DrawItemSelectionRect(wxWindow *win,
DrawFocusRect( win, dc, rect, flags );
}
+void wxRendererMSWBase::DrawChoice(wxWindow* win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags)
+{
+ DrawComboBox(win, dc, rect, flags);
+}
+
+void wxRendererMSWBase::DrawComboBox(wxWindow* win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags)
+{
+ // Draw the main part of the control same as TextCtrl
+ DrawTextCtrl(win, dc, rect, flags);
+
+ // Draw the button inside the border, on the right side
+ wxRect br(rect);
+ br.height -= 2;
+ br.x += br.width - br.height - 1;
+ br.width = br.height;
+ br.y += 1;
+
+ DrawComboBoxDropButton(win, dc, br, flags);
+}
// ============================================================================
// wxRendererNative and wxRendererMSW implementation
const wxRect& rect,
int flags)
{
+ wxCHECK_RET( dc.GetImpl(), wxT("Invalid wxDC") );
+
+ wxRect adjustedRect = dc.GetImpl()->MSWApplyGDIPlusTransform(rect);
+
RECT r;
- wxCopyRectToRECT(rect, r);
+ wxCopyRectToRECT(adjustedRect, r);
int style = DFCS_SCROLLCOMBOBOX;
if ( flags & wxCONTROL_DISABLED )
if ( flags & wxCONTROL_PRESSED )
style |= DFCS_PUSHED | DFCS_FLAT;
- ::DrawFrameControl(GraphicsHDC(&dc), &r, DFC_SCROLL, style);
+ ::DrawFrameControl(GetHdcOf(dc.GetTempHDC()), &r, DFC_SCROLL, style);
}
void
-wxRendererMSW::DoDrawButton(UINT kind,
- wxWindow * WXUNUSED(win),
- wxDC& dc,
- const wxRect& rect,
- int flags)
+wxRendererMSW::DoDrawFrameControl(UINT type,
+ UINT kind,
+ wxWindow * WXUNUSED(win),
+ wxDC& dc,
+ const wxRect& rect,
+ int flags)
{
+ wxCHECK_RET( dc.GetImpl(), wxT("Invalid wxDC") );
+
+ wxRect adjustedRect = dc.GetImpl()->MSWApplyGDIPlusTransform(rect);
+
RECT r;
- wxCopyRectToRECT(rect, r);
+ wxCopyRectToRECT(adjustedRect, r);
int style = kind;
if ( flags & wxCONTROL_CHECKED )
style |= DFCS_PUSHED;
if ( flags & wxCONTROL_CURRENT )
style |= DFCS_HOT;
+ if ( flags & wxCONTROL_UNDETERMINED )
+ // Using DFCS_BUTTON3STATE here doesn't work (as might be expected),
+ // use the following two styles to get the same look of a check box
+ // in the undetermined state.
+ style |= DFCS_INACTIVE | DFCS_CHECKED;
- ::DrawFrameControl(GraphicsHDC(&dc), &r, DFC_BUTTON, style);
+ ::DrawFrameControl(GetHdcOf(dc.GetTempHDC()), &r, type, style);
}
void
DoDrawButton(DFCS_BUTTONPUSH, win, dc, rect, flags);
}
+void
+wxRendererMSW::DrawTitleBarBitmap(wxWindow *win,
+ wxDC& dc,
+ const wxRect& rect,
+ wxTitleBarButton button,
+ int flags)
+{
+ UINT kind;
+ switch ( button )
+ {
+ case wxTITLEBAR_BUTTON_CLOSE:
+ kind = DFCS_CAPTIONCLOSE;
+ break;
+
+ case wxTITLEBAR_BUTTON_MAXIMIZE:
+ kind = DFCS_CAPTIONMAX;
+ break;
+
+ case wxTITLEBAR_BUTTON_ICONIZE:
+ kind = DFCS_CAPTIONMIN;
+ break;
+
+ case wxTITLEBAR_BUTTON_RESTORE:
+ kind = DFCS_CAPTIONRESTORE;
+ break;
+
+ case wxTITLEBAR_BUTTON_HELP:
+ kind = DFCS_CAPTIONHELP;
+ break;
+
+ default:
+ wxFAIL_MSG( "unsupported title bar button" );
+ return;
+ }
+
+ DoDrawFrameControl(DFC_CAPTION, kind, win, dc, rect, flags);
+}
+
wxSize wxRendererMSW::GetCheckBoxSize(wxWindow * WXUNUSED(win))
{
return wxSize(::GetSystemMetrics(SM_CXMENUCHECK),
return Header_Layout(hwndHeader, &hdl) ? wp.cy : DEFAULT_HEIGHT;
}
+int wxRendererMSW::GetHeaderButtonMargin(wxWindow *WXUNUSED(win))
+{
+ return 10;
+}
+
// Uses the theme to draw the border and fill for something like a wxTextCtrl
-void wxRendererMSW::DrawTextCtrl(wxWindow* win, wxDC& dc, const wxRect& rect, int flags)
+void wxRendererMSW::DrawTextCtrl(wxWindow* WXUNUSED(win),
+ wxDC& dc,
+ const wxRect& rect,
+ int WXUNUSED(flags))
{
wxColour fill;
wxColour bdr;
- COLORREF cref;
-
-#if wxUSE_UXTHEME
- wxUxThemeHandle hTheme(win, L"EDIT");
- if (hTheme)
- {
- wxUxThemeEngine::Get()->GetThemeColor(hTheme, EP_EDITTEXT,
- ETS_NORMAL, TMT_FILLCOLOR, &cref);
- fill = wxRGBToColour(cref);
-
- int etsState;
- if ( flags & wxCONTROL_DISABLED )
- etsState = ETS_DISABLED;
- else
- etsState = ETS_NORMAL;
-
- wxUxThemeEngine::Get()->GetThemeColor(hTheme, EP_EDITTEXT,
- etsState, TMT_BORDERCOLOR, &cref);
- bdr = wxRGBToColour(cref);
- }
- else
-#endif
{
fill = wxSystemSettings::GetColour(wxSYS_COLOUR_WINDOW);
bdr = *wxBLACK;
}
- dc.SetPen( bdr );
- dc.SetBrush( fill );
+ dc.SetPen(bdr);
+ dc.SetBrush(fill);
dc.DrawRectangle(rect);
}
-// Draw the equivalent of a wxComboBox
-void wxRendererMSW::DrawComboBox(wxWindow* win, wxDC& dc, const wxRect& rect, int flags)
-{
- // Draw the main part of the control same as TextCtrl
- DrawTextCtrl(win, dc, rect, flags);
-
- // Draw the button inside the border, on the right side
- wxRect br(rect);
- br.height -= 2;
- br.x += br.width - br.height - 1;
- br.width = br.height;
- br.y += 1;
-
- DrawComboBoxDropButton(win, dc, br, flags);
-}
-
-
-void wxRendererMSW::DrawChoice(wxWindow* win, wxDC& dc,
- const wxRect& rect, int flags)
-{
- DrawComboBox(win, dc, rect, flags);
-}
-
// ============================================================================
// wxRendererXP implementation
// ============================================================================
return;
}
+ wxCHECK_RET( dc.GetImpl(), wxT("Invalid wxDC") );
+
+ wxRect adjustedRect = dc.GetImpl()->MSWApplyGDIPlusTransform(rect);
+
RECT r;
- wxCopyRectToRECT(rect, r);
+ wxCopyRectToRECT(adjustedRect, r);
int state;
if ( flags & wxCONTROL_PRESSED )
wxUxThemeEngine::Get()->DrawThemeBackground
(
hTheme,
- GetHdcOf(*((wxMSWDCImpl*)dc.GetImpl())),
+ GetHdcOf(dc.GetTempHDC()),
CP_DROPDOWNBUTTON,
state,
&r,
return m_rendererNative.DrawHeaderButton(win, dc, rect, flags, sortArrow, params);
}
+ wxCHECK_MSG( dc.GetImpl(), -1, wxT("Invalid wxDC") );
+
+ wxRect adjustedRect = dc.GetImpl()->MSWApplyGDIPlusTransform(rect);
+
RECT r;
- wxCopyRectToRECT(rect, r);
+ wxCopyRectToRECT(adjustedRect, r);
int state;
if ( flags & wxCONTROL_PRESSED )
wxUxThemeEngine::Get()->DrawThemeBackground
(
hTheme,
- GetHdcOf(*((wxMSWDCImpl*)dc.GetImpl())),
+ GetHdcOf(dc.GetTempHDC()),
HP_HEADERITEM,
state,
&r,
return;
}
+ wxCHECK_RET( dc.GetImpl(), wxT("Invalid wxDC") );
+
+ wxRect adjustedRect = dc.GetImpl()->MSWApplyGDIPlusTransform(rect);
+
RECT r;
- wxCopyRectToRECT(rect, r);
+ wxCopyRectToRECT(adjustedRect, r);
int state = flags & wxCONTROL_EXPANDED ? GLPS_OPENED : GLPS_CLOSED;
wxUxThemeEngine::Get()->DrawThemeBackground
(
hTheme,
- GetHdcOf(*((wxMSWDCImpl*)dc.GetImpl())),
+ GetHdcOf(dc.GetTempHDC()),
TVP_GLYPH,
state,
&r,
if ( !hTheme )
return false;
- RECT r;
- wxCopyRectToRECT(rect, r);
+ DoDrawButtonLike(hTheme, kind, dc, rect, flags);
- // determine the base state depending on the button kind
- int state;
- switch ( kind )
- {
- case BP_PUSHBUTTON:
- state = PBS_NORMAL;
- break;
+ return true;
+}
- case BP_RADIOBUTTON:
- state = RBS_UNCHECKEDNORMAL;
- break;
+void
+wxRendererXP::DoDrawButtonLike(HTHEME htheme,
+ int part,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags)
+{
+ wxCHECK_RET( dc.GetImpl(), wxT("Invalid wxDC") );
- case BP_CHECKBOX:
- state = CBS_UNCHECKEDNORMAL;
- break;
+ wxRect adjustedRect = dc.GetImpl()->MSWApplyGDIPlusTransform(rect);
- default:
- wxFAIL_MSG( "unknown button kind" );
- return false;
- }
+ RECT r;
+ wxCopyRectToRECT(adjustedRect, r);
+
+ // the base state is always 1, whether it is PBS_NORMAL,
+ // {CBS,RBS}_UNCHECKEDNORMAL or CBS_NORMAL
+ int state = 1;
// XBS_XXX is followed by XBX_XXXHOT, then XBS_XXXPRESSED and DISABLED
enum
else if ( flags & wxCONTROL_CURRENT )
state += HOT_OFFSET;
// wxCONTROL_ISDEFAULT flag is only valid for push buttons
- else if ( kind == BP_PUSHBUTTON && (flags & wxCONTROL_ISDEFAULT) )
+ else if ( part == BP_PUSHBUTTON && (flags & wxCONTROL_ISDEFAULT) )
state = PBS_DEFAULTED;
wxUxThemeEngine::Get()->DrawThemeBackground
(
- hTheme,
- GetHdcOf(*((wxMSWDCImpl*)dc.GetImpl())),
- kind,
+ htheme,
+ GetHdcOf(dc.GetTempHDC()),
+ part,
state,
&r,
NULL
);
+}
- return true;
+void
+wxRendererXP::DrawTitleBarBitmap(wxWindow *win,
+ wxDC& dc,
+ const wxRect& rect,
+ wxTitleBarButton button,
+ int flags)
+{
+ wxUxThemeHandle hTheme(win, L"WINDOW");
+ if ( !hTheme )
+ {
+ m_rendererNative.DrawTitleBarBitmap(win, dc, rect, button, flags);
+ return;
+ }
+
+ int part;
+ switch ( button )
+ {
+ case wxTITLEBAR_BUTTON_CLOSE:
+ part = WP_CLOSEBUTTON;
+ break;
+
+ case wxTITLEBAR_BUTTON_MAXIMIZE:
+ part = WP_MAXBUTTON;
+ break;
+
+ case wxTITLEBAR_BUTTON_ICONIZE:
+ part = WP_MINBUTTON;
+ break;
+
+ case wxTITLEBAR_BUTTON_RESTORE:
+ part = WP_RESTOREBUTTON;
+ break;
+
+ case wxTITLEBAR_BUTTON_HELP:
+ part = WP_HELPBUTTON;
+ break;
+
+ default:
+ wxFAIL_MSG( "unsupported title bar button" );
+ return;
+ }
+
+ DoDrawButtonLike(hTheme, part, dc, rect, flags);
+}
+
+// Uses the theme to draw the border and fill for something like a wxTextCtrl
+void wxRendererXP::DrawTextCtrl(wxWindow* win,
+ wxDC& dc,
+ const wxRect& rect,
+ int flags)
+{
+ wxUxThemeHandle hTheme(win, L"EDIT");
+ if ( !hTheme )
+ {
+ m_rendererNative.DrawTextCtrl(win,dc,rect,flags);
+ return;
+ }
+
+ wxColour fill;
+ wxColour bdr;
+ COLORREF cref;
+
+ wxUxThemeEngine::Get()->GetThemeColor(hTheme, EP_EDITTEXT,
+ ETS_NORMAL, TMT_FILLCOLOR, &cref);
+ fill = wxRGBToColour(cref);
+
+ int etsState;
+ if ( flags & wxCONTROL_DISABLED )
+ etsState = ETS_DISABLED;
+ else
+ etsState = ETS_NORMAL;
+
+ wxUxThemeEngine::Get()->GetThemeColor(hTheme, EP_EDITTEXT,
+ etsState, TMT_BORDERCOLOR, &cref);
+ bdr = wxRGBToColour(cref);
+
+ dc.SetPen( bdr );
+ dc.SetBrush( fill );
+ dc.DrawRectangle(rect);
}
// ----------------------------------------------------------------------------