]>
git.saurik.com Git - wxWidgets.git/blob - src/gtk/app.cpp
   1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/app.cpp 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling, Julian Smart 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // For compilers that support precompilation, includes "wx.h". 
  11 #include "wx/wxprec.h" 
  19     #include "wx/memory.h" 
  23 #include "wx/thread.h" 
  29 #include "wx/gtk/private.h" 
  30 #include "wx/apptrait.h" 
  33     #include <hildon-widgets/hildon-program.h> 
  34 #endif // wxUSE_LIBHILDON 
  38 //----------------------------------------------------------------------------- 
  40 //----------------------------------------------------------------------------- 
  42 #if wxUSE_MIMETYPE && wxUSE_LIBGNOMEVFS 
  44     wxFORCE_LINK_MODULE(gnome_vfs
) 
  47 //----------------------------------------------------------------------------- 
  49 //----------------------------------------------------------------------------- 
  51 static GtkWidget 
*gs_RootWindow 
= (GtkWidget
*) NULL
; 
  53 //----------------------------------------------------------------------------- 
  55 //----------------------------------------------------------------------------- 
  57 // not static because used by textctrl.cpp 
  60 bool wxIsInsideYield 
= false; 
  62 bool wxApp::Yield(bool onlyIfNeeded
) 
  64     if ( wxIsInsideYield 
) 
  68             wxFAIL_MSG( wxT("wxYield called recursively" ) ); 
  75     if ( !wxThread::IsMain() ) 
  77         // can't call gtk_main_iteration() from other threads like this 
  80 #endif // wxUSE_THREADS 
  82     wxIsInsideYield 
= true; 
  85     // disable log flushing from here because a call to wxYield() shouldn't 
  86     // normally result in message boxes popping up &c 
  90     while (EventsPending()) 
  93     // It's necessary to call ProcessIdle() to update the frames sizes which 
  94     // might have been changed (it also will update other things set from 
  95     // OnUpdateUI() which is a nice (and desired) side effect). But we 
  96     // call ProcessIdle() only once since this is not meant for longish 
  97     // background jobs (controlled by wxIdleEvent::RequestMore() and the 
  98     // return value of Processidle(). 
 102     // let the logs be flashed again 
 106     wxIsInsideYield 
= false; 
 111 //----------------------------------------------------------------------------- 
 113 //----------------------------------------------------------------------------- 
 115 // One-shot signal emission hook, to install idle handler. 
 118 wx_emission_hook(GSignalInvocationHint
*, guint
, const GValue
*, gpointer data
) 
 120     wxApp
* app 
= wxTheApp
; 
 123     gulong
* hook_id 
= (gulong
*)data
; 
 124     // record that hook is not installed 
 131 // Add signal emission hooks, to re-install idle handler when needed. 
 132 static void wx_add_idle_hooks() 
 136         static gulong hook_id 
= 0; 
 139             static guint sig_id 
= 0; 
 141                 sig_id 
= g_signal_lookup("event", GTK_TYPE_WIDGET
); 
 142             hook_id 
= g_signal_add_emission_hook( 
 143                 sig_id
, 0, wx_emission_hook
, &hook_id
, NULL
); 
 146     // "size_allocate" hook 
 147     // Needed to match the behavior of the old idle system, 
 148     // but probably not necessary. 
 150         static gulong hook_id 
= 0; 
 153             static guint sig_id 
= 0; 
 155                 sig_id 
= g_signal_lookup("size_allocate", GTK_TYPE_WIDGET
); 
 156             hook_id 
= g_signal_add_emission_hook( 
 157                 sig_id
, 0, wx_emission_hook
, &hook_id
, NULL
); 
 163 static gboolean 
wxapp_idle_callback(gpointer
) 
 165     return wxTheApp
->DoIdle(); 
 173         // Allow another idle source to be added while this one is busy. 
 174         // Needed if an idle event handler runs a new event loop, 
 175         // for example by showing a dialog. 
 177         wxMutexLocker 
lock(*m_idleMutex
); 
 179         id_save 
= m_idleSourceId
; 
 183         // don't generate the idle events while the assert modal dialog is shown, 
 184         // this matches the behavior of wxMSW 
 193         needMore 
= ProcessIdle(); 
 194     } while (needMore 
&& gtk_events_pending() == 0); 
 198     wxMutexLocker 
lock(*m_idleMutex
); 
 200     // if a new idle source was added during ProcessIdle 
 201     if (m_idleSourceId 
!= 0) 
 204         g_source_remove(m_idleSourceId
); 
 207     // if more idle processing requested 
 210         // keep this source installed 
 211         m_idleSourceId 
= id_save
; 
 214     // add hooks and remove this source 
 219 //----------------------------------------------------------------------------- 
 220 // Access to the root window global 
 221 //----------------------------------------------------------------------------- 
 223 GtkWidget
* wxGetRootWindow() 
 225     if (gs_RootWindow 
== NULL
) 
 227         gs_RootWindow 
