1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/gtk/toplevel.cpp 
   4 // Author:      Robert Roebling 
   6 // Copyright:   (c) 1998 Robert Roebling 
   7 // Licence:     wxWindows licence 
   8 ///////////////////////////////////////////////////////////////////////////// 
  10 // ============================================================================ 
  12 // ============================================================================ 
  14 // ---------------------------------------------------------------------------- 
  16 // ---------------------------------------------------------------------------- 
  18 // For compilers that support precompilation, includes "wx.h". 
  19 #include "wx/wxprec.h" 
  22 #define XIconifyWindow XICONIFYWINDOW 
  27 #include "wx/toplevel.h" 
  29 #include "wx/dialog.h" 
  30 #include "wx/control.h" 
  32 #include "wx/dcclient.h" 
  33 #include "wx/gtk/private.h" 
  35 #include "wx/settings.h" 
  36 #include "wx/evtloop.h" 
  41 #include <gdk/gdkkeysyms.h> 
  44 #include "wx/gtk/win_gtk.h" 
  46 #include "wx/unix/utilsx11.h" 
  49 #include <X11/Xatom.h> 
  51 // ---------------------------------------------------------------------------- 
  53 // ---------------------------------------------------------------------------- 
  55 extern void wxapp_install_idle_handler(); 
  58 // ---------------------------------------------------------------------------- 
  60 // ---------------------------------------------------------------------------- 
  62 extern wxList           wxPendingDelete
; 
  64 extern int              g_openDialogs
; 
  65 extern wxWindowGTK     
*g_delayedFocus
; 
  67 // the frame that is currently active (i.e. its child has focus). It is 
  68 // used to generate wxActivateEvents 
  69 static wxTopLevelWindowGTK 
*g_activeFrame 
= (wxTopLevelWindowGTK
*) NULL
; 
  70 static wxTopLevelWindowGTK 
*g_lastActiveFrame 
= (wxTopLevelWindowGTK
*) NULL
; 
  72 // if we detect that the app has got/lost the focus, we set this variable to 
  73 // either TRUE or FALSE and an activate event will be sent during the next 
  74 // OnIdle() call and it is reset to -1: this value means that we shouldn't 
  75 // send any activate events at all 
  76 static int g_sendActivateEvent 
= -1; 
  78 //----------------------------------------------------------------------------- 
  79 // RequestUserAttention related functions 
  80 //----------------------------------------------------------------------------- 
  83 static void wxgtk_window_set_urgency_hint (GtkWindow 
*win
, 
  86     wxASSERT_MSG( GTK_WIDGET_REALIZED(win
), wxT("wxgtk_window_set_urgency_hint: GdkWindow not realized") ); 
  87     GdkWindow 
*window 
= GTK_WIDGET(win
)->window
; 
  90     wm_hints 
= XGetWMHints(GDK_WINDOW_XDISPLAY(window
), GDK_WINDOW_XWINDOW(window
)); 
  93         wm_hints 
= XAllocWMHints(); 
  96         wm_hints
->flags 
|= XUrgencyHint
; 
  98         wm_hints
->flags 
&= ~XUrgencyHint
; 
 100     XSetWMHints(GDK_WINDOW_XDISPLAY(window
), GDK_WINDOW_XWINDOW(window
), wm_hints
); 
 104 static gint 
gtk_frame_urgency_timer_callback( wxTopLevelWindowGTK 
*win 
) 
 106 #if defined(__WXGTK20__) && GTK_CHECK_VERSION(2,7,0) 
 107     if(!gtk_check_version(2,7,0)) 
 108         gtk_window_set_urgency_hint(GTK_WINDOW( win
->m_widget 
), FALSE
); 
 111         wxgtk_window_set_urgency_hint(GTK_WINDOW( win
->m_widget 
), FALSE
); 
 113     win
->m_urgency_hint 
= -2; 
 118 //----------------------------------------------------------------------------- 
 120 //----------------------------------------------------------------------------- 
 123 static gint 
gtk_frame_focus_in_callback( GtkWidget 
*widget
, 
 124                                          GdkEvent 
*WXUNUSED(event
), 
 125                                          wxTopLevelWindowGTK 
*win 
) 
 128         wxapp_install_idle_handler(); 
 130     switch ( g_sendActivateEvent 
) 
 133             // we've got focus from outside, synthetize wxActivateEvent 
 134             g_sendActivateEvent 
= 1; 
 138             // another our window just lost focus, it was already ours before 
 139             // - don't send any wxActivateEvent 
 140             g_sendActivateEvent 
= -1; 
 145     g_lastActiveFrame 
= g_activeFrame
; 
 147     // wxPrintf( wxT("active: %s\n"), win->GetTitle().c_str() ); 
 149     // MR: wxRequestUserAttention related block 
 150     switch( win
->m_urgency_hint 
) 
 153             gtk_timeout_remove( win
->m_urgency_hint 
); 
 154             // no break, fallthrough to remove hint too 
 156 #if defined(__WXGTK20__) && GTK_CHECK_VERSION(2,7,0) 
 157             if(!gtk_check_version(2,7,0)) 
 158                 gtk_window_set_urgency_hint(GTK_WINDOW( widget 
), FALSE
); 
 162                 wxgtk_window_set_urgency_hint(GTK_WINDOW( widget 
), FALSE
); 
 165             win
->m_urgency_hint 
= -2; 
 171     wxLogTrace(wxT("activate"), wxT("Activating frame %p (from focus_in)"), g_activeFrame
); 
 172     wxActivateEvent 
event(wxEVT_ACTIVATE
, true, g_activeFrame
->GetId()); 
 173     event
.SetEventObject(g_activeFrame
); 
 174     g_activeFrame
->GetEventHandler()->ProcessEvent(event
); 
 180 //----------------------------------------------------------------------------- 
 182 //----------------------------------------------------------------------------- 
 185 static gint 
gtk_frame_focus_out_callback( GtkWidget 
*widget
, 
 186                                           GdkEventFocus 
*WXUNUSED(gdk_event
), 
 187                                           wxTopLevelWindowGTK 
*win 
) 
 190         wxapp_install_idle_handler(); 
 192     // if the focus goes out of our app alltogether, OnIdle() will send 
 193     // wxActivateEvent, otherwise gtk_window_focus_in_callback() will reset 
 194     // g_sendActivateEvent to -1 
 195     g_sendActivateEvent 
= 0; 
 197     // wxASSERT_MSG( (g_activeFrame == win), wxT("TLW deactivatd although it wasn't active") ); 
 199     // wxPrintf( wxT("inactive: %s\n"), win->GetTitle().c_str() ); 
 203         wxLogTrace(wxT("activate"), wxT("Activating frame %p (from focus_in)"), g_activeFrame
); 
 204         wxActivateEvent 
event(wxEVT_ACTIVATE
, false, g_activeFrame
->GetId()); 
 205         event
.SetEventObject(g_activeFrame
); 
 206         g_activeFrame
->GetEventHandler()->ProcessEvent(event
); 
 208         g_activeFrame 
= NULL
; 
 215 //----------------------------------------------------------------------------- 
 216 // "focus" from m_window 
 217 //----------------------------------------------------------------------------- 
 220 static gint 
gtk_frame_focus_callback( GtkWidget 
*widget
, GtkDirectionType 
WXUNUSED(d
), wxWindow 
*WXUNUSED(win
) ) 
 223         wxapp_install_idle_handler(); 
 225     // This disables GTK's tab traversal 
 226     gtk_signal_emit_stop_by_name( GTK_OBJECT(widget
), "focus" ); 
 231 //----------------------------------------------------------------------------- 
 233 //----------------------------------------------------------------------------- 
 236 static void gtk_frame_size_callback( GtkWidget 
*WXUNUSED(widget
), GtkAllocation
* alloc
, wxTopLevelWindowGTK 
*win 
) 
 239         wxapp_install_idle_handler(); 
 244     if ((win
->m_width 
!= alloc
->width
) || (win
->m_height 
!= alloc
->height
)) 
 247         wxPrintf( "OnSize from " ); 
 248         if (win->GetClassInfo() && win->GetClassInfo()->GetClassName()) 
 249            wxPrintf( win->GetClassInfo()->GetClassName() ); 
 250         wxPrintf( " %d %d %d %d\n", (int)alloc->x, 
 253                                 (int)alloc->height ); 
 256         win
->m_width 
= alloc
->width
; 
 257         win
->m_height 
= alloc
->height
; 
 258         win
->GtkUpdateSize(); 
 263 //----------------------------------------------------------------------------- 
 265 //----------------------------------------------------------------------------- 
 268 static gint 
gtk_frame_delete_callback( GtkWidget 
*WXUNUSED(widget
), GdkEvent 
*WXUNUSED(event
), wxTopLevelWindowGTK 
*win 
) 
 271         wxapp_install_idle_handler(); 
 273     if (win
->IsEnabled() && 
 274         (g_openDialogs 
== 0 || (win
->GetExtraStyle() & wxTOPLEVEL_EX_DIALOG
) || 
 283 //----------------------------------------------------------------------------- 
 285 //----------------------------------------------------------------------------- 
 289 gtk_frame_configure_callback( GtkWidget 
*WXUNUSED(widget
), GdkEventConfigure 
*WXUNUSED(event
), wxTopLevelWindowGTK 
*win 
) 
 292         wxapp_install_idle_handler(); 
 294     if (!win
->m_hasVMT 
|| !win
->IsShown()) 
 300     gdk_window_get_root_origin( win
->m_widget
->window
, &x
, &y 
); 
 304     wxMoveEvent 
mevent( wxPoint(win
->m_x
,win
->m_y
), win
->GetId() ); 
 305     mevent
.SetEventObject( win 
); 
 306     win
->GetEventHandler()->ProcessEvent( mevent 
); 
 312 //----------------------------------------------------------------------------- 
 313 // "realize" from m_widget 
 314 //----------------------------------------------------------------------------- 
 316 // we cannot MWM hints and icons before the widget has been realized, 
 317 // so we do this directly after realization 
 321 gtk_frame_realized_callback( GtkWidget 
* WXUNUSED(widget
), 
 322                              wxTopLevelWindowGTK 
*win 
) 
 325         wxapp_install_idle_handler(); 
 327     // All this is for Motif Window Manager "hints" and is supposed to be 
 328     // recognized by other WM as well. Not tested. 
 329     gdk_window_set_decorations(win
->m_widget
->window
, 
 330                                (GdkWMDecoration
)win
->m_gdkDecor
); 
 331     gdk_window_set_functions(win
->m_widget
->window
, 
 332                                (GdkWMFunction
)win
->m_gdkFunc
); 
 334     // GTK's shrinking/growing policy 
 335     if ((win
->m_gdkFunc 
& GDK_FUNC_RESIZE
) == 0) 
 336         gtk_window_set_policy(GTK_WINDOW(win
->m_widget
), 0, 0, 1); 
 338         gtk_window_set_policy(GTK_WINDOW(win
->m_widget
), 1, 1, 1); 
 341     wxIconBundle iconsOld 
= win
->GetIcons(); 
 342     if ( iconsOld
.GetIcon(-1).Ok() ) 
 344         win
->SetIcon( wxNullIcon 
); 
 345         win
->SetIcons( iconsOld 
); 
 350 //----------------------------------------------------------------------------- 
 351 // "map_event" from m_widget 
 352 //----------------------------------------------------------------------------- 
 356 gtk_frame_map_callback( GtkWidget 
* WXUNUSED(widget
), 
 357                         GdkEvent 
* WXUNUSED(event
), 
 358                         wxTopLevelWindow 
*win 
) 
 360     win
->SetIconizeState(false); 
 364 //----------------------------------------------------------------------------- 
 365 // "unmap_event" from m_widget 
 366 //----------------------------------------------------------------------------- 
 370 gtk_frame_unmap_callback( GtkWidget 
* WXUNUSED(widget
), 
 371                           GdkEvent 
* WXUNUSED(event
), 
 372                           wxTopLevelWindow 
*win 
) 
 374     win
->SetIconizeState(true); 
 378 //----------------------------------------------------------------------------- 
 379 // "expose_event" of m_client 
 380 //----------------------------------------------------------------------------- 
 383 static int gtk_window_expose_callback( GtkWidget 
*widget
, GdkEventExpose 
*gdk_event
, wxWindow 
*win 
) 
 385     GtkPizza 
*pizza 
= GTK_PIZZA(widget
); 
 387     gtk_paint_flat_box (win
->m_widget
->style
, 
 388                         pizza
->bin_window
, GTK_STATE_NORMAL
, 
 399 //----------------------------------------------------------------------------- 
 400 // "draw" of m_client 
 401 //----------------------------------------------------------------------------- 
 406 static void gtk_window_draw_callback( GtkWidget 
*widget
, GdkRectangle 
*rect
, wxWindow 
*win 
) 
 408     GtkPizza 
*pizza 
= GTK_PIZZA(widget
); 
 410     gtk_paint_flat_box (win
->m_widget
->style
, 
 411                         pizza
->bin_window
, GTK_STATE_NORMAL
, 
 422 // ---------------------------------------------------------------------------- 
 423 // wxTopLevelWindowGTK itself 
 424 // ---------------------------------------------------------------------------- 
 426 //----------------------------------------------------------------------------- 
 427 // InsertChild for wxTopLevelWindowGTK 
 428 //----------------------------------------------------------------------------- 
 430 /* Callback for wxTopLevelWindowGTK. This very strange beast has to be used because 
 431  * C++ has no virtual methods in a constructor. We have to emulate a 
 432  * virtual function here as wxWidgets requires different ways to insert 
 433  * a child in container classes. */ 
 435 static void wxInsertChildInTopLevelWindow( wxTopLevelWindowGTK
* parent
, wxWindow
* child 
) 
 437     wxASSERT( GTK_IS_WIDGET(child
->m_widget
) ); 
 439     if (!parent
->m_insertInClientArea
) 
 441         // these are outside the client area 
 442         wxTopLevelWindowGTK
* frame 
= (wxTopLevelWindowGTK
*) parent
; 
 443         gtk_pizza_put( GTK_PIZZA(frame
->m_mainWidget
), 
 444                          GTK_WIDGET(child
->m_widget
), 
 452         // these are inside the client area 
 453         gtk_pizza_put( GTK_PIZZA(parent
->m_wxwindow
), 
 454                          GTK_WIDGET(child
->m_widget
), 
 461     // resize on OnInternalIdle 
 462     parent
->GtkUpdateSize(); 
 465 // ---------------------------------------------------------------------------- 
 466 // wxTopLevelWindowGTK creation 
 467 // ---------------------------------------------------------------------------- 
 469 void wxTopLevelWindowGTK::Init() 
 474     m_mainWidget 
= (GtkWidget
*) NULL
; 
 475     m_insertInClientArea 
= true; 
 476     m_isIconized 
= false; 
 477     m_fsIsShowing 
= false; 
 478     m_themeEnabled 
= true; 
 479     m_gdkDecor 
= m_gdkFunc 
= 0; 
 485 bool wxTopLevelWindowGTK::Create( wxWindow 
*parent
, 
 487                                   const wxString
& title
, 
 489                                   const wxSize
& sizeOrig
, 
 491                                   const wxString 
&name 
) 
 493     // always create a frame of some reasonable, even if arbitrary, size (at 
 494     // least for MSW compatibility) 
 495     wxSize size 
= sizeOrig
; 
 496     size
.x 
= WidthDefault(size
.x
); 
 497     size
.y 
= HeightDefault(size
.y
); 
 499     wxTopLevelWindows
.Append( this ); 
 501     m_needParent 
= false; 
 503     if (!PreCreation( parent
, pos
, size 
) || 
 504         !CreateBase( parent
, id
, pos
, size
, style
, wxDefaultValidator
, name 
)) 
 506         wxFAIL_MSG( wxT("wxTopLevelWindowGTK creation failed") ); 
 512     m_insertCallback 
= (wxInsertChildFunction
) wxInsertChildInTopLevelWindow
; 
 514     // NB: m_widget may be !=NULL if it was created by derived class' Create, 
 515     //     e.g. in wxTaskBarIconAreaGTK 
 516     if (m_widget 
== NULL
) 
 518         if (GetExtraStyle() & wxTOPLEVEL_EX_DIALOG
) 
 521             m_widget 
= gtk_window_new(GTK_WINDOW_TOPLEVEL
); 
 522             // Tell WM that this is a dialog window and make it center 
 523             // on parent by default (this is what GtkDialog ctor does): 
 524             gtk_window_set_type_hint(GTK_WINDOW(m_widget
), 
 525                                      GDK_WINDOW_TYPE_HINT_DIALOG
); 
 526             gtk_window_set_position(GTK_WINDOW(m_widget
), 
 527                                     GTK_WIN_POS_CENTER_ON_PARENT
); 
 529             m_widget 
= gtk_window_new(GTK_WINDOW_DIALOG
); 
 534             m_widget 
= gtk_window_new(GTK_WINDOW_TOPLEVEL
); 
 535 #if GTK_CHECK_VERSION(2,1,0) 
 536             if (!gtk_check_version(2,1,0)) 
 538                 if (style 
& wxFRAME_TOOL_WINDOW
) 
 540                     gtk_window_set_type_hint(GTK_WINDOW(m_widget
), 
 541                                              GDK_WINDOW_TYPE_HINT_UTILITY
); 
 543                     // On some WMs, like KDE, a TOOL_WINDOW will still show 
 544                     // on the taskbar, but on Gnome a TOOL_WINDOW will not. 
 545                     // For consistency between WMs and with Windows, we 
 546                     // should set the NO_TASKBAR flag which will apply 
 547                     // the set_skip_taskbar_hint if it is available, 
 548                     // ensuring no taskbar entry will appear. 
 549                     style 
|= wxFRAME_NO_TASKBAR
; 
 556     wxWindow 
*topParent 
= wxGetTopLevelParent(m_parent
); 
 557     if (topParent 
&& (((GTK_IS_WINDOW(topParent
->m_widget
)) && 
 558                        (GetExtraStyle() & wxTOPLEVEL_EX_DIALOG
)) || 
 559                        (style 
& wxFRAME_FLOAT_ON_PARENT
))) 
 561         gtk_window_set_transient_for( GTK_WINDOW(m_widget
), 
 562                                       GTK_WINDOW(topParent
->m_widget
) ); 
 565 #if GTK_CHECK_VERSION(2,2,0) 
 566     if (!gtk_check_version(2,2,0)) 
 568         if (style 
& wxFRAME_NO_TASKBAR
) 
 570             gtk_window_set_skip_taskbar_hint(GTK_WINDOW(m_widget
), TRUE
); 
 576     if (!gtk_check_version(2,4,0)) 
 578         if (style 
& wxSTAY_ON_TOP
) 
 580             gtk_window_set_keep_above(GTK_WINDOW(m_widget
), TRUE
); 
 586         gtk_window_set_wmclass( GTK_WINDOW(m_widget
), wxGTK_CONV( name 
), wxGTK_CONV( name 
) ); 
 588     gtk_window_set_title( GTK_WINDOW(m_widget
), wxGTK_CONV( title 
) ); 
 589     GTK_WIDGET_UNSET_FLAGS( m_widget
, GTK_CAN_FOCUS 
); 
 591     gtk_signal_connect( GTK_OBJECT(m_widget
), "delete_event", 
 592         GTK_SIGNAL_FUNC(gtk_frame_delete_callback
), (gpointer
)this ); 
 594     // m_mainWidget holds the toolbar, the menubar and the client area 
 595     m_mainWidget 
= gtk_pizza_new(); 
 596     gtk_widget_show( m_mainWidget 
); 
 597     GTK_WIDGET_UNSET_FLAGS( m_mainWidget
, GTK_CAN_FOCUS 
); 
 598     gtk_container_add( GTK_CONTAINER(m_widget
), m_mainWidget 
); 
 600     if (m_miniEdge 
== 0) // wxMiniFrame has its own version. 
 602        // For m_mainWidget themes 
 603        gtk_signal_connect( GTK_OBJECT(m_mainWidget
), "expose_event", 
 604                 GTK_SIGNAL_FUNC(gtk_window_expose_callback
), (gpointer
)this ); 
 606        gtk_signal_connect( GTK_OBJECT(m_mainWidget
), "draw", 
 607                 GTK_SIGNAL_FUNC(gtk_window_draw_callback
), (gpointer
)this ); 
 611     // m_wxwindow only represents the client area without toolbar and menubar 
 612     m_wxwindow 
= gtk_pizza_new(); 
 613     gtk_widget_show( m_wxwindow 
); 
 614     gtk_container_add( GTK_CONTAINER(m_mainWidget
), m_wxwindow 
); 
 616     // we donm't allow the frame to get the focus as otherwise 
 617     // the frame will grab it at arbitrary focus changes 
 618     GTK_WIDGET_UNSET_FLAGS( m_wxwindow
, GTK_CAN_FOCUS 
); 
 620     if (m_parent
) m_parent
->AddChild( this ); 
 622     // the user resized the frame by dragging etc. 
 623     gtk_signal_connect( GTK_OBJECT(m_widget
), "size_allocate", 
 624         GTK_SIGNAL_FUNC(gtk_frame_size_callback
), (gpointer
)this ); 
 628     if ((m_x 
!= -1) || (m_y 
!= -1)) 
 629         gtk_widget_set_uposition( m_widget
, m_x
, m_y 
); 
 631     gtk_window_set_default_size( GTK_WINDOW(m_widget
), m_width
, m_height 
); 
 633     //  we cannot set MWM hints and icons before the widget has 
 634     //  been realized, so we do this directly after realization 
 635     gtk_signal_connect( GTK_OBJECT(m_widget
), "realize", 
 636                         GTK_SIGNAL_FUNC(gtk_frame_realized_callback
), (gpointer
) this ); 
 638     // map and unmap for iconized state 
 639     gtk_signal_connect( GTK_OBJECT(m_widget
), "map_event", 
 640         GTK_SIGNAL_FUNC(gtk_frame_map_callback
), (gpointer
)this ); 
 641     gtk_signal_connect( GTK_OBJECT(m_widget
), "unmap_event", 
 642         GTK_SIGNAL_FUNC(gtk_frame_unmap_callback
), (gpointer
)this ); 
 644     // the only way to get the window size is to connect to this event 
 645     gtk_signal_connect( GTK_OBJECT(m_widget
), "configure_event", 
 646         GTK_SIGNAL_FUNC(gtk_frame_configure_callback
), (gpointer
)this ); 
 648     // disable native tab traversal 
 649     gtk_signal_connect( GTK_OBJECT(m_widget
), "focus", 
 650         GTK_SIGNAL_FUNC(gtk_frame_focus_callback
), (gpointer
)this ); 
 653     gtk_signal_connect( GTK_OBJECT(m_widget
), "focus_in_event", 
 654         GTK_SIGNAL_FUNC(gtk_frame_focus_in_callback
), (gpointer
)this ); 
 655     gtk_signal_connect( GTK_OBJECT(m_widget
), "focus_out_event", 
 656         GTK_SIGNAL_FUNC(gtk_frame_focus_out_callback
), (gpointer
)this ); 
 659     if ((m_miniEdge 
> 0) || (style 
& wxSIMPLE_BORDER
) || (style 
& wxNO_BORDER
)) 
 666         m_gdkDecor 
= (long) GDK_DECOR_BORDER
; 
 667         m_gdkFunc 
= (long) GDK_FUNC_MOVE
; 
 669         // All this is for Motif Window Manager "hints" and is supposed to be 
 670         // recognized by other WMs as well. 
 671         if ((style 
& wxCAPTION
) != 0) 
 673             m_gdkDecor 
|= GDK_DECOR_TITLE
; 
 675         if ((style 
& wxCLOSE_BOX
) != 0) 
 677             m_gdkFunc 
|= GDK_FUNC_CLOSE
; 
 679         if ((style 
& wxSYSTEM_MENU
) != 0) 
 681             m_gdkDecor 
|= GDK_DECOR_MENU
; 
 683         if ((style 
& wxMINIMIZE_BOX
) != 0) 
 685             m_gdkFunc 
|= GDK_FUNC_MINIMIZE
; 
 686             m_gdkDecor 
|= GDK_DECOR_MINIMIZE
; 
 688         if ((style 
& wxMAXIMIZE_BOX
) != 0) 
 690             m_gdkFunc 
|= GDK_FUNC_MAXIMIZE
; 
 691             m_gdkDecor 
|= GDK_DECOR_MAXIMIZE
; 
 693         if ((style 
& wxRESIZE_BORDER
) != 0) 
 695            m_gdkFunc 
|= GDK_FUNC_RESIZE
; 
 696            m_gdkDecor 
|= GDK_DECOR_RESIZEH
; 
 703 wxTopLevelWindowGTK::~wxTopLevelWindowGTK() 
 707         wxASSERT_MSG( false, _T("Window still grabbed")); 
 711     m_isBeingDeleted 
= true; 
 713     // it may also be GtkScrolledWindow in the case of an MDI child 
 714     if (GTK_IS_WINDOW(m_widget
)) 
 716         gtk_window_set_focus( GTK_WINDOW(m_widget
), NULL 
); 
 719     if (g_activeFrame 
== this) 
 720         g_activeFrame 
= NULL
; 
 721     if (g_lastActiveFrame 
== this) 
 722         g_lastActiveFrame 
= NULL
; 
 727 bool wxTopLevelWindowGTK::ShowFullScreen(bool show
, long style 
) 
 729     if (show 
== m_fsIsShowing
) 
 730         return false; // return what? 
 732     m_fsIsShowing 
= show
; 
 734     wxX11FullScreenMethod method 
= 
 735         wxGetFullScreenMethodX11((WXDisplay
*)GDK_DISPLAY(), 
 736                                  (WXWindow
)GDK_ROOT_WINDOW()); 
 738 #if GTK_CHECK_VERSION(2,2,0) 
 739     // NB: gtk_window_fullscreen() uses freedesktop.org's WMspec extensions 
 740     //     to switch to fullscreen, which is not always available. We must 
 741     //     check if WM supports the spec and use legacy methods if it 
 743     if ( (method 
== wxX11_FS_WMSPEC
) && !gtk_check_version(2,2,0) ) 
 746             gtk_window_fullscreen( GTK_WINDOW( m_widget 
) ); 
 748             gtk_window_unfullscreen( GTK_WINDOW( m_widget 
) ); 
 751 #endif // GTK+ >= 2.2.0 
 753         GdkWindow 
*window 
= m_widget
->window
; 
 757             m_fsSaveFlag 
= style
; 
 758             GetPosition( &m_fsSaveFrame
.x
, &m_fsSaveFrame
.y 
); 
 759             GetSize( &m_fsSaveFrame
.width
, &m_fsSaveFrame
.height 
); 
 761             int screen_width
,screen_height
; 
 762             wxDisplaySize( &screen_width
, &screen_height 
); 
 764             gint client_x
, client_y
, root_x
, root_y
; 
 767             if (method 
!= wxX11_FS_WMSPEC
) 
 769                 // don't do it always, Metacity hates it 
 770                 m_fsSaveGdkFunc 
= m_gdkFunc
; 
 771                 m_fsSaveGdkDecor 
= m_gdkDecor
; 
 772                 m_gdkFunc 
= m_gdkDecor 
= 0; 
 773                 gdk_window_set_decorations(window
, (GdkWMDecoration
)0); 
 774                 gdk_window_set_functions(window
, (GdkWMFunction
)0); 
 777             gdk_window_get_origin (m_widget
->window
, &root_x
, &root_y
); 
 778             gdk_window_get_geometry (m_widget
->window
, &client_x
, &client_y
, 
 779                          &width
, &height
, NULL
); 
 781             gdk_window_move_resize (m_widget
->window
, -client_x
, -client_y
, 
 782                         screen_width 
+ 1, screen_height 
+ 1); 
 784             wxSetFullScreenStateX11((WXDisplay
*)GDK_DISPLAY(), 
 785                                     (WXWindow
)GDK_ROOT_WINDOW(), 
 786                                     (WXWindow
)GDK_WINDOW_XWINDOW(window
), 
 787                                     show
, &m_fsSaveFrame
, method
); 
 791             if (method 
!= wxX11_FS_WMSPEC
) 
 793                 // don't do it always, Metacity hates it 
 794                 m_gdkFunc 
= m_fsSaveGdkFunc
; 
 795                 m_gdkDecor 
= m_fsSaveGdkDecor
; 
 796                 gdk_window_set_decorations(window
, (GdkWMDecoration
)m_gdkDecor
); 
 797                 gdk_window_set_functions(window
, (GdkWMFunction
)m_gdkFunc
); 
 800             wxSetFullScreenStateX11((WXDisplay
*)GDK_DISPLAY(), 
 801                                     (WXWindow
)GDK_ROOT_WINDOW(), 
 802                                     (WXWindow
)GDK_WINDOW_XWINDOW(window
), 
 803                                     show
, &m_fsSaveFrame
, method
); 
 805             SetSize(m_fsSaveFrame
.x
, m_fsSaveFrame
.y
, 
 806                     m_fsSaveFrame
.width
, m_fsSaveFrame
.height
); 
 810     // documented behaviour is to show the window if it's still hidden when 
 811     // showing it full screen 
 812     if ( show 
&& !IsShown() ) 
 818 // ---------------------------------------------------------------------------- 
 819 // overridden wxWindow methods 
 820 // ---------------------------------------------------------------------------- 
 822 bool wxTopLevelWindowGTK::Show( bool show 
) 
 824     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid frame") ); 
 826     if (show 
&& !m_sizeSet
) 
 828         /* by calling GtkOnSize here, we don't have to call 
 829            either after showing the frame, which would entail 
 830            much ugly flicker or from within the size_allocate 
 831            handler, because GTK 1.1.X forbids that. */ 
 833         GtkOnSize( m_x
, m_y
, m_width
, m_height 
); 
 837         gtk_widget_set_uposition( m_widget
, m_x
, m_y 
); 
 839     return wxWindow::Show( show 
); 
 842 void wxTopLevelWindowGTK::Raise() 
 845     gtk_window_present( GTK_WINDOW( m_widget 
) ); 
 851 void wxTopLevelWindowGTK::DoMoveWindow(int WXUNUSED(x
), int WXUNUSED(y
), int WXUNUSED(width
), int WXUNUSED(height
) ) 
 853     wxFAIL_MSG( wxT("DoMoveWindow called for wxTopLevelWindowGTK") ); 
 856 void wxTopLevelWindowGTK::DoSetSize( int x
, int y
, int width
, int height
, int sizeFlags 
) 
 858     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid frame") ); 
 860     // this shouldn't happen: wxFrame, wxMDIParentFrame and wxMDIChildFrame have m_wxwindow 
 861     wxASSERT_MSG( (m_wxwindow 
!= NULL
), wxT("invalid frame") ); 
 871     int old_width 
= m_width
; 
 872     int old_height 
= m_height
; 
 874     if ((sizeFlags 
& wxSIZE_ALLOW_MINUS_ONE
) == 0) 
 876         if (x 
!= -1) m_x 
= x
; 
 877         if (y 
!= -1) m_y 
= y
; 
 884     if (width 
!= -1) m_width 
= width
; 
 885     if (height 
!= -1) m_height 
= height
; 
 888     if ((sizeFlags & wxSIZE_AUTO_WIDTH) == wxSIZE_AUTO_WIDTH) 
 890         if (width == -1) m_width = 80; 
 893     if ((sizeFlags & wxSIZE_AUTO_HEIGHT) == wxSIZE_AUTO_HEIGHT) 
 895        if (height == -1) m_height = 26; 
 899     int minWidth 
= GetMinWidth(), 
 900         minHeight 
= GetMinHeight(), 
 901         maxWidth 
= GetMaxWidth(), 
 902         maxHeight 
= GetMaxHeight(); 
 905     // GPE's window manager doesn't like size hints 
 906     // at all, esp. when the user has to use the 
 914     if ((minWidth 
!= -1) && (m_width 
< minWidth
)) m_width 
= minWidth
; 
 915     if ((minHeight 
!= -1) && (m_height 
< minHeight
)) m_height 
= minHeight
; 
 916     if ((maxWidth 
!= -1) && (m_width 
> maxWidth
)) m_width 
= maxWidth
; 
 917     if ((maxHeight 
!= -1) && (m_height 
> maxHeight
)) m_height 
= maxHeight
; 
 919     if ((m_x 
!= -1) || (m_y 
!= -1)) 
 921         if ((m_x 
!= old_x
) || (m_y 
!= old_y
)) 
 923             gtk_widget_set_uposition( m_widget
, m_x
, m_y 
); 
 927     if ((m_width 
!= old_width
) || (m_height 
!= old_height
)) 
 929         if (m_widget
->window
) 
 930             gdk_window_resize( m_widget
->window
, m_width
, m_height 
); 
 932             gtk_window_set_default_size( GTK_WINDOW(m_widget
), m_width
, m_height 
); 
 934         /* we set the size in GtkOnSize, i.e. mostly the actual resizing is 
 935            done either directly before the frame is shown or in idle time 
 936            so that different calls to SetSize() don't lead to flicker. */ 
 943 void wxTopLevelWindowGTK::DoGetClientSize( int *width
, int *height 
) const 
 945     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid frame") ); 
 947     wxWindow::DoGetClientSize( width
, height 
); 
 951         *height 
-= m_miniEdge
*2 + m_miniTitle
; 
 955         *width 
-= m_miniEdge
*2; 
 959 void wxTopLevelWindowGTK::DoSetClientSize( int width
, int height 
) 
 961     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid frame") ); 
 964               width 
+ m_miniEdge
*2, height  
+ m_miniEdge
*2 + m_miniTitle
, 0); 
 967 void wxTopLevelWindowGTK::GtkOnSize( int WXUNUSED(x
), int WXUNUSED(y
), 
 968                                      int width
, int height 
) 
 970     // due to a bug in gtk, x,y are always 0 
 975     if (m_resizing
) return; 
 978     if ( m_wxwindow 
== NULL 
) return; 
 983     /* wxMDIChildFrame derives from wxFrame but it _is_ a wxWindow as it uses 
 984        wxWindow::Create to create it's GTK equivalent. m_mainWidget is only 
 985        set in wxFrame::Create so it is used to check what kind of frame we 
 986        have here. if m_mainWidget is NULL it is a wxMDIChildFrame and so we 
 987        skip the part which handles m_frameMenuBar, m_frameToolBar and (most 
 988        importantly) m_mainWidget */ 
 990     int minWidth 
= GetMinWidth(), 
 991         minHeight 
= GetMinHeight(), 
 992         maxWidth 
= GetMaxWidth(), 
 993         maxHeight 
= GetMaxHeight(); 
 996     // GPE's window manager doesn't like size hints 
 997     // at all, esp. when the user has to use the 
1005     if ((minWidth 
!= -1) && (m_width 
< minWidth
)) m_width 
= minWidth
; 
1006     if ((minHeight 
!= -1) && (m_height 
< minHeight
)) m_height 
= minHeight
; 
1007     if ((maxWidth 
!= -1) && (m_width 
> maxWidth
)) m_width 
= maxWidth
; 
1008     if ((maxHeight 
!= -1) && (m_height 
> maxHeight
)) m_height 
= maxHeight
; 
1013         gint            flag 
= 0; // GDK_HINT_POS; 
1016         if ((minWidth 
!= -1) || (minHeight 
!= -1)) flag 
|= GDK_HINT_MIN_SIZE
; 
1017         if ((maxWidth 
!= -1) || (maxHeight 
!= -1)) flag 
|= GDK_HINT_MAX_SIZE
; 
1019         geom
.min_width 
= minWidth
; 
1020         geom
.min_height 
= minHeight
; 
1022             // Because of the way we set GDK_HINT_MAX_SIZE above, if either of 
1023             // maxHeight or maxWidth is set, we must set them both, else the 
1024             // remaining -1 will be taken literally. 
1026             // I'm certain this also happens elsewhere, and is the probable 
1027             // cause of other such things as: 
1028             // Gtk-WARNING **: gtk_widget_size_allocate(): 
1029             //       attempt to allocate widget with width 65535 and height 600 
1030             // but I don't have time to track them all now.. 
1032             // Really we need to encapulate all this height/width business and 
1033             // stop any old method from ripping at the members directly and 
1034             // scattering -1's without regard for who might resolve them later. 
1036         geom
.max_width 
= ( maxHeight 
== -1 ) ? maxWidth
 
1037                          : ( maxWidth 
== -1 ) ? wxGetDisplaySize().GetWidth() 
1040         geom
.max_height 
= ( maxWidth 
== -1 ) ? maxHeight    
// ( == -1 here ) 
1041                           : ( maxHeight 
== -1 ) ? wxGetDisplaySize().GetHeight() 
1044         gtk_window_set_geometry_hints( GTK_WINDOW(m_widget
), 
1047                                        (GdkWindowHints
) flag 
); 
1049         /* I revert back to wxGTK's original behaviour. m_mainWidget holds the 
1050          * menubar, the toolbar and the client area, which is represented by 
1052          * this hurts in the eye, but I don't want to call SetSize() 
1053          * because I don't want to call any non-native functions here. */ 
1055         int client_x 
= m_miniEdge
; 
1056         int client_y 
= m_miniEdge 
+ m_miniTitle
; 
1057         int client_w 
= m_width 
- 2*m_miniEdge
; 
1058         int client_h 
= m_height 
- 2*m_miniEdge 
- m_miniTitle
; 
1060         gtk_pizza_set_size( GTK_PIZZA(m_mainWidget
), 
1062                               client_x
, client_y
, client_w
, client_h 
); 
1066         // If there is no m_mainWidget between m_widget and m_wxwindow there 
1067         // is no need to set the size or position of m_wxwindow. 
1072     // send size event to frame 
1073     wxSizeEvent 
event( wxSize(m_width
,m_height
), GetId() ); 
1074     event
.SetEventObject( this ); 
1075     GetEventHandler()->ProcessEvent( event 
); 
1080 void wxTopLevelWindowGTK::OnInternalIdle() 
1082     if (!m_sizeSet 
&& GTK_WIDGET_REALIZED(m_wxwindow
)) 
1084         GtkOnSize( m_x
, m_y
, m_width
, m_height 
); 
1086         // we'll come back later 
1088             wxapp_install_idle_handler(); 
1092     // set the focus if not done yet and if we can already do it 
1093     if ( GTK_WIDGET_REALIZED(m_wxwindow
) ) 
1095         if ( g_delayedFocus 
&& 
1096              wxGetTopLevelParent((wxWindow
*)g_delayedFocus
) == this ) 
1098             wxLogTrace(_T("focus"), 
1099                        _T("Setting focus from wxTLW::OnIdle() to %s(%s)"), 
1100                        g_delayedFocus
->GetClassInfo()->GetClassName(), 
1101                        g_delayedFocus
->GetLabel().c_str()); 
1103             g_delayedFocus
->SetFocus(); 
1104             g_delayedFocus 
= NULL
; 
1108     wxWindow::OnInternalIdle(); 
1110     // Synthetize activate events. 
1111     if ( g_sendActivateEvent 
!= -1 ) 
1113         bool activate 
= g_sendActivateEvent 
!= 0; 
1115         // if (!activate) wxPrintf( wxT("de") ); 
1116         // wxPrintf( wxT("activate\n") ); 
1119         g_sendActivateEvent 
= -1; 
1121         wxTheApp
->SetActive(activate
, (wxWindow 
*)g_lastActiveFrame
); 
1125 // ---------------------------------------------------------------------------- 
1127 // ---------------------------------------------------------------------------- 
1129 void wxTopLevelWindowGTK::SetTitle( const wxString 
&title 
) 
1131     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid frame") ); 
1133     if ( title 
== m_title 
) 
1138     gtk_window_set_title( GTK_WINDOW(m_widget
), wxGTK_CONV( title 
) ); 
1141 void wxTopLevelWindowGTK::SetIcon( const wxIcon 
&icon 
) 
1143     SetIcons( wxIconBundle( icon 
) ); 
1146 void wxTopLevelWindowGTK::SetIcons( const wxIconBundle 
&icons 
) 
1148     wxASSERT_MSG( (m_widget 
!= NULL
), wxT("invalid frame") ); 
1150     wxTopLevelWindowBase::SetIcons( icons 
); 
1154     size_t max 
= icons
.m_icons
.GetCount(); 
1156     for (size_t i 
= 0; i 
< max
; i
++) 
1158         if (icons
.m_icons
[i
].Ok()) 
1160             list 
= g_list_prepend(list
, icons
.m_icons
[i
].GetPixbuf()); 
1163     gtk_window_set_icon_list(GTK_WINDOW(m_widget
), list
); 
1166 #else // !__WXGTK20__ 
1167     GdkWindow
* window 
= m_widget
->window
; 
1171     wxIcon icon 
= icons
.GetIcon(-1); 
1174         wxMask 
*mask 
= icon
.GetMask(); 
1175         GdkBitmap 
*bm 
= (GdkBitmap 
*) NULL
; 
1176         if (mask
) bm 
= mask
->GetBitmap(); 
1178         gdk_window_set_icon( m_widget
->window
, (GdkWindow 
*) NULL
, icon
.GetPixmap(), bm 
); 
1181     wxSetIconsX11( (WXDisplay
*)GDK_WINDOW_XDISPLAY( window 
), 
1182                    (WXWindow
)GDK_WINDOW_XWINDOW( window 
), icons 
); 
1183 #endif // !__WXGTK20__ 
1186 // ---------------------------------------------------------------------------- 
1187 // frame state: maximized/iconized/normal 
1188 // ---------------------------------------------------------------------------- 
1190 void wxTopLevelWindowGTK::Maximize(bool maximize
) 
1194         gtk_window_maximize( GTK_WINDOW( m_widget 
) ); 
1196         gtk_window_unmaximize( GTK_WINDOW( m_widget 
) ); 
1198     wxFAIL_MSG( _T("not implemented") ); 
1202 bool wxTopLevelWindowGTK::IsMaximized() const 
1205     if(!m_widget
->window
) 
1208     return gdk_window_get_state(m_widget
->window
) & GDK_WINDOW_STATE_MAXIMIZED
; 
1210   //    wxFAIL_MSG( _T("not implemented") ); 
1212     // This is an approximation 
1217 void wxTopLevelWindowGTK::Restore() 
1220     // "Present" seems similar enough to "restore" 
1221     gtk_window_present( GTK_WINDOW( m_widget 
) ); 
1223     wxFAIL_MSG( _T("not implemented") ); 
1227 void wxTopLevelWindowGTK::Iconize( bool iconize 
) 
1231         gtk_window_iconify( GTK_WINDOW( m_widget 
) ); 
1233         gtk_window_deiconify( GTK_WINDOW( m_widget 
) ); 
1237        GdkWindow 
*window 
= m_widget
->window
; 
1239        // you should do it later, for example from OnCreate() handler 
1240        wxCHECK_RET( window
, _T("frame not created yet - can't iconize") ); 
1242        XIconifyWindow( GDK_WINDOW_XDISPLAY( window 
), 
1243                        GDK_WINDOW_XWINDOW( window 
), 
1244                        DefaultScreen( GDK_DISPLAY() ) ); 
1249 bool wxTopLevelWindowGTK::IsIconized() const 
1251     return m_isIconized
; 
1254 void wxTopLevelWindowGTK::SetIconizeState(bool iconize
) 
1256     if ( iconize 
!= m_isIconized 
) 
1258         m_isIconized 
= iconize
; 
1259         (void)SendIconizeEvent(iconize
); 
1263 void wxTopLevelWindowGTK::AddGrab() 
1268         gtk_grab_add( m_widget 
); 
1269         wxEventLoop().Run(); 
1270         gtk_grab_remove( m_widget 
); 
1274 void wxTopLevelWindowGTK::RemoveGrab() 
1285 static bool do_shape_combine_region(GdkWindow
* window
, const wxRegion
& region
) 
1289         if (region
.IsEmpty()) 
1291             gdk_window_shape_combine_mask(window
, NULL
, 0, 0); 
1296             gdk_window_shape_combine_region(window
, region
.GetRegion(), 0, 0); 
1298             wxBitmap bmp 
= region
.ConvertToBitmap(); 
1299             bmp
.SetMask(new wxMask(bmp
, *wxBLACK
)); 
1300             GdkBitmap
* mask 
= bmp
.GetMask()->GetBitmap(); 
1301             gdk_window_shape_combine_mask(window
, mask
, 0, 0); 
1310 bool wxTopLevelWindowGTK::SetShape(const wxRegion
& region
) 
1312     wxCHECK_MSG( HasFlag(wxFRAME_SHAPED
), false, 
1313                  _T("Shaped windows must be created with the wxFRAME_SHAPED style.")); 
1315     GdkWindow 
*window 
= NULL
; 
1318         window 
= GTK_PIZZA(m_wxwindow
)->bin_window
; 
1319         do_shape_combine_region(window
, region
); 
1321     window 
= m_widget
->window
; 
1322     return do_shape_combine_region(window
, region
); 
1325 bool wxTopLevelWindowGTK::IsActive() 
1327     return (this == (wxTopLevelWindowGTK
*)g_activeFrame
); 
1330 void wxTopLevelWindowGTK::RequestUserAttention(int flags
) 
1332     bool new_hint_value 
= false; 
1334     // FIXME: This is a workaround to focus handling problem 
1335     // If RequestUserAttention is called for example right after a wxSleep, OnInternalIdle hasn't 
1336     // yet been processed, and the internal focus system is not up to date yet. 
1337     // wxYieldIfNeeded ensures the processing of it, but can have unwanted side effects - MR 
1338     ::wxYieldIfNeeded(); 
1340     if(m_urgency_hint 
>= 0) 
1341         gtk_timeout_remove(m_urgency_hint
); 
1343     m_urgency_hint 
= -2; 
1345     if( GTK_WIDGET_REALIZED(m_widget
) && !IsActive() ) 
1347         new_hint_value 
= true; 
1349         if (flags 
& wxUSER_ATTENTION_INFO
) 
1351             m_urgency_hint 
= gtk_timeout_add(5000, (GtkFunction
)gtk_frame_urgency_timer_callback
, this); 
1353             m_urgency_hint 
= -1; 
1357 #if defined(__WXGTK20__) && GTK_CHECK_VERSION(2,7,0) 
1358     if(!gtk_check_version(2,7,0)) 
1359         gtk_window_set_urgency_hint(GTK_WINDOW( m_widget 
), new_hint_value
); 
1362         wxgtk_window_set_urgency_hint(GTK_WINDOW( m_widget 
), new_hint_value
); 
1365 void wxTopLevelWindowGTK::SetWindowStyleFlag( long style 
) 
1368     // Store which styles were changed 
1369     long styleChanges 
= style 
^ m_windowStyle
; 
1372     // Process wxWindow styles. This also updates the internal variable 
1373     // Therefore m_windowStyle bits carry now the _new_ style values 
1374     wxWindow::SetWindowStyleFlag(style
); 
1377     // just return for now if widget does not exist yet 
1382     if ( (styleChanges 
& wxSTAY_ON_TOP
) && !gtk_check_version(2,4,0) ) 
1383         gtk_window_set_keep_above(GTK_WINDOW(m_widget
), m_windowStyle 
& wxSTAY_ON_TOP
); 
1385 #if GTK_CHECK_VERSION(2,2,0) 
1386     if ( (styleChanges 
& wxFRAME_NO_TASKBAR
) && !gtk_check_version(2,2,0) ) 
1388         gtk_window_set_skip_taskbar_hint(GTK_WINDOW(m_widget
), m_windowStyle 
& wxFRAME_NO_TASKBAR
);