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/toplevel.h"
20 #include "wx/fontutil.h"
21 #include "wx/fontenum.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_LISTBOXTEXT
:
189 color
= wxColor(ListStyle()->text
[GTK_STATE_NORMAL
]);
192 case wxSYS_COLOUR_MENUTEXT
:
193 case wxSYS_COLOUR_WINDOWTEXT
:
194 case wxSYS_COLOUR_CAPTIONTEXT
:
195 case wxSYS_COLOUR_INACTIVECAPTIONTEXT
:
196 case wxSYS_COLOUR_BTNTEXT
:
197 color
= wxColor(ButtonStyle()->fg
[GTK_STATE_NORMAL
]);
200 case wxSYS_COLOUR_INFOBK
:
201 color
= wxColor(ToolTipStyle()->bg
[GTK_STATE_NORMAL
]);
204 case wxSYS_COLOUR_INFOTEXT
:
205 color
= wxColor(ToolTipStyle()->fg
[GTK_STATE_NORMAL
]);
208 case wxSYS_COLOUR_HIGHLIGHTTEXT
:
209 color
= wxColor(ButtonStyle()->fg
[GTK_STATE_SELECTED
]);
212 case wxSYS_COLOUR_APPWORKSPACE
:
213 color
= *wxWHITE
; // ?
216 case wxSYS_COLOUR_ACTIVECAPTION
:
217 case wxSYS_COLOUR_MENUHILIGHT
:
218 color
= wxColor(MenuItemStyle()->bg
[GTK_STATE_SELECTED
]);
221 case wxSYS_COLOUR_HOTLIGHT
:
222 case wxSYS_COLOUR_GRADIENTACTIVECAPTION
:
223 case wxSYS_COLOUR_GRADIENTINACTIVECAPTION
:
228 case wxSYS_COLOUR_MAX
:
230 wxFAIL_MSG( wxT("unknown system colour index") );
235 wxASSERT(color
.IsOk());
239 wxFont
wxSystemSettingsNative::GetFont( wxSystemFont index
)
244 case wxSYS_OEM_FIXED_FONT
:
245 case wxSYS_ANSI_FIXED_FONT
:
246 case wxSYS_SYSTEM_FIXED_FONT
:
247 font
= *wxNORMAL_FONT
;
250 case wxSYS_ANSI_VAR_FONT
:
251 case wxSYS_SYSTEM_FONT
:
252 case wxSYS_DEVICE_DEFAULT_FONT
:
253 case wxSYS_DEFAULT_GUI_FONT
:
254 if (!gs_fontSystem
.Ok())
256 wxNativeFontInfo info
;
257 info
.description
= ButtonStyle()->font_desc
;
258 gs_fontSystem
= wxFont(info
);
261 // (try to) heal the default font (on some common systems e.g. Ubuntu
262 // it's "Sans Serif" but the real font is called "Sans"):
263 if (!wxFontEnumerator::IsValidFacename(gs_fontSystem
.GetFaceName()) &&
264 gs_fontSystem
.GetFaceName() == "Sans Serif")
265 gs_fontSystem
.SetFaceName("Sans");
266 #endif // wxUSE_FONTENUM
268 info
.description
= NULL
;
270 font
= gs_fontSystem
;
277 wxASSERT( font
.IsOk() );
282 // helper: return the GtkSettings either for the screen the current window is
283 // on or for the default screen if window is NULL
284 static GtkSettings
*GetSettingsForWindowScreen(GdkWindow
*window
)
286 return window
? gtk_settings_get_for_screen(gdk_drawable_get_screen(window
))
287 : gtk_settings_get_default();
290 static int GetBorderWidth(wxSystemMetric index
, wxWindow
* win
)
294 wxPizza
* pizza
= WX_PIZZA(win
->m_wxwindow
);
296 pizza
->get_border_widths(x
, y
);
301 case wxSYS_FRAMESIZE_X
:
310 int wxSystemSettingsNative::GetMetric( wxSystemMetric index
, wxWindow
* win
)
312 GdkWindow
*window
= NULL
;
313 if(win
&& GTK_WIDGET_REALIZED(win
->GetHandle()))
314 window
= win
->GetHandle()->window
;
322 case wxSYS_FRAMESIZE_X
:
323 case wxSYS_FRAMESIZE_Y
:
326 wxTopLevelWindow
*tlw
= wxDynamicCast(win
, wxTopLevelWindow
);
328 return GetBorderWidth(index
, win
);
331 // Get the frame extents from the windowmanager.
332 // In most cases the top extent is the titlebar, so we use the bottom extent
335 if (wxGetFrameExtents(window
, NULL
, &right
, NULL
, &bottom
))
341 case wxSYS_FRAMESIZE_X
:
342 return right
; // width of right extent
344 return bottom
; // height of bottom extent
350 return -1; // no window specified
354 return gdk_display_get_default_cursor_size(
355 window
? gdk_drawable_get_display(window
)
356 : gdk_display_get_default());
360 gint dclick_distance
;
361 g_object_get(GetSettingsForWindowScreen(window
),
362 "gtk-double-click-distance", &dclick_distance
, NULL
);
364 return dclick_distance
* 2;
366 case wxSYS_DCLICK_MSEC
:
368 g_object_get(GetSettingsForWindowScreen(window
),
369 "gtk-double-click-time", &dclick
, NULL
);
375 g_object_get(GetSettingsForWindowScreen(window
),
376 "gtk-dnd-drag-threshold", &drag_threshold
, NULL
);
378 // The correct thing here would be to double the value
379 // since that is what the API wants. But the values
380 // are much bigger under GNOME than under Windows and
381 // just seem to much in many cases to be useful.
382 // drag_threshold *= 2;
384 return drag_threshold
;
392 return gdk_screen_get_width(gdk_drawable_get_screen(window
));
394 return gdk_screen_width();
398 return gdk_screen_get_height(gdk_drawable_get_screen(window
));
400 return gdk_screen_height();
402 case wxSYS_HSCROLL_Y
:
403 case wxSYS_VSCROLL_X
:
406 case wxSYS_CAPTION_Y
:
408 // No realized window specified, and no implementation for that case yet.
411 wxASSERT_MSG( wxDynamicCast(win
, wxTopLevelWindow
),
412 wxT("Asking for caption height of a non toplevel window") );
414 // Get the height of the top windowmanager border.
415 // This is the titlebar in most cases. The titlebar might be elsewhere, and
416 // we could check which is the thickest wm border to decide on which side the
417 // titlebar is, but this might lead to interesting behaviours in used code.
418 // Reconsider when we have a way to report to the user on which side it is.
421 if (wxGetFrameExtents(window
, NULL
, NULL
, &top
, NULL
))
423 return top
; // top frame extent
427 // Try a default approach without a window pointer, if possible
432 case wxSYS_PENWINDOWS_PRESENT
:
433 // No MS Windows for Pen computing extension available in X11 based gtk+.
437 return -1; // metric is unknown
441 bool wxSystemSettingsNative::HasFeature(wxSystemFeature index
)
445 case wxSYS_CAN_ICONIZE_FRAME
:
448 case wxSYS_CAN_DRAW_FRAME_DECORATIONS
: