1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/utilsgtk.cpp 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  16     #include "wx/string.h" 
  21 #include "wx/apptrait.h" 
  22 #include "wx/process.h" 
  23 #include "wx/sysopt.h" 
  24 #include "wx/unix/execute.h" 
  26 #include "wx/gtk/private/timer.h" 
  27 #include "wx/evtloop.h" 
  30     #include "wx/gtk/assertdlg_gtk.h" 
  32         #include "wx/stackwalk.h" 
  33     #endif // wxUSE_STACKWALKER 
  39 #include <sys/types.h> 
  40 #include <sys/wait.h>   // for WNOHANG 
  48 #ifdef HAVE_X11_XKBLIB_H 
  49     /* under HP-UX and Solaris 2.6, at least, XKBlib.h defines structures with 
  50      * field named "explicit" - which is, of course, an error for a C++ 
  51      * compiler. To be on the safe side, just redefine it everywhere. */ 
  52     #define explicit __wx_explicit 
  54     #include "X11/XKBlib.h" 
  57 #endif // HAVE_X11_XKBLIB_H 
  62     #include "X11/SM/SMlib.h" 
  65 //----------------------------------------------------------------------------- 
  67 //----------------------------------------------------------------------------- 
  69 extern GtkWidget 
*wxGetRootWindow(); 
  71 //---------------------------------------------------------------------------- 
  73 //---------------------------------------------------------------------------- 
  75 // on OS/2, we use the wxBell from wxBase library 
  83 /* Don't synthesize KeyUp events holding down a key and producing 
  84    KeyDown events with autorepeat. */ 
  85 #ifdef HAVE_X11_XKBLIB_H 
  86 bool wxSetDetectableAutoRepeat( bool flag 
) 
  89     XkbSetDetectableAutoRepeat( GDK_DISPLAY(), flag
, &result 
); 
  90     return result
;       /* true if keyboard hardware supports this mode */ 
  93 bool wxSetDetectableAutoRepeat( bool WXUNUSED(flag
) ) 
  99 // ---------------------------------------------------------------------------- 
 100 // display characterstics 
 101 // ---------------------------------------------------------------------------- 
 105     return GDK_DISPLAY(); 
 108 void wxDisplaySize( int *width
, int *height 
) 
 110     if (width
) *width 
= gdk_screen_width(); 
 111     if (height
) *height 
= gdk_screen_height(); 
 114 void wxDisplaySizeMM( int *width
, int *height 
) 
 116     if (width
) *width 
= gdk_screen_width_mm(); 
 117     if (height
) *height 
= gdk_screen_height_mm(); 
 120 void wxGetMousePosition( int* x
, int* y 
) 
 122     gdk_window_get_pointer( (GdkWindow
*) NULL
, x
, y
, (GdkModifierType
*) NULL 
); 
 125 bool wxColourDisplay() 
 132     return gdk_drawable_get_visual( wxGetRootWindow()->window 
)->depth
; 
 135 wxWindow
* wxFindWindowAtPoint(const wxPoint
& pt
) 
 137     return wxGenericFindWindowAtPoint(pt
); 
 142 WXDLLIMPEXP_CORE wxCharBuffer
 
 143 wxConvertToGTK(const wxString
& s
, wxFontEncoding enc
) 
 146     if ( enc 
== wxFONTENCODING_SYSTEM 
|| enc 
== wxFONTENCODING_DEFAULT 
) 
 148         wbuf 
= wxConvUI
->cMB2WC(s
.c_str()); 
 150     else // another encoding, use generic conversion class 
 152         wbuf 
= wxCSConv(enc
).cMB2WC(s
.c_str()); 
 155     if ( !wbuf 
&& !s
.empty() ) 
 157         // conversion failed, but we still want to show something to the user 
 158         // even if it's going to be wrong it is better than nothing 
 160         // we choose ISO8859-1 here arbitrarily, it's just the most common 
 161         // encoding probably and, also importantly here, conversion from it 
 162         // never fails as it's done internally by wxCSConv 
 163         wbuf 
= wxCSConv(wxFONTENCODING_ISO8859_1
).cMB2WC(s
.c_str()); 
 166     return wxConvUTF8
.cWC2MB(wbuf
); 
 169 WXDLLIMPEXP_CORE wxCharBuffer
 
 170 wxConvertFromGTK(const wxString
& s
, wxFontEncoding enc
) 
 172     // this conversion should never fail as GTK+ always uses UTF-8 internally 
 173     // so there are no complications here 
 174     const wxWCharBuffer 
wbuf(wxConvUTF8
.cMB2WC(s
.c_str())); 
 175     if ( enc 
== wxFONTENCODING_SYSTEM 
) 
 176         return wxConvUI
->cWC2MB(wbuf
); 
 178     return wxCSConv(enc
).cWC2MB(wbuf
); 
 181 #endif // !wxUSE_UNICODE 
 183 // Returns NULL if version is certainly greater or equal than major.minor.micro 
 184 // Returns string describing the error if version is lower than 
 185 // major.minor.micro OR it cannot be determined and one should not rely on the 
 186 // availability of pango version major.minor.micro, nor the non-availability 
 187 const gchar 
*wx_pango_version_check (int major
, int minor
, int micro
) 
 189 #ifdef PANGO_VERSION_MAJOR 
 190     if (!gtk_check_version (2,11,0)) 
 192         // GTK+ 2.11 requires Pango >= 1.15.3 and pango_version_check 
 193         // was added in Pango 1.15.2 thus we know for sure the pango lib we're 
 194         // using has the pango_version_check function: 
 195         return pango_version_check (major
, minor
, micro
); 
 198     return "can't check"; 
 199 #else // !PANGO_VERSION_MAJOR 
 204     return "too old headers"; 
 209 // ---------------------------------------------------------------------------- 
 210 // subprocess routines 
 211 // ---------------------------------------------------------------------------- 
 215 void GTK_EndProcessDetector(gpointer data
, gint source
, 
 216                             GdkInputCondition 
WXUNUSED(condition
) ) 
 218    wxEndProcessData 
*proc_data 
= (wxEndProcessData 
*)data
; 
 220    // has the process really terminated? unfortunately GDK (or GLib) seem to 
 221    // generate G_IO_HUP notification even when it simply tries to read from a 
 222    // closed fd and hasn't terminated at all 
 223    int pid 
= (proc_data
->pid 
> 0) ? proc_data
->pid 
: -(proc_data
->pid
); 
 225    int rc 
= waitpid(pid
, &status
, WNOHANG
); 
 229        // no, it didn't exit yet, continue waiting 
 233    // set exit code to -1 if something bad happened 
 234    proc_data
->exitcode 
= rc 
!= -1 && WIFEXITED(status
) ? WEXITSTATUS(status
) 
 237    // child exited, end waiting 
 240    // don't call us again! 
 241    gdk_input_remove(proc_data
->tag
); 
 243    wxHandleProcessTermination(proc_data
); 
 247 int wxAddProcessCallback(wxEndProcessData 
*proc_data
, int fd
) 
 249     int tag 
= gdk_input_add(fd
, 
 251                             GTK_EndProcessDetector
, 
 252                             (gpointer
)proc_data
); 
 259 // ---------------------------------------------------------------------------- 
 260 // wxPlatformInfo-related 
 261 // ---------------------------------------------------------------------------- 
 263 wxPortId 
wxGUIAppTraits::GetToolkitVersion(int *verMaj
, int *verMin
) const 
 266         *verMaj 
= gtk_major_version
; 
 268         *verMin 
= gtk_minor_version
; 
 275 wxTimerImpl 
*wxGUIAppTraits::CreateTimerImpl(wxTimer 
*timer
) 
 277     return new wxGTKTimerImpl(timer
); 
 280 #endif // wxUSE_TIMER 
 283 static wxString 
GetSM() 
 288         Dpy() { m_dpy 
= XOpenDisplay(NULL
); } 
 289         ~Dpy() { if ( m_dpy 
) XCloseDisplay(m_dpy
); } 
 291         operator Display 
*() const { return m_dpy
; } 
 297         return wxEmptyString
; 
 301     SmcConn smc_conn 
= SmcOpenConnection(NULL
, NULL
, 
 303                                          0 /* mask */, NULL 
/* callbacks */, 
 305                                          WXSIZEOF(smerr
), smerr
); 
 309         wxLogWarning(_("Failed to connect to session manager: %s"), smerr
); 
 310         return wxEmptyString
; 
 313     char *vendor 
= SmcVendor(smc_conn
); 
 314     wxString ret 
= wxString::FromAscii( vendor 
); 
 317     SmcCloseConnection(smc_conn
, 0, NULL
); 
 322 #endif // wxUSE_DETECT_SM 
 325 //----------------------------------------------------------------------------- 
 327 //----------------------------------------------------------------------------- 
 329 wxEventLoopBase 
*wxGUIAppTraits::CreateEventLoop() 
 331     return new wxEventLoop(); 
 336 void wxGUIAppTraits::SetLocale() 
 339     wxUpdateLocaleIsUtf8(); 
 345 #if wxUSE_STACKWALKER 
 347 // private helper class 
 348 class StackDump 
: public wxStackWalker
 
 351     StackDump(GtkAssertDialog 
*dlg
) { m_dlg
=dlg
; } 
 354     virtual void OnStackFrame(const wxStackFrame
& frame
) 
 356         wxString fncname 
= frame
.GetName(); 
 357         wxString fncargs 
= fncname
; 
 359         size_t n 
= fncname
.find(wxT('(')); 
 360         if (n 
!= wxString::npos
) 
 362             // remove arguments from function name 
 365             // remove function name and brackets from arguments 
 366             fncargs 
= fncargs
.substr(n
+1, fncargs
.length()-n
-2); 
 369             fncargs 