= gtk_window_new( GTK_WINDOW_TOPLEVEL 
); 
 228         gtk_widget_realize( gs_RootWindow 
); 
 230     return gs_RootWindow
; 
 233 //----------------------------------------------------------------------------- 
 235 //----------------------------------------------------------------------------- 
 237 IMPLEMENT_DYNAMIC_CLASS(wxApp
,wxEvtHandler
) 
 242     m_isInAssert 
= false; 
 243 #endif // __WXDEBUG__ 
 254 bool wxApp::OnInitGui() 
 256     if ( !wxAppBase::OnInitGui() ) 
 259     // if this is a wxGLApp (derived from wxApp), and we've already 
 260     // chosen a specific visual, then derive the GdkVisual from that 
 261     if ( GetXVisualInfo() ) 
 263         GdkVisual
* vis 
= gtk_widget_get_default_visual(); 
 265         GdkColormap 
*colormap 
= gdk_colormap_new( vis
, FALSE 
); 
 266         gtk_widget_set_default_colormap( colormap 
); 
 270         // On some machines, the default visual is just 256 colours, so 
 271         // we make sure we get the best. This can sometimes be wasteful. 
 274             if (m_forceTrueColour
) 
 276                 GdkVisual
* visual 
= gdk_visual_get_best_with_both( 24, GDK_VISUAL_TRUE_COLOR 
); 
 279                     wxLogError(wxT("Unable to initialize TrueColor visual.")); 
 282                 GdkColormap 
*colormap 
= gdk_colormap_new( visual
, FALSE 
); 
 283                 gtk_widget_set_default_colormap( colormap 
); 
 287                 if (gdk_visual_get_best() != gdk_visual_get_system()) 
 289                     GdkVisual
* visual 
= gdk_visual_get_best(); 
 290                     GdkColormap 
*colormap 
= gdk_colormap_new( visual
, FALSE 
); 
 291                     gtk_widget_set_default_colormap( colormap 
); 
 298     m_hildonProgram 
= hildon_program_get_instance(); 
 299     if ( !m_hildonProgram 
) 
 301         wxLogError(_("Unable to initialize Hildon program")); 
 304 #endif // wxUSE_LIBHILDON 
 309 GdkVisual 
*wxApp::GetGdkVisual() 
 311     GdkVisual 
*visual 
= NULL
; 
 313     XVisualInfo 
*xvi 
= (XVisualInfo 
*)GetXVisualInfo(); 
 315         visual 
= gdkx_visual_get( xvi
->visualid 
); 
 317         visual 
= gdk_drawable_get_visual( wxGetRootWindow()->window 
); 
 324 // use unusual names for the parameters to avoid conflict with wxApp::arg[cv] 
 325 bool wxApp::Initialize(int& argc_
, wxChar 
**argv_
) 
 327     if ( !wxAppBase::Initialize(argc_
, argv_
) ) 
 331     if (!g_thread_supported()) 
 336 #endif // wxUSE_THREADS 
 338     // We should have the wxUSE_WCHAR_T test on the _outside_ 
 340     // gtk+ 2.0 supports Unicode through UTF-8 strings 
 341     wxConvCurrent 
= &wxConvUTF8
; 
 342 #else // !wxUSE_WCHAR_T 
 344         wxConvCurrent 
= (wxMBConv
*) NULL
; 
 345 #endif // wxUSE_WCHAR_T/!wxUSE_WCHAR_T 
 347     // decide which conversion to use for the file names 
 349     // (1) this variable exists for the sole purpose of specifying the encoding 
 350     //     of the filenames for GTK+ programs, so use it if it is set 
 351     wxString 
encName(wxGetenv(_T("G_FILENAME_ENCODING"))); 
 352     encName 
= encName
.BeforeFirst(_T(',')); 
 353     if (encName
.CmpNoCase(_T("@locale")) == 0) 
 359         // (2) if a non default locale is set, assume that the user wants his 
 360         //     filenames in this locale too 
 361         encName 
= wxLocale::GetSystemEncodingName().Upper(); 
 362         // (3) finally use UTF-8 by default 
 363         if (encName
.empty() || encName 
== _T("US-ASCII")) 
 364             encName 
= _T("UTF-8"); 
 365         wxSetEnv(_T("G_FILENAME_ENCODING"), encName
); 
 369         encName 
= _T("UTF-8"); 
 371     // if wxUSE_INTL==0 it probably indicates that only "C" locale is supported 
 372     // by the program anyhow so prevent GTK+ from calling setlocale(LC_ALL, "") 
 373     // from gtk_init_check() as it does by default 
 374     gtk_disable_setlocale(); 
 377     static wxConvBrokenFileNames 
fileconv(encName
); 
 378     wxConvFileName 
= &fileconv
; 
 385     // gtk_init() wants UTF-8, not wchar_t, so convert 
 386     char **argvGTK 
