1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: wxGLCanvas, for using OpenGL/Mesa with wxWindows and GTK
4 // Author: Robert Roebling
8 // Copyright: (c) Robert Roebling
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "glcanvas.h"
19 #include "wx/colour.h"
20 #include "wx/module.h"
29 #include "wx/gtk/win_gtk.h"
31 //---------------------------------------------------------------------------
33 //---------------------------------------------------------------------------
35 XVisualInfo
*g_vi
= (XVisualInfo
*) NULL
;
37 //---------------------------------------------------------------------------
39 //---------------------------------------------------------------------------
41 IMPLEMENT_CLASS(wxGLContext
,wxObject
)
43 wxGLContext::wxGLContext( bool WXUNUSED(isRGB
), wxWindow
*win
, const wxPalette
& WXUNUSED(palette
) )
46 m_widget
= ((wxGLCanvas
*)win
)->m_glWidget
;
48 wxCHECK_RET( g_vi
, "invalid visual for OpenGl" );
50 m_glContext
= glXCreateContext( GDK_DISPLAY(), g_vi
, None
, GL_TRUE
);
52 wxCHECK_RET( m_glContext
, "Couldn't create OpenGl context" );
55 wxGLContext::wxGLContext(
56 bool WXUNUSED(isRGB
), wxWindow
*win
,
57 const wxPalette
& WXUNUSED(palette
),
58 const wxGLContext
*other
/* for sharing display lists */
62 m_widget
= ((wxGLCanvas
*)win
)->m_glWidget
;
64 wxCHECK_RET( g_vi
, "invalid visual for OpenGl" );
67 m_glContext
= glXCreateContext( GDK_DISPLAY(), g_vi
, other
->m_glContext
,
70 m_glContext
= glXCreateContext( GDK_DISPLAY(), g_vi
, None
, GL_TRUE
);
72 wxCHECK_RET( m_glContext
, "Couldn't create OpenGl context" );
75 wxGLContext::~wxGLContext()
77 if (!m_glContext
) return;
79 if (m_glContext
== glXGetCurrentContext())
81 glXMakeCurrent( GDK_DISPLAY(), None
, NULL
);
84 glXDestroyContext( GDK_DISPLAY(), m_glContext
);
87 void wxGLContext::SwapBuffers()
91 GdkWindow
*window
= GTK_MYFIXED(m_widget
)->bin_window
;
92 glXSwapBuffers( GDK_DISPLAY(), GDK_WINDOW_XWINDOW( window
) );
96 void wxGLContext::SetCurrent()
100 GdkWindow
*window
= GTK_MYFIXED(m_widget
)->bin_window
;
101 glXMakeCurrent( GDK_DISPLAY(), GDK_WINDOW_XWINDOW(window
), m_glContext
);
105 void wxGLContext::SetColour(const char *colour
)
110 wxColour
*col
= wxTheColourDatabase
->FindColour(colour
);
113 r
= (float)(col
->Red()/256.0);
114 g
= (float)(col
->Green()/256.0);
115 b
= (float)(col
->Blue()/256.0);
120 void wxGLContext::SetupPixelFormat()
124 void wxGLContext::SetupPalette( const wxPalette
& WXUNUSED(palette
) )
128 wxPalette
wxGLContext::CreateDefaultPalette()
130 return wxNullPalette
;
133 //-----------------------------------------------------------------------------
134 // "realize" from m_wxwindow
135 //-----------------------------------------------------------------------------
138 gtk_glwindow_realized_callback( GtkWidget
* WXUNUSED(widget
), wxGLCanvas
*win
)
140 win
->m_glContext
= new wxGLContext( TRUE
, win
, wxNullPalette
, win
->m_sharedContext
);
143 g_vi
= (XVisualInfo
*) NULL
;
148 //-----------------------------------------------------------------------------
149 // "expose_event" of m_wxwindow
150 //-----------------------------------------------------------------------------
153 gtk_glwindow_expose_callback( GtkWidget
*WXUNUSED(widget
), GdkEventExpose
*gdk_event
, wxGLCanvas
*win
)
155 win
->m_exposed
= TRUE
;
157 win
->GetUpdateRegion().Union( gdk_event
->area
.x
,
159 gdk_event
->area
.width
,
160 gdk_event
->area
.height
);
163 //-----------------------------------------------------------------------------
164 // "draw" of m_wxwindow
165 //-----------------------------------------------------------------------------
168 gtk_glwindow_draw_callback( GtkWidget
*WXUNUSED(widget
), GdkRectangle
*rect
, wxGLCanvas
*win
)
170 win
->m_exposed
= TRUE
;
172 win
->GetUpdateRegion().Union( rect
->x
, rect
->y
,
173 rect
->width
, rect
->height
);
176 //---------------------------------------------------------------------------
178 //---------------------------------------------------------------------------
180 IMPLEMENT_CLASS(wxGLCanvas
, wxScrolledWindow
)
182 BEGIN_EVENT_TABLE(wxGLCanvas
, wxScrolledWindow
)
183 EVT_SIZE(wxGLCanvas::OnSize
)
186 wxGLCanvas::wxGLCanvas( wxWindow
*parent
, wxWindowID id
,
187 const wxPoint
& pos
, const wxSize
& size
,
188 long style
, const wxString
& name
,
190 const wxPalette
& palette
)
192 Create( parent
, NULL
, id
, pos
, size
, style
, name
, attribList
, palette
);
195 wxGLCanvas::wxGLCanvas( wxWindow
*parent
,
196 const wxGLContext
*shared
,
198 const wxPoint
& pos
, const wxSize
& size
,
199 long style
, const wxString
& name
,
201 const wxPalette
& palette
)
203 Create( parent
, shared
, id
, pos
, size
, style
, name
, attribList
, palette
);
206 bool wxGLCanvas::Create( wxWindow
*parent
,
207 const wxGLContext
*shared
,
209 const wxPoint
& pos
, const wxSize
& size
,
210 long style
, const wxString
& name
,
212 const wxPalette
& palette
)
214 m_sharedContext
= (wxGLContext
*)shared
; // const_cast
221 int data
[] = { GLX_RGBA
,
223 GLX_DEPTH_SIZE
, 1, // use largest available depth buffer
229 attribList
= (int*) data
;
233 int data
[512], arg
=0, p
=0;
235 while( (attribList
[arg
]!=0) && (p
<512) )
237 switch( attribList
[arg
++] )
239 case WX_GL_RGBA
: data
[p
++] = GLX_RGBA
; break;
240 case WX_GL_DOUBLEBUFFER
: data
[p
++] = GLX_DOUBLEBUFFER
; break;
241 case WX_GL_DEPTH_SIZE
:
242 data
[p
++]=GLX_DEPTH_SIZE
; data
[p
++]=attribList
[arg
++]; break;
244 data
[p
++]=GLX_RED_SIZE
; data
[p
++]=attribList
[arg
++]; break;
245 case WX_GL_MIN_GREEN
:
246 data
[p
++]=GLX_GREEN_SIZE
; data
[p
++]=attribList
[arg
++]; break;
248 data
[p
++]=GLX_BLUE_SIZE
; data
[p
++]=attribList
[arg
++]; break;
255 attribList
= (int*) data
;
259 Display
*dpy
= GDK_DISPLAY();
261 g_vi
= glXChooseVisual( dpy
, DefaultScreen(dpy
), attribList
);
263 wxCHECK_MSG( g_vi
, FALSE
, "required visual couldn't be found" );
265 GdkVisual
*visual
= gdkx_visual_get( g_vi
->visualid
);
266 GdkColormap
*colormap
= gdk_colormap_new( gdkx_visual_get(g_vi
->visualid
), TRUE
);
268 gtk_widget_push_colormap( colormap
);
269 gtk_widget_push_visual( visual
);
271 wxScrolledWindow::Create( parent
, id
, pos
, size
, style
, name
);
273 m_glWidget
= m_wxwindow
;
275 gtk_my_fixed_set_clear( GTK_MYFIXED(m_wxwindow
), FALSE
);
277 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "realize",
278 GTK_SIGNAL_FUNC(gtk_glwindow_realized_callback
), (gpointer
) this );
280 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "expose_event",
281 GTK_SIGNAL_FUNC(gtk_glwindow_expose_callback
), (gpointer
)this );
283 gtk_signal_connect( GTK_OBJECT(m_wxwindow
), "draw",
284 GTK_SIGNAL_FUNC(gtk_glwindow_draw_callback
), (gpointer
)this );
286 gtk_widget_pop_visual();
287 gtk_widget_pop_colormap();
292 wxGLCanvas::~wxGLCanvas()
294 if (m_glContext
) delete m_glContext
;
297 void wxGLCanvas::SwapBuffers()
299 if (m_glContext
) m_glContext
->SwapBuffers();
302 void wxGLCanvas::OnSize(wxSizeEvent
& WXUNUSED(event
))
305 GetClientSize( &width
, &height
);
307 if (m_glContext
&& GTK_WIDGET_REALIZED(m_glWidget
) )
311 glViewport(0, 0, (GLint
)width
, (GLint
)height
);
312 glMatrixMode(GL_PROJECTION
);
314 glFrustum( -1.0, 1.0, -1.0, 1.0, 5.0, 15.0 );
315 glMatrixMode(GL_MODELVIEW
);
319 void wxGLCanvas::SetCurrent()
321 if (m_glContext
) m_glContext
->SetCurrent();
324 void wxGLCanvas::SetColour( const char *colour
)
326 if (m_glContext
) m_glContext
->SetColour( colour
);
329 void wxGLCanvas::OnInternalIdle()
331 if (m_glContext
&& m_exposed
)
333 wxPaintEvent
event( GetId() );
334 event
.SetEventObject( this );
335 GetEventHandler()->ProcessEvent( event
);
338 GetUpdateRegion().Clear();