= wxEmptyString
; 
 371         // append this stack frame's info in the dialog 
 372         if (!frame
.GetFileName().empty() || !fncname
.empty()) 
 373             gtk_assert_dialog_append_stack_frame(m_dlg
, 
 376                                                 frame
.GetFileName().mb_str(), 
 381     GtkAssertDialog 
*m_dlg
; 
 384 // the callback functions must be extern "C" to comply with GTK+ declarations 
 387     void get_stackframe_callback(StackDump 
*dump
) 
 389         // skip over frames up to including wxOnAssert() 
 390         dump
->ProcessFrames(3); 
 394 #endif      // wxUSE_STACKWALKER 
 396 bool wxGUIAppTraits::ShowAssertDialog(const wxString
& msg
) 
 398     // under GTK2 we prefer to use a dialog widget written using directly GTK+; 
 399     // in fact we cannot use a dialog written using wxWidgets: it would need 
 400     // the wxWidgets idle processing to work correctly! 
 401     GtkWidget 
*dialog 
= gtk_assert_dialog_new(); 
 402     gtk_assert_dialog_set_message(GTK_ASSERT_DIALOG(dialog
), msg
.mb_str()); 
 404 #if wxUSE_STACKWALKER 
 405     // don't show more than maxLines or we could get a dialog too tall to be 
 406     // shown on screen: 20 should be ok everywhere as even with 15 pixel high 
 407     // characters it is still only 300 pixels... 
 408     static const int maxLines 
= 20; 
 410     // save current stack frame... 
 411     StackDump 
dump(GTK_ASSERT_DIALOG(dialog
)); 
 412     dump
.SaveStack(maxLines
); 
 414     // ...but process it only if the user needs it 
 415     gtk_assert_dialog_set_backtrace_callback(GTK_ASSERT_DIALOG(dialog
), 
 416                                              (GtkAssertDialogStackFrameCallback
)get_stackframe_callback
, 
 418 #endif      // wxUSE_STACKWALKER 
 420     gint result 
= gtk_dialog_run(GTK_DIALOG (dialog
)); 
 421     bool returnCode 
= false; 
 424     case GTK_ASSERT_DIALOG_STOP
: 
 427     case GTK_ASSERT_DIALOG_CONTINUE
: 
 430     case GTK_ASSERT_DIALOG_CONTINUE_SUPPRESSING
: 
 436         wxFAIL_MSG( _T("unexpected return code from GtkAssertDialog") ); 
 439     gtk_widget_destroy(dialog
); 
 443 #endif  // __WXDEBUG__ 
 445 wxString 
wxGUIAppTraits::GetDesktopEnvironment() const 
 447     wxString de 
= wxSystemOptions::GetOption(_T("gtk.desktop")); 
 451         static const wxString s_SM 
= GetSM(); 
 453         if (s_SM 
== wxT("GnomeSM")) 
 455         else if (s_SM 
== wxT("KDE")) 
 458 #endif // wxUSE_DETECT_SM 
 465 // see the hack below in wxCmdLineParser::GetUsageString(). 
 466 // TODO: replace this hack with a g_option_group_get_entries() 
 467 //       call as soon as such function exists 
 472   gchar           
*help_description
; 
 474   GDestroyNotify   destroy_notify
; 
 477   GTranslateFunc   translate_func
; 
 478   GDestroyNotify   translate_notify
; 
 479   gpointer     translate_data
; 
 481   GOptionEntry    
*entries
; 
 484   GOptionParseFunc pre_parse_func
; 
 485   GOptionParseFunc post_parse_func
; 
 486   GOptionErrorFunc error_func
; 
 489 wxString 
wxGetNameFromGtkOptionEntry(const GOptionEntry 
*opt
) 
 494         ret 
<< _T("-") << opt
->short_name
; 
 499         ret 
<< _T("--") << opt
->long_name
; 
 501         if (opt
->arg_description
) 
 502             ret 
<< _T("=") << opt
->arg_description
; 
 505     return _T("  ") + ret
; 
 508 #endif // __WXGTK26__ 
 511 wxGUIAppTraits::GetStandardCmdLineOptions(wxArrayString
& names
, 
 512                                           wxArrayString
& desc
) const 
 517     // check whether GTK version is greater than 2.6 but also lower than 2.12 
 518     // because, as we use the undocumented _GOptionGroup struct, we don't want 
 519     // to run this code with future versions which might change it (2.11 is the 
 520     // latest one at the time of this writing) 
 521     if (!gtk_check_version(2,6,0) && 
 522         gtk_check_version(2,12,0)) 
 524         usage 
<< _("The following standard GTK+ options are also supported:\n"); 
 526         // passing true here means that the function can open the default 
 527         // display while parsing (not really used here anyhow) 
 528         GOptionGroup 
*gtkOpts 
= gtk_get_option_group(true); 
 530         // WARNING: here we access the internals of GOptionGroup: 
 531         GOptionEntry 
*entries 
= ((_GOptionGroup
*)gtkOpts
)->entries
; 
 532         unsigned int n_entries 
= ((_GOptionGroup
*)gtkOpts
)->n_entries
; 
 533         wxArrayString namesOptions
, descOptions
; 
 535         for ( size_t n 
= 0; n 
< n_entries
; n
++ ) 
 537             if ( entries
[n
].flags 
& G_OPTION_FLAG_HIDDEN 
) 
 540             names
.push_back(wxGetNameFromGtkOptionEntry(&entries
[n
])); 
 542             const gchar 
* const entryDesc 
= entries
[n
].description
; 
 543             desc
.push_back(entryDesc 
? wxString(entryDesc
) : _T("")); 
 546         g_option_group_free (gtkOpts
); 
 551 #endif // __WXGTK26__