///////////////////////////////////////////////////////////////////////////// // Name: src/gtk1/glcanvas.cpp // Purpose: wxGLCanvas, for using OpenGL/Mesa with wxWidgets and GTK // Author: Robert Roebling // Modified by: // Created: 17/08/98 // Copyright: (c) Robert Roebling // Licence: wxWindows licence ///////////////////////////////////////////////////////////////////////////// // For compilers that support precompilation, includes "wx.h". #include "wx/wxprec.h" #if wxUSE_GLCANVAS #include "wx/glcanvas.h" #ifndef WX_PRECOMP #include "wx/app.h" #include "wx/frame.h" #include "wx/colour.h" #include "wx/module.h" #endif // WX_PRECOMP extern "C" { #include "gtk/gtk.h" #include "gdk/gdk.h" #include "gdk/gdkx.h" } #include "wx/gtk1/win_gtk.h" #include "wx/gtk1/private.h" //----------------------------------------------------------------------------- // idle system //----------------------------------------------------------------------------- extern void wxapp_install_idle_handler(); extern bool g_isIdle; #if WXWIN_COMPATIBILITY_2_8 //----------------------------------------------------------------------------- // "realize" from m_wxwindow: used to create m_glContext implicitly //----------------------------------------------------------------------------- extern "C" { static gint gtk_glwindow_realized_callback( GtkWidget *WXUNUSED(widget), wxGLCanvas *win ) { win->GTKInitImplicitContext(); return FALSE; } } #endif // WXWIN_COMPATIBILITY_2_8 //----------------------------------------------------------------------------- // "map" from m_wxwindow //----------------------------------------------------------------------------- extern "C" { static gint gtk_glwindow_map_callback( GtkWidget * WXUNUSED(widget), wxGLCanvas *win ) { wxPaintEvent event( win->GetId() ); event.SetEventObject( win ); win->HandleWindowEvent( event ); win->GetUpdateRegion().Clear(); return FALSE; } } //----------------------------------------------------------------------------- // "expose_event" of m_wxwindow //----------------------------------------------------------------------------- extern "C" { static void gtk_glwindow_expose_callback( GtkWidget *WXUNUSED(widget), GdkEventExpose *gdk_event, wxGLCanvas *win ) { if (g_isIdle) wxapp_install_idle_handler(); win->GetUpdateRegion().Union( gdk_event->area.x, gdk_event->area.y, gdk_event->area.width, gdk_event->area.height ); } } //----------------------------------------------------------------------------- // "draw" of m_wxwindow //----------------------------------------------------------------------------- extern "C" { static void gtk_glwindow_draw_callback( GtkWidget *WXUNUSED(widget), GdkRectangle *rect, wxGLCanvas *win ) { if (g_isIdle) wxapp_install_idle_handler(); win->GetUpdateRegion().Union( rect->x, rect->y, rect->width, rect->height ); } } //----------------------------------------------------------------------------- // "size_allocate" of m_wxwindow //----------------------------------------------------------------------------- extern "C" { static void gtk_glcanvas_size_callback( GtkWidget *WXUNUSED(widget), GtkAllocation* alloc, wxGLCanvas *win ) { if (g_isIdle) wxapp_install_idle_handler(); if (!win->m_hasVMT) return; wxSizeEvent event( wxSize(win->m_width,win->m_height), win->GetId() ); event.SetEventObject( win ); win->HandleWindowEvent( event ); } } //--------------------------------------------------------------------------- // wxGlCanvas //--------------------------------------------------------------------------- wxIMPLEMENT_CLASS(wxGLCanvas, wxWindow); wxGLCanvas::wxGLCanvas(wxWindow *parent, const wxGLAttributes& dispAttrs, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name, const wxPalette& palette) #if WXWIN_COMPATIBILITY_2_8 : m_createImplicitContext(false) #endif { Create(parent, dispAttrs, id, pos, size, style, name, palette); } wxGLCanvas::wxGLCanvas(wxWindow *parent, wxWindowID id, const int *attribList, const wxPoint& pos, const wxSize& size, long style, const wxString& name, const wxPalette& palette) #if WXWIN_COMPATIBILITY_2_8 : m_createImplicitContext(false) #endif { Create(parent, id, pos, size, style, name, attribList, palette); } #if WXWIN_COMPATIBILITY_2_8 wxGLCanvas::wxGLCanvas(wxWindow *parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name, const int *attribList, const wxPalette& palette) : m_createImplicitContext(true) { Create(parent, id, pos, size, style, name, attribList, palette); } wxGLCanvas::wxGLCanvas(wxWindow *parent, const wxGLContext *shared, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name, const int *attribList, const wxPalette& palette) : m_createImplicitContext(true) { m_sharedContext = const_cast(shared); Create(parent, id, pos, size, style, name, attribList, palette); } wxGLCanvas::wxGLCanvas(wxWindow *parent, const wxGLCanvas *shared, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name, const int *attribList, const wxPalette& palette ) : m_createImplicitContext(true) { m_sharedContextOf = const_cast(shared); Create(parent, id, pos, size, style, name, attribList, palette); } #endif // WXWIN_COMPATIBILITY_2_8 bool wxGLCanvas::Create(wxWindow *parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name, const int *attribList, const wxPalette& palette) { // Separate 'GLXFBConfig/XVisual' attributes. // Also store context attributes for wxGLContext ctor wxGLAttributes dispAttrs; if ( ! ParseAttribList(attribList, dispAttrs, &m_GLCTXAttrs) ) return false; return Create(parent, dispAttrs, id, pos, size, style, name, palette); } bool wxGLCanvas::Create(wxWindow *parent, const wxGLAttributes& dispAttrs, wxWindowID id, const wxPoint& pos, const wxSize& size, long style, const wxString& name, const wxPalette& palette) { m_noExpose = true; m_nativeSizeEvent = true; if ( !InitVisual(dispAttrs) ) return false; GdkVisual *visual = gdkx_visual_get( GetXVisualInfo()->visualid ); GdkColormap *colormap = gdk_colormap_new( visual, TRUE ); gtk_widget_push_colormap( colormap ); gtk_widget_push_visual( visual ); wxWindow::Create( parent, id, pos, size, style, name ); m_glWidget = m_wxwindow; gtk_pizza_set_clear( GTK_PIZZA(m_wxwindow), FALSE ); #if WXWIN_COMPATIBILITY_2_8 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "realize", GTK_SIGNAL_FUNC(gtk_glwindow_realized_callback), (gpointer) this); #endif // WXWIN_COMPATIBILITY_2_8 gtk_signal_connect( GTK_OBJECT(m_wxwindow), "map", GTK_SIGNAL_FUNC(gtk_glwindow_map_callback), (gpointer) this); gtk_signal_connect( GTK_OBJECT(m_wxwindow), "expose_event", GTK_SIGNAL_FUNC(gtk_glwindow_expose_callback), (gpointer) this); gtk_signal_connect( GTK_OBJECT(m_wxwindow), "draw", GTK_SIGNAL_FUNC(gtk_glwindow_draw_callback), (gpointer) this); gtk_signal_connect( GTK_OBJECT(m_widget), "size_allocate", GTK_SIGNAL_FUNC(gtk_glcanvas_size_callback), (gpointer) this); gtk_widget_pop_visual(); gtk_widget_pop_colormap(); #if WXWIN_COMPATIBILITY_2_8 // if our parent window is already visible, we had been realized before we // connected to the "realize" signal and hence our m_glContext hasn't been // initialized yet and we have to do it now if (GTK_WIDGET_REALIZED(m_wxwindow)) gtk_glwindow_realized_callback( m_wxwindow, this ); #endif // WXWIN_COMPATIBILITY_2_8 if (GTK_WIDGET_MAPPED(m_wxwindow)) gtk_glwindow_map_callback( m_wxwindow, this ); return true; } Window wxGLCanvas::GetXWindow() const { GdkWindow *window = GTK_PIZZA(m_wxwindow)->bin_window; return window ? GDK_WINDOW_XWINDOW(window) : 0; } void wxGLCanvas::OnInternalIdle() { if (!m_updateRegion.IsEmpty()) { wxPaintEvent event( GetId() ); event.SetEventObject( this ); HandleWindowEvent( event ); GetUpdateRegion().Clear(); } wxWindow::OnInternalIdle(); } #if WXWIN_COMPATIBILITY_2_8 void wxGLCanvas::GTKInitImplicitContext() { if ( !m_glContext && m_createImplicitContext ) { wxGLContext *share = m_sharedContext; if ( !share && m_sharedContextOf ) share = m_sharedContextOf->m_glContext; m_glContext = new wxGLContext(this, share); } } #endif // WXWIN_COMPATIBILITY_2_8 #endif // wxUSE_GLCANVAS