= new char *[argc_ 
+ 1]; 
 387     for ( i 
= 0; i 
< argc_
; i
++ ) 
 389         argvGTK
[i
] = wxStrdupA(wxConvUTF8
.cWX2MB(argv_
[i
])); 
 392     argvGTK
[argc_
] = NULL
; 
 397     init_result 
= true;  // is there a _check() version of this? 
 398     gpe_application_init( &argcGTK
, &argvGTK 
); 
 400     init_result 
= gtk_init_check( &argcGTK
, &argvGTK 
); 
 402     wxUpdateLocaleIsUtf8(); 
 404     if ( argcGTK 
!= argc_ 
) 
 406         // we have to drop the parameters which were consumed by GTK+ 
 407         for ( i 
= 0; i 
< argcGTK
; i
++ ) 
 409             while ( strcmp(wxConvUTF8
.cWX2MB(argv_
[i
]), argvGTK
[i
]) != 0 ) 
 411                 memmove(argv_ 
+ i
, argv_ 
+ i 
+ 1, (argc_ 
- i
)*sizeof(*argv_
)); 
 417     //else: gtk_init() didn't modify our parameters 
 420     for ( i 
= 0; i 
< argcGTK
; i
++ ) 
 426 #else // !wxUSE_UNICODE 
 427     // gtk_init() shouldn't actually change argv_ itself (just its contents) so 
 428     // it's ok to pass pointer to it 
 429     init_result 
= gtk_init_check( &argc_
, &argv_ 
); 
 430 #endif // wxUSE_UNICODE/!wxUSE_UNICODE 
 432     // update internal arg[cv] as GTK+ may have removed processed options: 
 438         // if there are still GTK+ standard options unparsed in the command 
 439         // line, it means that they were not syntactically correct and GTK+ 
 440         // already printed a warning on the command line and we should now 
 442         wxArrayString opt
, desc
; 
 443         m_traits
->GetStandardCmdLineOptions(opt
, desc
); 
 445         for ( i 
= 0; i 
< argc_
; i
++ ) 
 447             // leave just the names of the options with values 
 448             const wxString str 
= wxString(argv_
[i
]).BeforeFirst('='); 
 450             for ( size_t j 
= 0; j 
< opt
.size(); j
++ ) 
 452                 // remove the leading spaces from the option string as it does 
 454                 if ( opt
[j
].Trim(false).BeforeFirst('=') == str 
) 
 456                     // a GTK+ option can be left on the command line only if 
 457                     // there was an error in (or before, in another standard 
 458                     // options) it, so abort, just as we do if incorrect 
 459                     // program option is given 
 460                     wxLogError(_("Invalid GTK+ command line option, use \"%s --help\""), 
 470         wxLogError(_("Unable to initialize GTK+, is DISPLAY set properly?")); 
 474     // we can not enter threads before gtk_init is done 
 477     wxSetDetectableAutoRepeat( true ); 
 480     wxFont::SetDefaultEncoding(wxLocale::GetSystemEncoding()); 
 484     m_idleMutex 
= new wxMutex
; 
 486     // make sure GtkWidget type is loaded, idle hooks need it 
 487     g_type_class_ref(GTK_TYPE_WIDGET
); 
 493 void wxApp::CleanUp() 
 495     if (m_idleSourceId 
!= 0) 
 496         g_source_remove(m_idleSourceId
); 
 498     // release reference acquired by Initialize() 
 499     g_type_class_unref(g_type_class_peek(GTK_TYPE_WIDGET
)); 
 503     wxAppBase::CleanUp(); 
 505     // delete this mutex as late as possible as it's used from WakeUpIdle(), in 
 506     // particular do it after calling the base class CleanUp() which can result 
 507     // in it being called 
 514 void wxApp::WakeUpIdle() 
 517     wxMutexLocker 
lock(*m_idleMutex
); 
 519     if (m_idleSourceId 
== 0) 
 520         m_idleSourceId 
= g_idle_add_full(G_PRIORITY_LOW
, wxapp_idle_callback
, NULL
, NULL
); 
 523 // Checking for pending events requires first removing our idle source, 
 524 // otherwise it will cause the check to always return true. 
 525 bool wxApp::EventsPending() 
 528     wxMutexLocker 
lock(*m_idleMutex
); 
 530     if (m_idleSourceId 
!= 0) 
 532         g_source_remove(m_idleSourceId
); 
 536     return gtk_events_pending() != 0; 
 541 void wxApp::OnAssertFailure(const wxChar 
*file
, 
 548     // block wx idle events while assert dialog is showing 
 551     wxAppBase::OnAssertFailure(file
, line
, func
, cond
, msg
); 
 553     m_isInAssert 
= false; 
 556 #endif // __WXDEBUG__ 
 559 void wxGUIAppTraits::MutexGuiEnter() 
 564 void wxGUIAppTraits::MutexGuiLeave() 
 568 #endif // wxUSE_THREADS