1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/settings.cpp
4 // Author: Robert Roebling
5 // Modified by: Mart Raudsepp (GetMetric)
7 // Copyright: (c) 1998 Robert Roebling
8 // Licence: wxWindows licence
9 /////////////////////////////////////////////////////////////////////////////
11 // For compilers that support precompilation, includes "wx.h".
12 #include "wx/wxprec.h"
14 #include "wx/settings.h"
17 #include "wx/cmndata.h"
18 #include "wx/toplevel.h"
21 #include "wx/fontutil.h"
24 #include "wx/gtk/private/win_gtk.h"
26 bool wxGetFrameExtents(GdkWindow
* window
, int* left
, int* right
, int* top
, int* bottom
);
28 // ----------------------------------------------------------------------------
29 // wxSystemSettings implementation
30 // ----------------------------------------------------------------------------
32 static wxFont gs_fontSystem
;
34 static GtkContainer
* ContainerWidget()
36 static GtkContainer
* s_widget
;
39 s_widget
= GTK_CONTAINER(gtk_fixed_new());
40 GtkWidget
* window
= gtk_window_new(GTK_WINDOW_TOPLEVEL
);
41 gtk_container_add(GTK_CONTAINER(window
), GTK_WIDGET(s_widget
));
47 static void style_set(GtkWidget
*, GtkStyle
*, void*)
49 gs_fontSystem
= wxNullFont
;
53 static const GtkStyle
* ButtonStyle()
55 static GtkWidget
* s_widget
;
58 s_widget
= gtk_button_new();
59 gtk_container_add(ContainerWidget(), s_widget
);
60 gtk_widget_ensure_style(s_widget
);
61 g_signal_connect(s_widget
, "style_set", G_CALLBACK(style_set
), NULL
);
63 return s_widget
->style
;
66 static const GtkStyle
* ListStyle()
68 static GtkWidget
* s_widget
;
71 s_widget
= gtk_tree_view_new_with_model(
72 GTK_TREE_MODEL(gtk_list_store_new(1, G_TYPE_INT
)));
73 gtk_container_add(ContainerWidget(), s_widget
);
74 gtk_widget_ensure_style(s_widget
);
76 return s_widget
->style
;
79 static const GtkStyle
* TextCtrlStyle()
81 static GtkWidget
* s_widget
;
84 s_widget
= gtk_text_view_new();
85 gtk_container_add(ContainerWidget(), s_widget
);
86 gtk_widget_ensure_style(s_widget
);
88 return s_widget
->style
;
91 static const GtkStyle
* MenuItemStyle()
93 static GtkWidget
* s_widget
;
96 s_widget
= gtk_menu_item_new();
97 gtk_container_add(ContainerWidget(), s_widget
);
98 gtk_widget_ensure_style(s_widget
);
100 return s_widget
->style
;
103 static const GtkStyle
* MenuBarStyle()
105 static GtkWidget
* s_widget
;
106 if (s_widget
== NULL
)
108 s_widget
= gtk_menu_bar_new();
109 gtk_container_add(ContainerWidget(), s_widget
);
110 gtk_widget_ensure_style(s_widget
);
112 return s_widget
->style
;
115 static const GtkStyle
* ToolTipStyle()
117 static GtkWidget
* s_widget
;
118 if (s_widget
== NULL
)
120 s_widget
= gtk_window_new(GTK_WINDOW_POPUP
);
121 const char* name
= "gtk-tooltip";
122 if (gtk_check_version(2, 11, 0))
123 name
= "gtk-tooltips";
124 gtk_widget_set_name(s_widget
, name
);
125 gtk_widget_ensure_style(s_widget
);
127 return s_widget
->style
;
130 wxColour
wxSystemSettingsNative::GetColour( wxSystemColour index
)
135 case wxSYS_COLOUR_SCROLLBAR
:
136 case wxSYS_COLOUR_BACKGROUND
:
137 //case wxSYS_COLOUR_DESKTOP:
138 case wxSYS_COLOUR_INACTIVECAPTION
:
139 case wxSYS_COLOUR_MENU
:
140 case wxSYS_COLOUR_WINDOWFRAME
:
141 case wxSYS_COLOUR_ACTIVEBORDER
:
142 case wxSYS_COLOUR_INACTIVEBORDER
:
143 case wxSYS_COLOUR_BTNFACE
:
144 //case wxSYS_COLOUR_3DFACE:
145 case wxSYS_COLOUR_3DLIGHT
:
146 color
= wxColor(ButtonStyle()->bg
[GTK_STATE_NORMAL
]);
149 case wxSYS_COLOUR_WINDOW
:
150 color
= wxColor(TextCtrlStyle()->base
[GTK_STATE_NORMAL
]);
153 case wxSYS_COLOUR_MENUBAR
:
154 color
= wxColor(MenuBarStyle()->bg
[GTK_STATE_NORMAL
]);
157 case wxSYS_COLOUR_3DDKSHADOW
:
161 case wxSYS_COLOUR_GRAYTEXT
:
162 case wxSYS_COLOUR_BTNSHADOW
:
163 //case wxSYS_COLOUR_3DSHADOW:
165 wxColour
faceColour(GetColour(wxSYS_COLOUR_3DFACE
));
167 wxColour((unsigned char) (faceColour
.Red() * 2 / 3),
168 (unsigned char) (faceColour
.Green() * 2 / 3),
169 (unsigned char) (faceColour
.Blue() * 2 / 3));
173 case wxSYS_COLOUR_BTNHIGHLIGHT
:
174 //case wxSYS_COLOUR_BTNHILIGHT:
175 //case wxSYS_COLOUR_3DHIGHLIGHT:
176 //case wxSYS_COLOUR_3DHILIGHT:
180 case wxSYS_COLOUR_HIGHLIGHT
:
181 color
= wxColor(ButtonStyle()->bg
[GTK_STATE_SELECTED
]);
184 case wxSYS_COLOUR_LISTBOX
:
185 color
= wxColor(ListStyle()->base
[GTK_STATE_NORMAL
]);
188 case wxSYS_COLOUR_MENUTEXT
:
189 case wxSYS_COLOUR_WINDOWTEXT
:
190 case wxSYS_COLOUR_CAPTIONTEXT
:
191 case wxSYS_COLOUR_INACTIVECAPTIONTEXT
:
192 case wxSYS_COLOUR_BTNTEXT
:
193 color
= wxColor(ButtonStyle()->fg
[GTK_STATE_NORMAL
]);
196 case wxSYS_COLOUR_INFOBK
:
197 color
= wxColor(ToolTipStyle()->bg
[GTK_STATE_NORMAL
]);
200 case wxSYS_COLOUR_INFOTEXT
:
201 color
= wxColor(ToolTipStyle()->fg
[GTK_STATE_NORMAL
]);
204 case wxSYS_COLOUR_HIGHLIGHTTEXT
:
205 color
= wxColor(ButtonStyle()->fg
[GTK_STATE_SELECTED
]);
208 case wxSYS_COLOUR_APPWORKSPACE
:
209 color
= *wxWHITE
; // ?
212 case wxSYS_COLOUR_ACTIVECAPTION
:
213 case wxSYS_COLOUR_MENUHILIGHT
:
214 color
= wxColor(MenuItemStyle()->bg
[GTK_STATE_SELECTED
]);
217 case wxSYS_COLOUR_HOTLIGHT
:
218 case wxSYS_COLOUR_GRADIENTACTIVECAPTION
:
219 case wxSYS_COLOUR_GRADIENTINACTIVECAPTION
:
224 case wxSYS_COLOUR_MAX
:
226 wxFAIL_MSG( _T("unknown system colour index") );
234 wxFont
wxSystemSettingsNative::GetFont( wxSystemFont index
)
239 case wxSYS_OEM_FIXED_FONT
:
240 case wxSYS_ANSI_FIXED_FONT
:
241 case wxSYS_SYSTEM_FIXED_FONT
:
242 font
= *wxNORMAL_FONT
;
245 case wxSYS_ANSI_VAR_FONT
:
246 case wxSYS_SYSTEM_FONT
:
247 case wxSYS_DEVICE_DEFAULT_FONT
:
248 case wxSYS_DEFAULT_GUI_FONT
:
249 if (!gs_fontSystem
.Ok())
251 wxNativeFontInfo info
;
252 info
.description
= ButtonStyle()->font_desc
;
253 gs_fontSystem
= wxFont(info
);
254 info
.description
= NULL
;
256 font
= gs_fontSystem
;
265 // helper: return the GtkSettings either for the screen the current window is
266 // on or for the default screen if window is NULL
267 static GtkSettings
*GetSettingsForWindowScreen(GdkWindow
*window
)
269 return window
? gtk_settings_get_for_screen(gdk_drawable_get_screen(window
))
270 : gtk_settings_get_default();
273 static int GetBorderWidth(wxSystemMetric index
, wxWindow
* win
)
277 wxPizza
* pizza
= WX_PIZZA(win
->m_wxwindow
);
279 pizza
->get_border_widths(x
, y
);
284 case wxSYS_FRAMESIZE_X
:
293 int wxSystemSettingsNative::GetMetric( wxSystemMetric index
, wxWindow
* win
)
295 GdkWindow
*window
= NULL
;
296 if(win
&& GTK_WIDGET_REALIZED(win
->GetHandle()))
297 window
= win
->GetHandle()->window
;
305 case wxSYS_FRAMESIZE_X
:
306 case wxSYS_FRAMESIZE_Y
:
309 wxTopLevelWindow
*tlw
= wxDynamicCast(win
, wxTopLevelWindow
);
311 return GetBorderWidth(index
, win
);
314 // Get the frame extents from the windowmanager.
315 // In most cases the top extent is the titlebar, so we use the bottom extent
318 if (wxGetFrameExtents(window
, NULL
, &right
, NULL
, &bottom
))
324 case wxSYS_FRAMESIZE_X
:
325 return right
; // width of right extent
327 return bottom
; // height of bottom extent
333 return -1; // no window specified
337 return gdk_display_get_default_cursor_size(
338 window
? gdk_drawable_get_display(window
)
339 : gdk_display_get_default());
343 gint dclick_distance
;
344 g_object_get(GetSettingsForWindowScreen(window
),
345 "gtk-double-click-distance", &dclick_distance
, NULL
);
347 return dclick_distance
* 2;
349 case wxSYS_DCLICK_MSEC
:
351 g_object_get(GetSettingsForWindowScreen(window
),
352 "gtk-double-click-time", &dclick
, NULL
);
358 g_object_get(GetSettingsForWindowScreen(window
),
359 "gtk-dnd-drag-threshold", &drag_threshold
, NULL
);
361 // The correct thing here would be to double the value
362 // since that is what the API wants. But the values
363 // are much bigger under GNOME than under Windows and
364 // just seem to much in many cases to be useful.
365 // drag_threshold *= 2;
367 return drag_threshold
;
375 return gdk_screen_get_width(gdk_drawable_get_screen(window
));
377 return gdk_screen_width();
381 return gdk_screen_get_height(gdk_drawable_get_screen(window
));
383 return gdk_screen_height();
385 case wxSYS_HSCROLL_Y
:
386 case wxSYS_VSCROLL_X
:
389 case wxSYS_CAPTION_Y
:
391 // No realized window specified, and no implementation for that case yet.
394 wxASSERT_MSG( wxDynamicCast(win
, wxTopLevelWindow
),
395 wxT("Asking for caption height of a non toplevel window") );
397 // Get the height of the top windowmanager border.
398 // This is the titlebar in most cases. The titlebar might be elsewhere, and
399 // we could check which is the thickest wm border to decide on which side the
400 // titlebar is, but this might lead to interesting behaviours in used code.
401 // Reconsider when we have a way to report to the user on which side it is.
404 if (wxGetFrameExtents(window
, NULL
, NULL
, &top
, NULL
))
406 return top
; // top frame extent
410 // Try a default approach without a window pointer, if possible
415 case wxSYS_PENWINDOWS_PRESENT
:
416 // No MS Windows for Pen computing extension available in X11 based gtk+.
420 return -1; // metric is unknown
424 bool wxSystemSettingsNative::HasFeature(wxSystemFeature index
)
428 case wxSYS_CAN_ICONIZE_FRAME
:
431 case wxSYS_CAN_DRAW_FRAME_DECORATIONS
: