1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/renderer.cpp
3 // Purpose: implementation of wxRendererNative for wxGTK
4 // Author: Vadim Zeitlin
8 // Copyright: (c) 2003 Vadim Zeitlin <vadim@wxwindows.org>
9 // License: wxWindows licence
10 ///////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // for compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
27 #include "wx/renderer.h"
30 #include "wx/window.h"
31 #include "wx/dcclient.h"
32 #include "wx/settings.h"
36 #include "wx/gtk/win_gtk.h"
38 // ----------------------------------------------------------------------------
39 // wxRendererGTK: our wxRendererNative implementation
40 // ----------------------------------------------------------------------------
42 class WXDLLEXPORT wxRendererGTK
: public wxDelegateRendererNative
45 // draw the header control button (used by wxListCtrl)
46 virtual int DrawHeaderButton(wxWindow
*win
,
50 wxHeaderSortIconType sortArrow
= wxHDR_SORT_ICON_NONE
,
51 wxHeaderButtonParams
* params
= NULL
);
53 // draw the expanded/collapsed icon for a tree control item
54 virtual void DrawTreeItemButton(wxWindow
*win
,
59 virtual void DrawSplitterBorder(wxWindow
*win
,
63 virtual void DrawSplitterSash(wxWindow
*win
,
70 virtual void DrawComboBoxDropButton(wxWindow
*win
,
75 virtual void DrawDropArrow(wxWindow
*win
,
80 virtual void DrawCheckBox(wxWindow
*win
,
85 virtual void DrawPushButton(wxWindow
*win
,
90 virtual void DrawItemSelectionRect(wxWindow
*win
,
95 virtual wxSplitterRenderParams
GetSplitterParams(const wxWindow
*win
);
98 // FIXME: shouldn't we destroy these windows somewhere?
100 // used by DrawPushButton and DrawDropArrow
101 static GtkWidget
*GetButtonWidget();
103 // used by DrawTreeItemButton()
104 static GtkWidget
*GetTreeWidget();
106 // used by DrawCheckBox()
107 static GtkWidget
*GetCheckButtonWidget();
109 // Used by DrawHeaderButton
110 static GtkWidget
*GetHeaderButtonWidget();
113 // ============================================================================
115 // ============================================================================
118 wxRendererNative
& wxRendererNative::GetDefault()
120 static wxRendererGTK s_rendererGTK
;
122 return s_rendererGTK
;
125 // ----------------------------------------------------------------------------
127 // ----------------------------------------------------------------------------
130 wxRendererGTK::GetButtonWidget()
132 static GtkWidget
*s_button
= NULL
;
133 static GtkWidget
*s_window
= NULL
;
137 s_window
= gtk_window_new( GTK_WINDOW_POPUP
);
138 gtk_widget_realize( s_window
);
139 s_button
= gtk_button_new();
140 gtk_container_add( GTK_CONTAINER(s_window
), s_button
);
141 gtk_widget_realize( s_button
);
148 wxRendererGTK::GetCheckButtonWidget()
150 static GtkWidget
*s_button
= NULL
;
151 static GtkWidget
*s_window
= NULL
;
155 s_window
= gtk_window_new( GTK_WINDOW_POPUP
);
156 gtk_widget_realize( s_window
);
157 s_button
= gtk_check_button_new();
158 gtk_container_add( GTK_CONTAINER(s_window
), s_button
);
159 gtk_widget_realize( s_button
);
166 wxRendererGTK::GetTreeWidget()
168 static GtkWidget
*s_tree
= NULL
;
169 static GtkWidget
*s_window
= NULL
;
173 s_tree
= gtk_tree_view_new();
174 s_window
= gtk_window_new( GTK_WINDOW_POPUP
);
175 gtk_widget_realize( s_window
);
176 gtk_container_add( GTK_CONTAINER(s_window
), s_tree
);
177 gtk_widget_realize( s_tree
);
184 // This one just makes a button be a child of a tree widget. This is
185 // apparently how gtk themes decide to draw column headers differently than
188 wxRendererGTK::GetHeaderButtonWidget()
190 static GtkWidget
*s_button
= NULL
;
194 GtkWidget
* treewidget
= GetTreeWidget();
195 s_button
= gtk_button_new();
196 gtk_widget_set_parent( s_button
, treewidget
);
197 gtk_widget_realize( s_button
);
203 // ----------------------------------------------------------------------------
204 // list/tree controls drawing
205 // ----------------------------------------------------------------------------
208 wxRendererGTK::DrawHeaderButton(wxWindow
*win
,
212 wxHeaderSortIconType sortArrow
,
213 wxHeaderButtonParams
* params
)
216 GtkWidget
*button
= GetHeaderButtonWidget();
218 GdkWindow
* gdk_window
= dc
.GetGDKWindow();
219 wxASSERT_MSG( gdk_window
,
220 wxT("cannot use wxRendererNative on wxDC of this type") );
223 if (win
->GetLayoutDirection() == wxLayout_RightToLeft
)
230 flags
& wxCONTROL_DISABLED
? GTK_STATE_INSENSITIVE
: GTK_STATE_NORMAL
,
235 dc
.LogicalToDeviceX(rect
.x
) - x_diff
, rect
.y
, rect
.width
, rect
.height
238 return DrawHeaderButtonContents(win
, dc
, rect
, flags
, sortArrow
, params
);
241 // draw a ">" or "v" button
243 wxRendererGTK::DrawTreeItemButton(wxWindow
* win
,
244 wxDC
& dc
, const wxRect
& rect
, int flags
)
246 GtkWidget
*tree
= GetTreeWidget();
248 GdkWindow
* gdk_window
= dc
.GetGDKWindow();
249 wxASSERT_MSG( gdk_window
,
250 wxT("cannot use wxRendererNative on wxDC of this type") );
253 if ( flags
& wxCONTROL_CURRENT
)
254 state
= GTK_STATE_PRELIGHT
;
256 state
= GTK_STATE_NORMAL
;
259 if (win
->GetLayoutDirection() == wxLayout_RightToLeft
)
262 // VZ: I don't know how to get the size of the expander so as to centre it
263 // in the given rectangle, +2/3 below is just what looks good here...
272 dc
.LogicalToDeviceX(rect
.x
) + 6 - x_diff
,
273 dc
.LogicalToDeviceY(rect
.y
) + 3,
274 flags
& wxCONTROL_EXPANDED
? GTK_EXPANDER_EXPANDED
275 : GTK_EXPANDER_COLLAPSED
280 // ----------------------------------------------------------------------------
281 // splitter sash drawing
282 // ----------------------------------------------------------------------------
284 static int GetGtkSplitterFullSize()
286 static GtkWidget
*s_paned
= NULL
;
288 s_paned
= gtk_vpaned_new();
291 gtk_widget_style_get (s_paned
, "handle_size", &handle_size
, NULL
);
296 wxSplitterRenderParams
297 wxRendererGTK::GetSplitterParams(const wxWindow
*WXUNUSED(win
))
299 // we don't draw any border, hence 0 for the second field
300 return wxSplitterRenderParams
302 GetGtkSplitterFullSize(),
304 true // hot sensitive
309 wxRendererGTK::DrawSplitterBorder(wxWindow
* WXUNUSED(win
),
311 const wxRect
& WXUNUSED(rect
),
318 wxRendererGTK::DrawSplitterSash(wxWindow
*win
,
322 wxOrientation orient
,
325 if ( !win
->m_wxwindow
->window
)
327 // window not realized yet
331 GdkWindow
* gdk_window
= dc
.GetGDKWindow();
332 wxASSERT_MSG( gdk_window
,
333 wxT("cannot use wxRendererNative on wxDC of this type") );
335 wxCoord full_size
= GetGtkSplitterFullSize();
337 // are we drawing vertical or horizontal splitter?
338 const bool isVert
= orient
== wxVERTICAL
;
344 int h
= win
->GetClientSize().GetHeight();
348 rect
.width
= full_size
;
353 int w
= win
->GetClientSize().GetWidth();
357 rect
.height
= full_size
;
362 if (win
->GetLayoutDirection() == wxLayout_RightToLeft
)
367 win
->m_wxwindow
->style
,
369 flags
& wxCONTROL_CURRENT
? GTK_STATE_PRELIGHT
: GTK_STATE_NORMAL
,
371 NULL
/* no clipping */,
374 dc
.LogicalToDeviceX(rect
.x
) - x_diff
,
375 dc
.LogicalToDeviceY(rect
.y
),
378 isVert
? GTK_ORIENTATION_VERTICAL
: GTK_ORIENTATION_HORIZONTAL
383 wxRendererGTK::DrawDropArrow(wxWindow
*WXUNUSED(win
),
388 GtkWidget
*button
= GetButtonWidget();
390 // If we give GTK_PIZZA(win->m_wxwindow)->bin_window as
391 // a window for gtk_paint_xxx function, then it won't
392 // work for wxMemoryDC. So that is why we assume wxDC
393 // is wxWindowDC (wxClientDC, wxMemoryDC and wxPaintDC
394 // are derived from it) and use its m_window.
395 GdkWindow
* gdk_window
= dc
.GetGDKWindow();
396 wxASSERT_MSG( gdk_window
,
397 wxT("cannot use wxRendererNative on wxDC of this type") );
399 // draw arrow so that there is even space horizontally
401 int arrowX
= rect
.width
/4 + 1;
402 int arrowWidth
= rect
.width
- (arrowX
*2);
404 // scale arrow's height accoording to the width
405 int arrowHeight
= rect
.width
/3;
406 int arrowY
= (rect
.height
-arrowHeight
)/2 +
407 ((rect
.height
-arrowHeight
) & 1);
411 if ( flags
& wxCONTROL_PRESSED
)
412 state
= GTK_STATE_ACTIVE
;
413 else if ( flags
& wxCONTROL_DISABLED
)
414 state
= GTK_STATE_INSENSITIVE
;
415 else if ( flags
& wxCONTROL_CURRENT
)
416 state
= GTK_STATE_PRELIGHT
;
418 state
= GTK_STATE_NORMAL
;
420 // draw arrow on button
426 flags
& wxCONTROL_PRESSED
? GTK_SHADOW_IN
: GTK_SHADOW_OUT
,
440 wxRendererGTK::DrawComboBoxDropButton(wxWindow
*win
,
445 DrawPushButton(win
,dc
,rect
,flags
);
446 DrawDropArrow(win
,dc
,rect
);
450 wxRendererGTK::DrawCheckBox(wxWindow
*WXUNUSED(win
),
455 GtkWidget
*button
= GetCheckButtonWidget();
457 // for reason why we do this, see DrawDropArrow
458 GdkWindow
* gdk_window
= dc
.GetGDKWindow();
459 wxASSERT_MSG( gdk_window
,
460 wxT("cannot use wxRendererNative on wxDC of this type") );
464 if ( flags
& wxCONTROL_PRESSED
)
465 state
= GTK_STATE_ACTIVE
;
466 else if ( flags
& wxCONTROL_DISABLED
)
467 state
= GTK_STATE_INSENSITIVE
;
468 else if ( flags
& wxCONTROL_CURRENT
)
469 state
= GTK_STATE_PRELIGHT
;
471 state
= GTK_STATE_NORMAL
;
478 flags
& wxCONTROL_CHECKED
? GTK_SHADOW_IN
: GTK_SHADOW_OUT
,
482 dc
.LogicalToDeviceX(rect
.x
)+2,
483 dc
.LogicalToDeviceY(rect
.y
)+3,
489 wxRendererGTK::DrawPushButton(wxWindow
*WXUNUSED(win
),
494 GtkWidget
*button
= GetButtonWidget();
496 // for reason why we do this, see DrawDropArrow
497 GdkWindow
* gdk_window
= dc
.GetGDKWindow();
498 wxASSERT_MSG( gdk_window
,
499 wxT("cannot use wxRendererNative on wxDC of this type") );
504 if ( flags
& wxCONTROL_PRESSED
)
505 state
= GTK_STATE_ACTIVE
;
506 else if ( flags
& wxCONTROL_DISABLED
)
507 state
= GTK_STATE_INSENSITIVE
;
508 else if ( flags
& wxCONTROL_CURRENT
)
509 state
= GTK_STATE_PRELIGHT
;
511 state
= GTK_STATE_NORMAL
;
518 flags
& wxCONTROL_PRESSED
? GTK_SHADOW_IN
: GTK_SHADOW_OUT
,
522 rect
.x
, rect
.y
, rect
.width
, rect
.height
527 wxRendererGTK::DrawItemSelectionRect(wxWindow
*win
,
532 GdkWindow
* gdk_window
= dc
.GetGDKWindow();
533 wxASSERT_MSG( gdk_window
,
534 wxT("cannot use wxRendererNative on wxDC of this type") );
537 if (win
->GetLayoutDirection() == wxLayout_RightToLeft
)
541 if (flags
& wxCONTROL_SELECTED
)
543 // the wxCONTROL_FOCUSED state is deduced
544 // directly from the m_wxwindow by GTK+
545 state
= GTK_STATE_SELECTED
;
547 gtk_paint_flat_box( win
->m_widget
->style
,
554 dc
.LogicalToDeviceX(rect
.x
) - x_diff
,
555 dc
.LogicalToDeviceY(rect
.y
),
559 else // !wxCONTROL_SELECTED
561 state
= GTK_STATE_NORMAL
;
564 if (flags
& wxCONTROL_CURRENT
)
566 gtk_paint_focus( win
->m_widget
->style
,
572 dc
.LogicalToDeviceX(rect
.x
),
573 dc
.LogicalToDeviceY(rect
.y
),