1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/gtk/glcanvas.cpp
3 // Purpose: wxGLCanvas, for using OpenGL/Mesa with wxWidgets and GTK
4 // Author: Robert Roebling
8 // Copyright: (c) Robert Roebling
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // For compilers that support precompilation, includes "wx.h".
13 #include "wx/wxprec.h"
17 #include "wx/glcanvas.h"
22 #include "wx/colour.h"
23 #include "wx/module.h"
33 #include "wx/gtk/win_gtk.h"
34 #include "wx/gtk/private.h"
36 //---------------------------------------------------------------------------
38 //---------------------------------------------------------------------------
40 XVisualInfo
*g_vi
= (XVisualInfo
*) NULL
;
42 // ----------------------------------------------------------------------------
44 // ----------------------------------------------------------------------------
46 // wrapper around glXMakeContextCurrent/glXMakeCurrent depending on GLX
48 static void wxMakeContextCurrent(GLXDrawable drawable
, GLXContext context
)
50 if (wxGLCanvas::GetGLXVersion() >= 13)
51 glXMakeContextCurrent( GDK_DISPLAY(), drawable
, drawable
, context
);
52 else // GLX <= 1.2 doesn't have glXMakeContextCurrent()
53 glXMakeCurrent( GDK_DISPLAY(), drawable
, context
);
56 //---------------------------------------------------------------------------
58 //---------------------------------------------------------------------------
60 IMPLEMENT_CLASS(wxGLContext
,wxObject
)
62 wxGLContext::wxGLContext(wxWindow
* win
, const wxGLContext
* other
)
64 wxGLCanvas
*gc
= (wxGLCanvas
*) win
;
66 if (wxGLCanvas::GetGLXVersion() >= 13)
68 GLXFBConfig
*fbc
= gc
->m_fbc
;
69 wxCHECK_RET( fbc
, _T("invalid GLXFBConfig for OpenGl") );
70 m_glContext
= glXCreateNewContext( GDK_DISPLAY(), fbc
[0], GLX_RGBA_TYPE
,
71 other
? other
->m_glContext
: None
,
76 XVisualInfo
*vi
= (XVisualInfo
*) gc
->m_vi
;
77 wxCHECK_RET( vi
, _T("invalid visual for OpenGl") );
78 m_glContext
= glXCreateContext( GDK_DISPLAY(), vi
,
79 other
? other
->m_glContext
: None
,
83 wxASSERT_MSG( m_glContext
, _T("Couldn't create OpenGl context") );
86 wxGLContext::~wxGLContext()
91 if ( m_glContext
== glXGetCurrentContext() )
92 wxMakeContextCurrent(None
, NULL
);
94 glXDestroyContext( GDK_DISPLAY(), m_glContext
);
97 void wxGLContext::SetCurrent(const wxGLCanvas
& win
) const
102 GdkWindow
*window
= GTK_PIZZA(win
.m_wxwindow
)->bin_window
;
103 wxCHECK_RET( window
, _T("window must be shown") );
105 wxMakeContextCurrent(GDK_WINDOW_XWINDOW(window
), m_glContext
);
109 #if WXWIN_COMPATIBILITY_2_8
111 //-----------------------------------------------------------------------------
112 // "realize" from m_wxwindow: used to create m_glContext implicitly
113 //-----------------------------------------------------------------------------
117 gtk_glwindow_realized_callback( GtkWidget
*WXUNUSED(widget
), wxGLCanvas
*win
)
119 win
->GTKInitImplicitContext();
125 #endif // WXWIN_COMPATIBILITY_2_8
127 //-----------------------------------------------------------------------------
128 // "map" from m_wxwindow
129 //-----------------------------------------------------------------------------
133 gtk_glwindow_map_callback( GtkWidget
* WXUNUSED(widget
), wxGLCanvas
*win
)
135 wxPaintEvent
event( win
->GetId() );
136 event
.SetEventObject( win
);
137 win
->GetEventHandler()->ProcessEvent( event
);
139 win
->m_exposed
= false;
140 win
->GetUpdateRegion().Clear();
146 //-----------------------------------------------------------------------------
147 // "expose_event" of m_wxwindow
148 //-----------------------------------------------------------------------------
152 gtk_glwindow_expose_callback( GtkWidget
*WXUNUSED(widget
), GdkEventExpose
*gdk_event
, wxGLCanvas
*win
)
154 // don't need to install idle handler, its done from "event" signal
156 win
->m_exposed
= true;
158 win
->GetUpdateRegion().Union( gdk_event
->area
.x
,
160 gdk_event
->area
.width
,
161 gdk_event
->area
.height
);
166 //-----------------------------------------------------------------------------
167 // "size_allocate" of m_wxwindow
168 //-----------------------------------------------------------------------------
172 gtk_glcanvas_size_callback( GtkWidget
*WXUNUSED(widget
), GtkAllocation
* alloc
, wxGLCanvas
*win
)
175 wxapp_install_idle_handler();
180 wxSizeEvent
event( wxSize(win
->m_width
,win
->m_height
), win
->GetId() );
181 event
.SetEventObject( win
);
182 win
->GetEventHandler()->ProcessEvent( event
);
186 //---------------------------------------------------------------------------
188 //---------------------------------------------------------------------------
190 IMPLEMENT_CLASS(wxGLCanvas
, wxWindow
)
192 wxGLCanvas::wxGLCanvas(wxWindow
*parent
,
194 const int *attribList
,
198 const wxString
& name
,
199 const wxPalette
& palette
)
200 #if WXWIN_COMPATIBILITY_2_8
201 : m_createImplicitContext(false)
204 Create(parent
, id
, pos
, size
, style
, name
, attribList
, palette
);
207 #if WXWIN_COMPATIBILITY_2_8
209 wxGLCanvas::wxGLCanvas(wxWindow
*parent
,
214 const wxString
& name
,
215 const int *attribList
,
216 const wxPalette
& palette
)
217 : m_createImplicitContext(true)
219 Create(parent
, id
, pos
, size
, style
, name
, attribList
, palette
);
222 wxGLCanvas::wxGLCanvas(wxWindow
*parent
,
223 const wxGLContext
*shared
,
228 const wxString
& name
,
229 const int *attribList
,
230 const wxPalette
& palette
)
231 : m_createImplicitContext(true)
233 m_sharedContext
= wx_const_cast(wxGLContext
*, shared
);
235 Create(parent
, id
, pos
, size
, style
, name
, attribList
, palette
);
238 wxGLCanvas::wxGLCanvas(wxWindow
*parent
,
239 const wxGLCanvas
*shared
,
241 const wxPoint
& pos
, const wxSize
& size
,
242 long style
, const wxString
& name
,
243 const int *attribList
,
244 const wxPalette
& palette
)
245 : m_createImplicitContext(true)
247 m_sharedContextOf
= wx_const_cast(wxGLCanvas
*, shared
);
249 Create(parent
, id
, pos
, size
, style
, name
, attribList
, palette
);
252 #endif // WXWIN_COMPATIBILITY_2_8
254 bool wxGLCanvas::Create(wxWindow
*parent
,
259 const wxString
& name
,
260 const int *attribList
,
261 const wxPalette
& palette
)
265 m_nativeSizeEvent
= true;
269 if (wxGLCanvas::GetGLXVersion() >= 13)
271 // GLX >= 1.3 uses a GLXFBConfig
272 GLXFBConfig
* fbc
= NULL
;
273 if (wxTheApp
->m_glFBCInfo
!= NULL
)
275 fbc
= (GLXFBConfig
*) wxTheApp
->m_glFBCInfo
;
276 m_canFreeFBC
= false; // owned by wxTheApp - don't free upon destruction
280 fbc
= (GLXFBConfig
*) wxGLCanvas::ChooseGLFBC(attribList
);
283 m_fbc
= fbc
; // save for later use
284 wxCHECK_MSG( m_fbc
, false, _T("required FBConfig couldn't be found") );
287 XVisualInfo
*vi
= NULL
;
288 if (wxTheApp
->m_glVisualInfo
!= NULL
)
290 vi
= (XVisualInfo
*)wxTheApp
->m_glVisualInfo
;
291 m_canFreeVi
= false; // owned by wxTheApp - don't free upon destruction
295 if (wxGLCanvas::GetGLXVersion() >= 13)
297 vi
= glXGetVisualFromFBConfig(GDK_DISPLAY(), m_fbc
[0]);
300 vi
= (XVisualInfo
*) ChooseGLVisual(attribList
);
305 m_vi
= vi
; // save for later use
307 wxCHECK_MSG( m_vi
, false, _T("required visual couldn't be found") );
309 GdkColormap
*colormap
;
311 // MR: This needs a fix for lower gtk+ versions too. Might need to rethink logic (FIXME)
312 #if defined(__WXGTK20__) && GTK_CHECK_VERSION(2,2,0)
313 if (!gtk_check_version(2,2,0))
315 wxWindow::Create( parent
, id
, pos
, size
, style
, name
);
317 m_glWidget
= m_wxwindow
;
319 GdkScreen
*screen
= gtk_widget_get_screen( m_glWidget
);
320 colormap
= gdk_screen_get_default_colormap(screen
);
321 visual
= gdk_colormap_get_visual(colormap
);
323 if (GDK_VISUAL_XVISUAL(visual
)->visualid
!= vi
->visualid
)
325 visual
= gdk_x11_screen_lookup_visual( screen
, vi
->visualid
);
326 colormap
= gdk_colormap_new(visual
, FALSE
);
329 gtk_widget_set_colormap( m_glWidget
, colormap
);
332 #endif // GTK+ >= 2.2
334 visual
= gdkx_visual_get( vi
->visualid
);
335 colormap
= gdk_colormap_new( visual
, TRUE
);
337 gtk_widget_push_colormap( colormap
);
339 wxWindow::Create( parent
, id
, pos
, size
, style
, name
);
340 m_glWidget
= m_wxwindow
;
343 gtk_widget_set_double_buffered( m_glWidget
, FALSE
);
345 #if WXWIN_COMPATIBILITY_2_8
346 g_signal_connect(m_wxwindow
, "realize", G_CALLBACK(gtk_glwindow_realized_callback
), this);
347 #endif // WXWIN_COMPATIBILITY_2_8
348 g_signal_connect(m_wxwindow
, "map", G_CALLBACK(gtk_glwindow_map_callback
), this);
349 g_signal_connect(m_wxwindow
, "expose_event", G_CALLBACK(gtk_glwindow_expose_callback
), this);
350 g_signal_connect(m_widget
, "size_allocate", G_CALLBACK(gtk_glcanvas_size_callback
), this);
352 if (gtk_check_version(2,2,0) != NULL
)
354 gtk_widget_pop_colormap();
357 #if WXWIN_COMPATIBILITY_2_8
358 // if our parent window is already visible, we had been realized before we
359 // connected to the "realize" signal and hence our m_glContext hasn't been
360 // initialized yet and we have to do it now
361 if (GTK_WIDGET_REALIZED(m_wxwindow
))
362 gtk_glwindow_realized_callback( m_wxwindow
, this );
363 #endif // WXWIN_COMPATIBILITY_2_8
365 if (GTK_WIDGET_MAPPED(m_wxwindow
))
366 gtk_glwindow_map_callback( m_wxwindow
, this );
371 wxGLCanvas::~wxGLCanvas()
373 GLXFBConfig
* fbc
= (GLXFBConfig
*) m_fbc
;
374 if (fbc
&& m_canFreeFBC
)
377 XVisualInfo
*vi
= (XVisualInfo
*) m_vi
;
378 if (vi
&& m_canFreeVi
)
382 void* wxGLCanvas::ChooseGLVisual(const int *attribList
)
385 GetGLAttribListFromWX( attribList
, data
);
387 Display
*dpy
= GDK_DISPLAY();
389 return glXChooseVisual( dpy
, DefaultScreen(dpy
), data
);
392 void* wxGLCanvas::ChooseGLFBC(const int *attribList
)
395 GetGLAttribListFromWX( attribList
, data
);
398 return glXChooseFBConfig( GDK_DISPLAY(), DefaultScreen(GDK_DISPLAY()),
404 wxGLCanvas::GetGLAttribListFromWX(const int *wx_attribList
, int *gl_attribList
)
406 if ( !wx_attribList
)
408 if (wxGLCanvas::GetGLXVersion() >= 13)
410 // leave GLX >= 1.3 choose the default attributes
411 gl_attribList
[0] = 0;
416 // default settings if attriblist = 0
417 gl_attribList
[i
++] = GLX_RGBA
;
418 gl_attribList
[i
++] = GLX_DOUBLEBUFFER
;
419 gl_attribList
[i
++] = GLX_DEPTH_SIZE
; gl_attribList
[i
++] = 1;
420 gl_attribList
[i
++] = GLX_RED_SIZE
; gl_attribList
[i
++] = 1;
421 gl_attribList
[i
++] = GLX_GREEN_SIZE
; gl_attribList
[i
++] = 1;
422 gl_attribList
[i
++] = GLX_BLUE_SIZE
; gl_attribList
[i
++] = 1;
423 gl_attribList
[i
++] = GLX_ALPHA_SIZE
; gl_attribList
[i
++] = 0;
424 gl_attribList
[i
++] = None
;
427 else // have non-default attributes
430 while( (wx_attribList
[arg
]!=0) && (p
<510) )
432 switch( wx_attribList
[arg
++] )
435 if (wxGLCanvas::GetGLXVersion() <= 12)
437 // for GLX >= 1.3, GLX_RGBA is useless (setting this flags will crash on most opengl implm)
438 gl_attribList
[p
++] = GLX_RGBA
;
441 case WX_GL_BUFFER_SIZE
:
442 gl_attribList
[p
++] = GLX_BUFFER_SIZE
;
443 gl_attribList
[p
++] = wx_attribList
[arg
++];
446 gl_attribList
[p
++] = GLX_LEVEL
;
447 gl_attribList
[p
++] = wx_attribList
[arg
++];
449 case WX_GL_DOUBLEBUFFER
:
450 gl_attribList
[p
++] = GLX_DOUBLEBUFFER
;
451 gl_attribList
[p
++] = 1;
454 gl_attribList
[p
++] = GLX_STEREO
;
456 case WX_GL_AUX_BUFFERS
:
457 gl_attribList
[p
++] = GLX_AUX_BUFFERS
;
458 gl_attribList
[p
++] = wx_attribList
[arg
++];
461 gl_attribList
[p
++] = GLX_RED_SIZE
;
462 gl_attribList
[p
++] = wx_attribList
[arg
++];
464 case WX_GL_MIN_GREEN
:
465 gl_attribList
[p
++] = GLX_GREEN_SIZE
;
466 gl_attribList
[p
++] = wx_attribList
[arg
++];
469 gl_attribList
[p
++] = GLX_BLUE_SIZE
;
470 gl_attribList
[p
++] = wx_attribList
[arg
++];
472 case WX_GL_MIN_ALPHA
:
473 gl_attribList
[p
++] = GLX_ALPHA_SIZE
;
474 gl_attribList
[p
++] = wx_attribList
[arg
++];
476 case WX_GL_DEPTH_SIZE
:
477 gl_attribList
[p
++] = GLX_DEPTH_SIZE
;
478 gl_attribList
[p
++] = wx_attribList
[arg
++];
480 case WX_GL_STENCIL_SIZE
:
481 gl_attribList
[p
++] = GLX_STENCIL_SIZE
;
482 gl_attribList
[p
++] = wx_attribList
[arg
++];
484 case WX_GL_MIN_ACCUM_RED
:
485 gl_attribList
[p
++] = GLX_ACCUM_RED_SIZE
;
486 gl_attribList
[p
++] = wx_attribList
[arg
++];
488 case WX_GL_MIN_ACCUM_GREEN
:
489 gl_attribList
[p
++] = GLX_ACCUM_GREEN_SIZE
;
490 gl_attribList
[p
++] = wx_attribList
[arg
++];
492 case WX_GL_MIN_ACCUM_BLUE
:
493 gl_attribList
[p
++] = GLX_ACCUM_BLUE_SIZE
;
494 gl_attribList
[p
++] = wx_attribList
[arg
++];
496 case WX_GL_MIN_ACCUM_ALPHA
:
497 gl_attribList
[p
++] = GLX_ACCUM_ALPHA_SIZE
;
498 gl_attribList
[p
++] = wx_attribList
[arg
++];
505 gl_attribList
[p
] = 0;
510 int wxGLCanvas::GetGLXVersion()
512 static int s_glxVersion
= 0;
513 if ( s_glxVersion
== 0 )
515 // check the GLX version
516 int glxMajorVer
, glxMinorVer
;
517 bool ok
= glXQueryVersion(GDK_DISPLAY(), &glxMajorVer
, &glxMinorVer
);
518 wxASSERT_MSG( ok
, _T("GLX version not found") );
520 s_glxVersion
= 10; // 1.0 by default
522 s_glxVersion
= glxMajorVer
*10 + glxMinorVer
;
528 void wxGLCanvas::SwapBuffers()
530 GdkWindow
*window
= GTK_PIZZA(m_wxwindow
)->bin_window
;
531 glXSwapBuffers( GDK_DISPLAY(), GDK_WINDOW_XWINDOW( window
) );
534 void wxGLCanvas::OnInternalIdle()
538 wxPaintEvent
event( GetId() );
539 event
.SetEventObject( this );
540 GetEventHandler()->ProcessEvent( event
);
543 GetUpdateRegion().Clear();
546 wxWindow::OnInternalIdle();
549 #if WXWIN_COMPATIBILITY_2_8
551 void wxGLCanvas::GTKInitImplicitContext()
553 if ( !m_glContext
&& m_createImplicitContext
)
555 wxGLContext
*share
= m_sharedContext
;
556 if ( !share
&& m_sharedContextOf
)
557 share
= m_sharedContextOf
->m_glContext
;
559 m_glContext
= new wxGLContext(this, share
);
563 #endif // WXWIN_COMPATIBILITY_2_8
565 //---------------------------------------------------------------------------
567 //---------------------------------------------------------------------------
569 bool wxGLApp::InitGLVisual(const int *attribList
)
571 if ( wxGLCanvas::GetGLXVersion() >= 13 )
575 m_glFBCInfo
= wxGLCanvas::ChooseGLFBC(attribList
);
581 XFree(m_glVisualInfo
);
582 m_glVisualInfo
= glXGetVisualFromFBConfig(GDK_DISPLAY(), ((GLXFBConfig
*)m_glFBCInfo
)[0]);
587 XFree(m_glVisualInfo
);
588 m_glVisualInfo
= wxGLCanvas::ChooseGLVisual(attribList
);
591 return m_glVisualInfo
!= NULL
;
594 #endif // wxUSE_GLCANVAS