1 /////////////////////////////////////////////////////////////////////////////
4 // Author: Robert Roebling
6 // Copyright: (c) 1998 Robert Roebling, Julian Smart
7 // Licence: wxWindows licence
8 /////////////////////////////////////////////////////////////////////////////
11 #pragma implementation "app.h"
15 #include "wx/gdicmn.h"
19 #include "wx/memory.h"
21 #include "wx/settings.h"
22 #include "wx/dialog.h"
24 #if wxUSE_WX_RESOURCES
25 #include "wx/resource.h"
28 #include "wx/module.h"
32 #include "wx/thread.h"
41 #include "wx/gtk/win_gtk.h"
43 //-----------------------------------------------------------------------------
45 //-----------------------------------------------------------------------------
47 wxApp
*wxTheApp
= (wxApp
*) NULL
;
48 wxAppInitializerFunction
wxAppBase::m_appInitFn
= (wxAppInitializerFunction
) NULL
;
52 //-----------------------------------------------------------------------------
54 //-----------------------------------------------------------------------------
56 /* forward declaration */
57 gint
wxapp_idle_callback( gpointer
WXUNUSED(data
) );
58 void wxapp_install_idle_handler();
61 gint
wxapp_wakeup_timerout_callback( gpointer
WXUNUSED(data
) );
64 //-----------------------------------------------------------------------------
66 //-----------------------------------------------------------------------------
73 //-----------------------------------------------------------------------------
75 //-----------------------------------------------------------------------------
79 bool has_idle
= (wxTheApp
->m_idleTag
!= 0);
83 /* We need to temporarily remove idle callbacks or the loop will
85 gtk_idle_remove( wxTheApp
->m_idleTag
);
86 wxTheApp
->m_idleTag
= 0;
89 while (gtk_events_pending())
92 /* it's necessary to call ProcessIdle() to update the frames sizes which
93 might have been changed (it also will update other things set from
94 OnUpdateUI() which is a nice (and desired) side effect) */
95 while (wxTheApp
->ProcessIdle()) { }
99 /* re-add idle handler */
100 wxTheApp
->m_idleTag
= gtk_idle_add( wxapp_idle_callback
, (gpointer
) NULL
);
106 //-----------------------------------------------------------------------------
108 //-----------------------------------------------------------------------------
113 if (!wxThread::IsMain())
118 wxapp_install_idle_handler();
121 if (!wxThread::IsMain())
126 //-----------------------------------------------------------------------------
128 //-----------------------------------------------------------------------------
130 gint
wxapp_idle_callback( gpointer
WXUNUSED(data
) )
132 if (!wxTheApp
) return TRUE
;
134 // when getting called from GDK's time-out handler
135 // we are no longer within GDK's grab on the GUI
136 // thread so we must lock it here ourselves
139 /* sent idle event to all who request them */
140 while (wxTheApp
->ProcessIdle()) { }
142 /* we don't want any more idle events until the next event is
144 gtk_idle_remove( wxTheApp
->m_idleTag
);
145 wxTheApp
->m_idleTag
= 0;
147 /* indicate that we are now in idle mode - even so deeply
148 in idle mode that we don't get any idle events anymore.
149 this is like wxMSW where an idle event is sent only
150 once each time after the event queue has been completely
154 // release lock again
160 void wxapp_install_idle_handler()
162 wxASSERT_MSG( wxTheApp
->m_idleTag
== 0, wxT("attempt to install idle handler twice") );
164 /* This routine gets called by all event handlers
165 indicating that the idle is over. It may also
166 get called from other thread for sending events
167 to the main thread (and processing these in
170 wxTheApp
->m_idleTag
= gtk_idle_add( wxapp_idle_callback
, (gpointer
) NULL
);
177 void wxapp_install_thread_wakeup()
179 if (wxTheApp
->m_wakeUpTimerTag
) return;
181 wxTheApp
->m_wakeUpTimerTag
= gtk_timeout_add( 500, wxapp_wakeup_timerout_callback
, (gpointer
) NULL
);
184 void wxapp_uninstall_thread_wakeup()
186 if (!wxTheApp
->m_wakeUpTimerTag
) return;
188 gtk_timeout_remove( wxTheApp
->m_wakeUpTimerTag
);
189 wxTheApp
->m_wakeUpTimerTag
= 0;
192 gint
wxapp_wakeup_timerout_callback( gpointer
WXUNUSED(data
) )
194 // when getting called from GDK's time-out handler
195 // we are no longer within GDK's grab on the GUI
196 // thread so we must lock it here ourselves
199 wxapp_uninstall_thread_wakeup();
201 // unblock other threads wishing to do some GUI things
204 // wake up other threads
207 // block other thread again
210 wxapp_install_thread_wakeup();
212 // release lock again
218 #endif // wxUSE_THREADS
220 //-----------------------------------------------------------------------------
222 //-----------------------------------------------------------------------------
224 IMPLEMENT_DYNAMIC_CLASS(wxApp
,wxEvtHandler
)
226 BEGIN_EVENT_TABLE(wxApp
, wxEvtHandler
)
227 EVT_IDLE(wxApp::OnIdle
)
234 m_topWindow
= (wxWindow
*) NULL
;
235 m_exitOnFrameDelete
= TRUE
;
237 m_idleTag
= gtk_idle_add( wxapp_idle_callback
, (gpointer
) NULL
);
240 m_wakeUpTimerTag
= 0;
241 wxapp_install_thread_wakeup();
244 m_colorCube
= (unsigned char*) NULL
;
249 if (m_idleTag
) gtk_idle_remove( m_idleTag
);
252 wxapp_uninstall_thread_wakeup();
255 if (m_colorCube
) free(m_colorCube
);
258 bool wxApp::OnInitGui()
260 GdkVisual
*visual
= gdk_visual_get_system();
262 /* on some machines, the default visual is just 256 colours, so
263 we make sure we get the best. this can sometimes be wasteful,
264 of course, but what do these guys pay $30.000 for? */
266 if (gdk_visual_get_best() != gdk_visual_get_system())
268 GdkVisual* vis = gdk_visual_get_best();
269 gtk_widget_set_default_visual( vis );
271 GdkColormap *colormap = gdk_colormap_new( vis, FALSE );
272 gtk_widget_set_default_colormap( colormap );
278 /* Nothing to do for 15, 16, 24, 32 bit displays */
279 if (visual
->depth
> 8) return TRUE
;
281 /* initialize color cube for 8-bit color reduction dithering */
283 GdkColormap
*cmap
= gtk_widget_get_default_colormap();
285 m_colorCube
= (unsigned char*)malloc(32 * 32 * 32);
287 for (int r
= 0; r
< 32; r
++)
289 for (int g
= 0; g
< 32; g
++)
291 for (int b
= 0; b
< 32; b
++)
293 int rr
= (r
<< 3) | (r
>> 2);
294 int gg
= (g
<< 3) | (g
>> 2);
295 int bb
= (b
<< 3) | (b
>> 2);
299 GdkColor
*colors
= cmap
->colors
;
304 for (int i
= 0; i
< cmap
->size
; i
++)
306 int rdiff
= ((rr
<< 8) - colors
[i
].red
);
307 int gdiff
= ((gg
<< 8) - colors
[i
].green
);
308 int bdiff
= ((bb
<< 8) - colors
[i
].blue
);
309 int sum
= ABS (rdiff
) + ABS (gdiff
) + ABS (bdiff
);
312 index
= i
; max
= sum
;
318 #if (GTK_MINOR_VERSION > 0)
319 /* assume 8-bit true or static colors. this really
321 GdkVisual
* vis
= gdk_colormap_get_visual( cmap
);
322 index
= (r
>> (5 - vis
->red_prec
)) << vis
->red_shift
;
323 index
|= (g
>> (5 - vis
->green_prec
)) << vis
->green_shift
;
324 index
|= (b
>> (5 - vis
->blue_prec
)) << vis
->blue_shift
;
326 wxFAIL_MSG( wxT("Unsupported graphics hardware") );
329 m_colorCube
[ (r
*1024) + (g
*32) + b
] = index
;
337 bool wxApp::ProcessIdle()
340 event
.SetEventObject( this );
341 ProcessEvent( event
);
343 return event
.MoreRequested();
346 void wxApp::OnIdle( wxIdleEvent
&event
)
348 static bool s_inOnIdle
= FALSE
;
350 /* Avoid recursion (via ProcessEvent default case) */
356 /* Resend in the main thread events which have been prepared in other
358 ProcessPendingEvents();
360 /* 'Garbage' collection of windows deleted with Close(). */
361 DeletePendingObjects();
363 /* flush the logged messages if any */
365 wxLog
*log
= wxLog::GetActiveTarget();
366 if (log
!= NULL
&& log
->HasPendingMessages())
370 /* Send OnIdle events to all windows */
371 bool needMore
= SendIdleEvents();
374 event
.RequestMore(TRUE
);
379 bool wxApp::SendIdleEvents()
381 bool needMore
= FALSE
;
383 wxWindowList::Node
* node
= wxTopLevelWindows
.GetFirst();
386 wxWindow
* win
= node
->GetData();
387 if (SendIdleEvents(win
))
389 node
= node
->GetNext();
395 bool wxApp::SendIdleEvents( wxWindow
* win
)
397 bool needMore
= FALSE
;
400 event
.SetEventObject(win
);
402 win
->ProcessEvent(event
);
404 win
->OnInternalIdle();
406 if (event
.MoreRequested())
409 wxNode
* node
= win
->GetChildren().First();
412 wxWindow
* win
= (wxWindow
*) node
->Data();
413 if (SendIdleEvents(win
))
421 int wxApp::MainLoop()
427 void wxApp::ExitMainLoop()
429 if (gtk_main_level() > 0)
433 bool wxApp::Initialized()
435 return m_initialized
;
438 bool wxApp::Pending()
440 return (gtk_events_pending() > 0);
443 void wxApp::Dispatch()
445 gtk_main_iteration();
448 void wxApp::DeletePendingObjects()
450 wxNode
*node
= wxPendingDelete
.First();
453 wxObject
*obj
= (wxObject
*)node
->Data();
457 if (wxPendingDelete
.Find(obj
))
460 node
= wxPendingDelete
.First();
464 bool wxApp::Initialize()
466 wxBuffer
= new wxChar
[BUFSIZ
+ 512];
468 wxClassInfo::InitializeClasses();
470 wxSystemSettings::Init();
472 // GL: I'm annoyed ... I don't know where to put this and I don't want to
473 // create a module for that as it's part of the core.
475 wxPendingEvents
= new wxList();
476 wxPendingEventsLocker
= new wxCriticalSection();
480 wxTheFontNameDirectory = new wxFontNameDirectory;
481 wxTheFontNameDirectory->Initialize();
484 wxTheColourDatabase
= new wxColourDatabase( wxKEY_STRING
);
485 wxTheColourDatabase
->Initialize();
487 wxInitializeStockLists();
488 wxInitializeStockObjects();
490 wxModule::RegisterModules();
491 if (!wxModule::InitializeModules()) return FALSE
;
496 void wxApp::CleanUp()
498 wxModule::CleanUpModules();
500 if (wxTheColourDatabase
)
501 delete wxTheColourDatabase
;
502 wxTheColourDatabase
= (wxColourDatabase
*) NULL
;
505 if (wxTheFontNameDirectory) delete wxTheFontNameDirectory;
506 wxTheFontNameDirectory = (wxFontNameDirectory*) NULL;
509 wxDeleteStockObjects();
511 wxDeleteStockLists();
514 wxTheApp
= (wxApp
*) NULL
;
516 // GL: I'm annoyed ... I don't know where to put this and I don't want to
517 // create a module for that as it's part of the core.
519 delete wxPendingEvents
;
520 delete wxPendingEventsLocker
;
523 wxSystemSettings::Done();
527 wxClassInfo::CleanUpClasses();
529 // check for memory leaks
530 #if (defined(__WXDEBUG__) && wxUSE_MEMORY_TRACING) || wxUSE_DEBUG_CONTEXT
531 if (wxDebugContext::CountObjectsLeft() > 0)
533 wxLogDebug(wxT("There were memory leaks.\n"));
534 wxDebugContext::Dump();
535 wxDebugContext::PrintStatistics();
540 // do this as the very last thing because everything else can log messages
541 wxLog::DontCreateOnDemand();
543 wxLog
*oldLog
= wxLog::SetActiveTarget( (wxLog
*) NULL
);
549 //-----------------------------------------------------------------------------
551 //-----------------------------------------------------------------------------
553 int wxEntry( int argc
, char *argv
[] )
562 if (!wxOKlibc()) wxConvCurrent
= &wxConvLocal
;
564 if (!wxOKlibc()) wxConvCurrent
= (wxMBConv
*) NULL
;
569 gtk_init( &argc
, &argv
);
571 wxSetDetectableAutoRepeat( TRUE
);
573 if (!wxApp::Initialize())
581 wxCHECK_MSG( wxApp::GetInitializerFunction(), -1,
582 wxT("wxWindows error: No initializer - use IMPLEMENT_APP macro.\n") );
584 wxAppInitializerFunction app_ini
= wxApp::GetInitializerFunction();
586 wxObject
*test_app
= app_ini();
588 wxTheApp
= (wxApp
*) test_app
;
591 wxCHECK_MSG( wxTheApp
, -1, wxT("wxWindows error: no application object") );
593 wxTheApp
->argc
= argc
;
595 wxTheApp
->argv
= new wxChar
*[argc
+1];
597 while (mb_argc
< argc
)
599 wxTheApp
->argv
[mb_argc
] = wxStrdup(wxConvLibc
.cMB2WX(argv
[mb_argc
]));
602 wxTheApp
->argv
[mb_argc
] = (wxChar
*)NULL
;
604 wxTheApp
->argv
= argv
;
607 wxString
name(wxFileNameFromPath(argv
[0]));
608 wxStripExtension( name
);
609 wxTheApp
->SetAppName( name
);
613 if ( !wxTheApp
->OnInitGui() )
616 // Here frames insert themselves automatically into wxTopLevelWindows by
617 // getting created in OnInit().
620 if ( !wxTheApp
->OnInit() )
626 /* delete pending toplevel windows (typically a single
627 dialog) so that, if there isn't any left, we don't
629 wxTheApp
->DeletePendingObjects();
631 wxTheApp
->m_initialized
= wxTopLevelWindows
.GetCount() != 0;
633 if (wxTheApp
->Initialized())
635 retValue
= wxTheApp
->OnRun();
637 wxWindow
*topWindow
= wxTheApp
->GetTopWindow();
640 /* Forcibly delete the window. */
641 if (topWindow
->IsKindOf(CLASSINFO(wxFrame
)) ||
642 topWindow
->IsKindOf(CLASSINFO(wxDialog
)) )
644 topWindow
->Close( TRUE
);
645 wxTheApp
->DeletePendingObjects();
650 wxTheApp
->SetTopWindow( (wxWindow
*) NULL
);
658 // flush the logged messages if any
659 wxLog
*log
= wxLog::GetActiveTarget();
660 if (log
!= NULL
&& log
->HasPendingMessages())
663 // continuing to use user defined log target is unsafe from now on because
664 // some resources may be already unavailable, so replace it by something
666 wxLog
*oldlog
= wxLog::SetActiveTarget(new wxLogStderr
);
678 #include "wx/gtk/info.xpm"
679 #include "wx/gtk/error.xpm"
680 #include "wx/gtk/question.xpm"
681 #include "wx/gtk/warning.xpm"
684 wxApp::GetStdIcon(int which
) const
688 case wxICON_INFORMATION
:
689 return wxIcon(info_xpm
);
691 case wxICON_QUESTION
:
692 return wxIcon(question_xpm
);
694 case wxICON_EXCLAMATION
:
695 return wxIcon(warning_xpm
);
698 wxFAIL_MSG(wxT("requested non existent standard icon"));
699 // still fall through
702 return wxIcon(error_xpm
);