]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/gtk/utilsgtk.cpp
Tinderbox build fix.
[wxWidgets.git] / src / gtk / utilsgtk.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/gtk/utilsgtk.cpp
3// Purpose:
4// Author: Robert Roebling
5// Id: $Id$
6// Copyright: (c) 1998 Robert Roebling
7// Licence: wxWindows licence
8/////////////////////////////////////////////////////////////////////////////
9
10// For compilers that support precompilation, includes "wx.h".
11#include "wx/wxprec.h"
12
13#include "wx/utils.h"
14
15#ifndef WX_PRECOMP
16 #include "wx/string.h"
17 #include "wx/intl.h"
18 #include "wx/log.h"
19#endif
20
21#include "wx/apptrait.h"
22
23#include "wx/process.h"
24
25#include "wx/unix/execute.h"
26
27#include <stdarg.h>
28#include <string.h>
29#include <sys/stat.h>
30#include <sys/types.h>
31#include <sys/wait.h> // for WNOHANG
32#include <unistd.h>
33
34#include "glib.h"
35#include "gdk/gdk.h"
36#include "gtk/gtk.h"
37#include "gdk/gdkx.h"
38
39#ifdef HAVE_X11_XKBLIB_H
40 /* under HP-UX and Solaris 2.6, at least, XKBlib.h defines structures with
41 * field named "explicit" - which is, of course, an error for a C++
42 * compiler. To be on the safe side, just redefine it everywhere. */
43 #define explicit __wx_explicit
44
45 #include "X11/XKBlib.h"
46
47 #undef explicit
48#endif // HAVE_X11_XKBLIB_H
49
50
51#if wxUSE_DETECT_SM
52 #include "X11/Xlib.h"
53 #include "X11/SM/SMlib.h"
54#endif
55
56//-----------------------------------------------------------------------------
57// data
58//-----------------------------------------------------------------------------
59
60extern GtkWidget *wxGetRootWindow();
61
62//----------------------------------------------------------------------------
63// misc.
64//----------------------------------------------------------------------------
65#ifndef __EMX__
66// on OS/2, we use the wxBell from wxBase library
67
68void wxBell()
69{
70 gdk_beep();
71}
72#endif
73
74/* Don't synthesize KeyUp events holding down a key and producing
75 KeyDown events with autorepeat. */
76#ifdef HAVE_X11_XKBLIB_H
77bool wxSetDetectableAutoRepeat( bool flag )
78{
79 Bool result;
80 XkbSetDetectableAutoRepeat( GDK_DISPLAY(), flag, &result );
81 return result; /* true if keyboard hardware supports this mode */
82}
83#else
84bool wxSetDetectableAutoRepeat( bool WXUNUSED(flag) )
85{
86 return false;
87}
88#endif
89
90// Escapes string so that it is valid Pango markup XML string:
91wxString wxEscapeStringForPangoMarkup(const wxString& str)
92{
93 size_t len = str.length();
94 wxString out;
95 out.Alloc(len);
96 for (size_t i = 0; i < len; i++)
97 {
98 wxChar c = str[i];
99 switch (c)
100 {
101 case _T('&'):
102 out << _T("&amp;");
103 break;
104 case _T('<'):
105 out << _T("&lt;");
106 break;
107 case _T('>'):
108 out << _T("&gt;");
109 break;
110 case _T('\''):
111 out << _T("&apos;");
112 break;
113 case _T('"'):
114 out << _T("&quot;");
115 break;
116 default:
117 out << c;
118 break;
119 }
120 }
121 return out;
122}
123
124
125// ----------------------------------------------------------------------------
126// display characterstics
127// ----------------------------------------------------------------------------
128
129void *wxGetDisplay()
130{
131 return GDK_DISPLAY();
132}
133
134void wxDisplaySize( int *width, int *height )
135{
136 if (width) *width = gdk_screen_width();
137 if (height) *height = gdk_screen_height();
138}
139
140void wxDisplaySizeMM( int *width, int *height )
141{
142 if (width) *width = gdk_screen_width_mm();
143 if (height) *height = gdk_screen_height_mm();
144}
145
146void wxClientDisplayRect(int *x, int *y, int *width, int *height)
147{
148 // This is supposed to return desktop dimensions minus any window
149 // manager panels, menus, taskbars, etc. If there is a way to do that
150 // for this platform please fix this function, otherwise it defaults
151 // to the entire desktop.
152 if (x) *x = 0;
153 if (y) *y = 0;
154 wxDisplaySize(width, height);
155}
156
157void wxGetMousePosition( int* x, int* y )
158{
159 gdk_window_get_pointer( (GdkWindow*) NULL, x, y, (GdkModifierType*) NULL );
160}
161
162bool wxColourDisplay()
163{
164 return true;
165}
166
167int wxDisplayDepth()
168{
169 return gdk_drawable_get_visual( wxGetRootWindow()->window )->depth;
170}
171
172wxWindow* wxFindWindowAtPoint(const wxPoint& pt)
173{
174 return wxGenericFindWindowAtPoint(pt);
175}
176
177#if !wxUSE_UNICODE
178
179wxCharBuffer wxConvertToGTK(const wxString& s, wxFontEncoding enc)
180{
181 wxCharBuffer buf;
182 if ( enc == wxFONTENCODING_UTF8 )
183 {
184 // no need for conversion at all, but do check that we have a valid
185 // UTF-8 string because passing invalid UTF-8 to GTK+ is going to
186 // result in a GTK+ error message and, especially, loss of data which
187 // was supposed to be shown in the GUI
188 if ( wxConvUTF8.ToWChar(NULL, 0, s, s.length()) == wxCONV_FAILED )
189 {
190 // warn the programmer that something is probably wrong in his code
191 //
192 // NB: don't include the string in output because chances are that
193 // this invalid UTF-8 string could result in more errors itself
194 // if the application shows logs in the GUI and so we get into
195 // an infinite loop
196 wxLogDebug(_T("Invalid UTF-8 string in wxConvertToGTK()"));
197
198 // but still try to show at least something on the screen
199 wxMBConvUTF8 utf8permissive(wxMBConvUTF8::MAP_INVALID_UTF8_TO_OCTAL);
200 wxWCharBuffer wbuf(utf8permissive.cMB2WC(s));
201 buf = wxConvUTF8.cWC2MB(wbuf);
202 }
203 else // valid UTF-8 string, no need to convert
204 {
205 buf = wxCharBuffer(s);
206 }
207 }
208 else // !UTF-8
209 {
210 wxWCharBuffer wbuf;
211 if ( enc == wxFONTENCODING_SYSTEM || enc == wxFONTENCODING_DEFAULT )
212 {
213 wbuf = wxConvUI->cMB2WC(s);
214 }
215 else // another encoding, use generic conversion class
216 {
217 wbuf = wxCSConv(enc).cMB2WC(s);
218 }
219
220 if ( wbuf )
221 buf = wxConvUTF8.cWC2MB(wbuf);
222 }
223
224 return buf;
225}
226
227#endif // !wxUSE_UNICODE
228
229// ----------------------------------------------------------------------------
230// subprocess routines
231// ----------------------------------------------------------------------------
232
233extern "C" {
234static
235void GTK_EndProcessDetector(gpointer data, gint source,
236 GdkInputCondition WXUNUSED(condition) )
237{
238 wxEndProcessData *proc_data = (wxEndProcessData *)data;
239
240 // has the process really terminated? unfortunately GDK (or GLib) seem to
241 // generate G_IO_HUP notification even when it simply tries to read from a
242 // closed fd and hasn't terminated at all
243 int pid = (proc_data->pid > 0) ? proc_data->pid : -(proc_data->pid);
244 int status = 0;
245 int rc = waitpid(pid, &status, WNOHANG);
246
247 if ( rc == 0 )
248 {
249 // no, it didn't exit yet, continue waiting
250 return;
251 }
252
253 // set exit code to -1 if something bad happened
254 proc_data->exitcode = rc != -1 && WIFEXITED(status) ? WEXITSTATUS(status)
255 : -1;
256
257 // child exited, end waiting
258 close(source);
259
260 // don't call us again!
261 gdk_input_remove(proc_data->tag);
262
263 wxHandleProcessTermination(proc_data);
264}
265}
266
267int wxAddProcessCallback(wxEndProcessData *proc_data, int fd)
268{
269 int tag = gdk_input_add(fd,
270 GDK_INPUT_READ,
271 GTK_EndProcessDetector,
272 (gpointer)proc_data);
273
274 return tag;
275}
276
277
278
279// ----------------------------------------------------------------------------
280// wxPlatformInfo-related
281// ----------------------------------------------------------------------------
282
283wxPortId wxGUIAppTraits::GetToolkitVersion(int *verMaj, int *verMin) const
284{
285 if ( verMaj )
286 *verMaj = gtk_major_version;
287 if ( verMin )
288 *verMin = gtk_minor_version;
289
290 return wxPORT_GTK;
291}
292
293#if wxUSE_DETECT_SM
294static wxString GetSM()
295{
296 Display *dpy;
297 SmcConn smc_conn;
298 char *vendor;
299 char *client_id_ret;
300 dpy = XOpenDisplay(NULL);
301
302 smc_conn = SmcOpenConnection(NULL, NULL,
303 999, 999,
304 0 /* mask */, NULL /* callbacks */,
305 NULL, &client_id_ret, 0, NULL);
306
307 if (smc_conn)
308 {
309 vendor = SmcVendor(smc_conn);
310 wxString ret = wxString::FromAscii( vendor );
311 free(vendor);
312
313 SmcCloseConnection(smc_conn, 0, NULL);
314 free(client_id_ret);
315
316 XCloseDisplay(dpy);
317
318 return ret;
319 }
320
321 return wxEmptyString;
322}
323#endif
324
325wxString wxGUIAppTraits::GetDesktopEnvironment() const
326{
327#if wxUSE_DETECT_SM
328 wxString SM = GetSM();
329
330 if (SM == wxT("GnomeSM"))
331 return wxT("GNOME");
332
333 if (SM == wxT("KDE"))
334 return wxT("KDE");
335#endif
336
337 return wxEmptyString;
338}
339
340
341