]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/generic/renderg.cpp
fixed SaveDIB() crash with BW images (patch 1045884)
[wxWidgets.git] / src / generic / renderg.cpp
... / ...
CommitLineData
1///////////////////////////////////////////////////////////////////////////////
2// Name: generic/renderg.cpp
3// Purpose: generic implementation of wxRendererNative (for any platform)
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 license
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#endif //WX_PRECOMP
30
31#include "wx/gdicmn.h"
32#include "wx/dc.h"
33
34#include "wx/settings.h"
35#include "wx/splitter.h"
36#include "wx/dcmirror.h"
37#include "wx/module.h"
38#include "wx/renderer.h"
39
40// ----------------------------------------------------------------------------
41// wxRendererGeneric: our wxRendererNative implementation
42// ----------------------------------------------------------------------------
43
44class WXDLLEXPORT wxRendererGeneric : public wxRendererNative
45{
46public:
47 wxRendererGeneric();
48
49 virtual void DrawHeaderButton(wxWindow *win,
50 wxDC& dc,
51 const wxRect& rect,
52 int flags = 0);
53
54 virtual void DrawTreeItemButton(wxWindow *win,
55 wxDC& dc,
56 const wxRect& rect,
57 int flags = 0);
58
59 virtual void DrawSplitterBorder(wxWindow *win,
60 wxDC& dc,
61 const wxRect& rect,
62 int flags = 0);
63
64 virtual void DrawSplitterSash(wxWindow *win,
65 wxDC& dc,
66 const wxSize& size,
67 wxCoord position,
68 wxOrientation orient,
69 int flags = 0);
70
71
72 virtual wxSplitterRenderParams GetSplitterParams(const wxWindow *win);
73
74 virtual wxRendererVersion GetVersion() const
75 {
76 return wxRendererVersion(wxRendererVersion::Current_Version,
77 wxRendererVersion::Current_Age);
78 }
79
80
81 // Cleanup by deleting standard renderer
82 static void Cleanup();
83
84 // Get the generic object
85 static wxRendererGeneric* DoGetGeneric();
86
87protected:
88 // draw the rectange using the first pen for the left and top sides and
89 // the second one for the bottom and right ones
90 void DrawShadedRect(wxDC& dc, wxRect *rect,
91 const wxPen& pen1, const wxPen& pen2);
92
93 // the standard pens
94 wxPen m_penBlack,
95 m_penDarkGrey,
96 m_penLightGrey,
97 m_penHighlight;
98
99 static wxRendererGeneric* sm_rendererGeneric;
100};
101
102// ============================================================================
103// wxRendererGeneric implementation
104// ============================================================================
105
106// Get the generic object
107wxRendererGeneric* wxRendererGeneric::DoGetGeneric()
108{
109 if (!sm_rendererGeneric)
110 sm_rendererGeneric = new wxRendererGeneric;
111 return sm_rendererGeneric;
112}
113
114// ----------------------------------------------------------------------------
115// wxRendererGeneric creation
116// ----------------------------------------------------------------------------
117
118/* static */
119wxRendererNative& wxRendererNative::GetGeneric()
120{
121 return * wxRendererGeneric::DoGetGeneric();
122}
123
124void wxRendererGeneric::Cleanup()
125{
126 if (sm_rendererGeneric)
127 delete sm_rendererGeneric;
128
129 sm_rendererGeneric = NULL;
130}
131
132wxRendererGeneric* wxRendererGeneric::sm_rendererGeneric = NULL;
133
134wxRendererGeneric::wxRendererGeneric()
135 : m_penBlack(wxSystemSettings::GetColour(wxSYS_COLOUR_3DDKSHADOW)),
136 m_penDarkGrey(wxSystemSettings::GetColour(wxSYS_COLOUR_3DSHADOW)),
137 m_penLightGrey(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE)),
138 m_penHighlight(wxSystemSettings::GetColour(wxSYS_COLOUR_3DHIGHLIGHT))
139{
140}
141
142// ----------------------------------------------------------------------------
143// wxRendererGeneric helpers
144// ----------------------------------------------------------------------------
145
146void
147wxRendererGeneric::DrawShadedRect(wxDC& dc,
148 wxRect *rect,
149 const wxPen& pen1,
150 const wxPen& pen2)
151{
152 // draw the rectangle
153 dc.SetPen(pen1);
154 dc.DrawLine(rect->GetLeft(), rect->GetTop(),
155 rect->GetLeft(), rect->GetBottom());
156 dc.DrawLine(rect->GetLeft() + 1, rect->GetTop(),
157 rect->GetRight(), rect->GetTop());
158 dc.SetPen(pen2);
159 dc.DrawLine(rect->GetRight(), rect->GetTop(),
160 rect->GetRight(), rect->GetBottom());
161 dc.DrawLine(rect->GetLeft(), rect->GetBottom(),
162 rect->GetRight() + 1, rect->GetBottom());
163
164 // adjust the rect
165 rect->Inflate(-1);
166}
167
168// ----------------------------------------------------------------------------
169// tree/list ctrl drawing
170// ----------------------------------------------------------------------------
171
172void
173wxRendererGeneric::DrawHeaderButton(wxWindow * WXUNUSED(win),
174 wxDC& dc,
175 const wxRect& rect,
176 int WXUNUSED(flags))
177{
178 const int CORNER = 1;
179
180 const wxCoord x = rect.x,
181 y = rect.y,
182 w = rect.width,
183 h = rect.height;
184
185 dc.SetBrush(*wxTRANSPARENT_BRUSH);
186
187 dc.SetPen(m_penBlack);
188 dc.DrawLine( x+w-CORNER+1, y, x+w, y+h ); // right (outer)
189 dc.DrawRectangle( x, y+h, w+1, 1 ); // bottom (outer)
190
191 dc.SetPen(m_penDarkGrey);
192 dc.DrawLine( x+w-CORNER, y, x+w-1, y+h ); // right (inner)
193 dc.DrawRectangle( x+1, y+h-1, w-2, 1 ); // bottom (inner)
194
195 dc.SetPen(m_penHighlight);
196 dc.DrawRectangle( x, y, w-CORNER+1, 1 ); // top (outer)
197 dc.DrawRectangle( x, y, 1, h ); // left (outer)
198 dc.DrawLine( x, y+h-1, x+1, y+h-1 );
199 dc.DrawLine( x+w-1, y, x+w-1, y+1 );
200}
201
202// draw the plus or minus sign
203void
204wxRendererGeneric::DrawTreeItemButton(wxWindow * WXUNUSED(win),
205 wxDC& dc,
206 const wxRect& rect,
207 int flags)
208{
209 // white background
210 dc.SetPen(*wxGREY_PEN);
211 dc.SetBrush(*wxWHITE_BRUSH);
212 dc.DrawRectangle(rect);
213
214 // black lines
215 const wxCoord xMiddle = rect.x + rect.width/2;
216 const wxCoord yMiddle = rect.y + rect.height/2;
217
218 // half of the length of the horz lines in "-" and "+"
219 const wxCoord halfWidth = rect.width/2 - 2;
220 dc.SetPen(*wxBLACK_PEN);
221 dc.DrawLine(xMiddle - halfWidth, yMiddle,
222 xMiddle + halfWidth + 1, yMiddle);
223
224 if ( !(flags & wxCONTROL_EXPANDED) )
225 {
226 // turn "-" into "+"
227 const wxCoord halfHeight = rect.height/2 - 2;
228 dc.DrawLine(xMiddle, yMiddle - halfHeight,
229 xMiddle, yMiddle + halfHeight + 1);
230 }
231}
232
233// ----------------------------------------------------------------------------
234// sash drawing
235// ----------------------------------------------------------------------------
236
237wxSplitterRenderParams
238wxRendererGeneric::GetSplitterParams(const wxWindow *win)
239{
240 // see below
241 wxCoord sashWidth,
242 border;
243
244 if ( win->HasFlag(wxSP_3DSASH) )
245 sashWidth = 7;
246 else if ( win->HasFlag(wxSP_NOSASH) )
247 sashWidth = 0;
248 else // no 3D effect
249 sashWidth = 3;
250
251 if ( win->HasFlag(wxSP_3DBORDER) )
252 border = 2;
253 else // no 3D effect
254 border = 0;
255
256 return wxSplitterRenderParams(sashWidth, border, false);
257}
258
259void
260wxRendererGeneric::DrawSplitterBorder(wxWindow *win,
261 wxDC& dc,
262 const wxRect& rectOrig,
263 int WXUNUSED(falgs))
264{
265 if ( win->HasFlag(wxSP_3DBORDER) )
266 {
267 wxRect rect = rectOrig;
268 DrawShadedRect(dc, &rect, m_penDarkGrey, m_penHighlight);
269 DrawShadedRect(dc, &rect, m_penBlack, m_penLightGrey);
270 }
271}
272
273void
274wxRendererGeneric::DrawSplitterSash(wxWindow *win,
275 wxDC& dcReal,
276 const wxSize& sizeReal,
277 wxCoord position,
278 wxOrientation orient,
279 int WXUNUSED(flags))
280{
281 // to avoid duplicating the same code for horizontal and vertical sashes,
282 // simply mirror the DC instead if needed (i.e. if horz splitter)
283 wxMirrorDC dc(dcReal, orient != wxVERTICAL);
284 wxSize size = dc.Reflect(sizeReal);
285
286
287 // we draw a Win32-like grey sash with possible 3D border here:
288 //
289 // ---- this is position
290 // /
291 // v
292 // dWGGGDd
293 // GWGGGDB
294 // GWGGGDB where G is light grey (face)
295 // GWGGGDB W white (light)
296 // GWGGGDB D dark grey (shadow)
297 // GWGGGDB B black (dark shadow)
298 // GWGGGDB
299 // GWGGGDB and lower letters are our border (already drawn)
300 // GWGGGDB
301 // wWGGGDd
302 //
303 // only the middle 3 columns are drawn unless wxSP_3D is specified
304
305 const wxCoord h = size.y;
306 wxCoord offset = 0;
307
308 // If we're drawing the border, draw the sash 3d lines shorter
309 if ( win->HasFlag(wxSP_3DBORDER) )
310 {
311 offset = 1;
312 }
313
314 dc.SetPen(*wxTRANSPARENT_PEN);
315 dc.SetBrush(wxBrush(wxSystemSettings::GetColour(wxSYS_COLOUR_3DFACE)));
316
317 if ( win->HasFlag(wxSP_3DSASH) )
318 {
319 // Draw the 3D sash
320 dc.DrawRectangle(position + 2, 0, 3, h);
321
322 dc.SetPen(m_penLightGrey);
323 dc.DrawLine(position, offset, position, h - offset);
324
325 dc.SetPen(m_penHighlight);
326 dc.DrawLine(position + 1, 0, position + 1, h);
327
328 dc.SetPen(m_penDarkGrey);
329 dc.DrawLine(position + 5, 0, position + 5, h);
330
331 dc.SetPen(m_penBlack);
332 dc.DrawLine(position + 6, offset, position + 6, h - offset);
333 }
334 else
335 {
336 // Draw a flat sash
337 dc.DrawRectangle(position, 0, 3, h);
338 }
339}
340
341// A module to allow cleanup of generic renderer.
342class wxGenericRendererModule: public wxModule
343{
344DECLARE_DYNAMIC_CLASS(wxGenericRendererModule)
345public:
346 wxGenericRendererModule() {}
347 bool OnInit() { return true; };
348 void OnExit() { wxRendererGeneric::Cleanup(); };
349};
350
351IMPLEMENT_DYNAMIC_CLASS(wxGenericRendererModule, wxModule)
352