]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/gtk/glcanvas.cpp
Fix crash when drawing bitmaps with mask in wxGTK with GTK+ < 2.20.
[wxWidgets.git] / src / gtk / glcanvas.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/gtk/glcanvas.cpp
3// Purpose: wxGLCanvas, for using OpenGL/Mesa with wxWidgets and GTK
4// Author: Robert Roebling
5// Modified by:
6// Created: 17/08/98
7// RCS-ID: $Id$
8// Copyright: (c) Robert Roebling
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#if wxUSE_GLCANVAS
16
17#include "wx/glcanvas.h"
18
19#include <gtk/gtk.h>
20#include <gdk/gdkx.h>
21#include "wx/gtk/private/gtk2-compat.h"
22
23#if WXWIN_COMPATIBILITY_2_8
24
25//-----------------------------------------------------------------------------
26// "realize" from m_wxwindow: used to create m_glContext implicitly
27//-----------------------------------------------------------------------------
28
29extern "C" {
30static void
31gtk_glwindow_realized_callback( GtkWidget *WXUNUSED(widget), wxGLCanvas *win )
32{
33 win->GTKInitImplicitContext();
34}
35}
36
37#endif // WXWIN_COMPATIBILITY_2_8
38
39//-----------------------------------------------------------------------------
40// "map" from m_wxwindow
41//-----------------------------------------------------------------------------
42
43#ifndef __WXGTK3__
44extern "C" {
45static void
46gtk_glwindow_map_callback( GtkWidget * WXUNUSED(widget), wxGLCanvas *win )
47{
48 wxPaintEvent event( win->GetId() );
49 event.SetEventObject( win );
50 win->HandleWindowEvent( event );
51
52 win->m_exposed = false;
53 win->GetUpdateRegion().Clear();
54}
55}
56#endif
57
58//-----------------------------------------------------------------------------
59// "expose_event" of m_wxwindow
60//-----------------------------------------------------------------------------
61
62extern "C" {
63#ifdef __WXGTK3__
64static gboolean draw(GtkWidget*, cairo_t* cr, wxGLCanvas* win)
65{
66 win->m_exposed = true;
67 if (win->m_cairoPaintContext == NULL)
68 {
69 win->m_cairoPaintContext = cr;
70 cairo_reference(cr);
71 }
72 double x1, y1, x2, y2;
73 cairo_clip_extents(cr, &x1, &y1, &x2, &y2);
74 win->GetUpdateRegion().Union(int(x1), int(y1), int(x2 - x1), int(y2 - y1));
75 return false;
76}
77#else
78static gboolean
79gtk_glwindow_expose_callback( GtkWidget *WXUNUSED(widget), GdkEventExpose *gdk_event, wxGLCanvas *win )
80{
81 win->m_exposed = true;
82
83 win->GetUpdateRegion().Union( gdk_event->area.x,
84 gdk_event->area.y,
85 gdk_event->area.width,
86 gdk_event->area.height );
87 return false;
88}
89#endif
90}
91
92//-----------------------------------------------------------------------------
93// "size_allocate" of m_wxwindow
94//-----------------------------------------------------------------------------
95
96extern "C" {
97static void
98gtk_glcanvas_size_callback(GtkWidget *WXUNUSED(widget),
99 GtkAllocation * WXUNUSED(alloc),
100 wxGLCanvas *win)
101{
102 wxSizeEvent event( wxSize(win->m_width,win->m_height), win->GetId() );
103 event.SetEventObject( win );
104 win->HandleWindowEvent( event );
105}
106}
107
108//-----------------------------------------------------------------------------
109// emission hook for "parent-set"
110//-----------------------------------------------------------------------------
111
112extern "C" {
113static gboolean
114parent_set_hook(GSignalInvocationHint*, guint, const GValue* param_values, void* data)
115{
116 wxGLCanvas* win = (wxGLCanvas*)data;
117 if (g_value_peek_pointer(&param_values[0]) == win->m_wxwindow)
118 {
119 const XVisualInfo* xvi = win->GetXVisualInfo();
120 GdkVisual* visual = gtk_widget_get_visual(win->m_wxwindow);
121 if (GDK_VISUAL_XVISUAL(visual)->visualid != xvi->visualid)
122 {
123 GdkScreen* screen = gtk_widget_get_screen(win->m_wxwindow);
124 visual = gdk_x11_screen_lookup_visual(screen, xvi->visualid);
125#ifdef __WXGTK3__
126 gtk_widget_set_visual(win->m_wxwindow, visual);
127#else
128 GdkColormap* colormap = gdk_colormap_new(visual, false);
129 gtk_widget_set_colormap(win->m_wxwindow, colormap);
130 g_object_unref(colormap);
131#endif
132 }
133 // remove hook
134 return false;
135 }
136 return true;
137}
138}
139
140//---------------------------------------------------------------------------
141// wxGlCanvas
142//---------------------------------------------------------------------------
143
144IMPLEMENT_CLASS(wxGLCanvas, wxWindow)
145
146wxGLCanvas::wxGLCanvas(wxWindow *parent,
147 wxWindowID id,
148 const int *attribList,
149 const wxPoint& pos,
150 const wxSize& size,
151 long style,
152 const wxString& name,
153 const wxPalette& palette)
154#if WXWIN_COMPATIBILITY_2_8
155 : m_createImplicitContext(false)
156#endif
157{
158 Create(parent, id, pos, size, style, name, attribList, palette);
159}
160
161#if WXWIN_COMPATIBILITY_2_8
162
163wxGLCanvas::wxGLCanvas(wxWindow *parent,
164 wxWindowID id,
165 const wxPoint& pos,
166 const wxSize& size,
167 long style,
168 const wxString& name,
169 const int *attribList,
170 const wxPalette& palette)
171 : m_createImplicitContext(true)
172{
173 m_sharedContext = NULL;
174 m_sharedContextOf = NULL;
175
176 Create(parent, id, pos, size, style, name, attribList, palette);
177}
178
179wxGLCanvas::wxGLCanvas(wxWindow *parent,
180 const wxGLContext *shared,
181 wxWindowID id,
182 const wxPoint& pos,
183 const wxSize& size,
184 long style,
185 const wxString& name,
186 const int *attribList,
187 const wxPalette& palette)
188 : m_createImplicitContext(true)
189{
190 m_sharedContext = const_cast<wxGLContext *>(shared);
191
192 Create(parent, id, pos, size, style, name, attribList, palette);
193}
194
195wxGLCanvas::wxGLCanvas(wxWindow *parent,
196 const wxGLCanvas *shared,
197 wxWindowID id,
198 const wxPoint& pos, const wxSize& size,
199 long style, const wxString& name,
200 const int *attribList,
201 const wxPalette& palette )
202 : m_createImplicitContext(true)
203{
204 m_sharedContext = NULL;
205 m_sharedContextOf = const_cast<wxGLCanvas *>(shared);
206
207 Create(parent, id, pos, size, style, name, attribList, palette);
208}
209
210#endif // WXWIN_COMPATIBILITY_2_8
211
212bool wxGLCanvas::Create(wxWindow *parent,
213 wxWindowID id,
214 const wxPoint& pos,
215 const wxSize& size,
216 long style,
217 const wxString& name,
218 const int *attribList,
219 const wxPalette& palette)
220{
221#if wxUSE_PALETTE
222 wxASSERT_MSG( !palette.IsOk(), wxT("palettes not supported") );
223#endif // wxUSE_PALETTE
224 wxUnusedVar(palette); // Unused when wxDEBUG_LEVEL==0
225
226 m_exposed = false;
227 m_noExpose = true;
228 m_nativeSizeEvent = true;
229#ifdef __WXGTK3__
230 m_cairoPaintContext = NULL;
231 m_backgroundStyle = wxBG_STYLE_PAINT;
232#endif
233
234 if ( !InitVisual(attribList) )
235 return false;
236
237 // watch for the "parent-set" signal on m_wxwindow so we can set colormap
238 // before m_wxwindow is realized (which will occur before
239 // wxWindow::Create() returns if parent is already visible)
240 unsigned sig_id = g_signal_lookup("parent-set", GTK_TYPE_WIDGET);
241 g_signal_add_emission_hook(sig_id, 0, parent_set_hook, this, NULL);
242
243 wxWindow::Create( parent, id, pos, size, style, name );
244
245 gtk_widget_set_double_buffered(m_wxwindow, false);
246
247#if WXWIN_COMPATIBILITY_2_8
248 g_signal_connect(m_wxwindow, "realize", G_CALLBACK(gtk_glwindow_realized_callback), this);
249#endif // WXWIN_COMPATIBILITY_2_8
250#ifdef __WXGTK3__
251 g_signal_connect(m_wxwindow, "draw", G_CALLBACK(draw), this);
252#else
253 g_signal_connect(m_wxwindow, "map", G_CALLBACK(gtk_glwindow_map_callback), this);
254 g_signal_connect(m_wxwindow, "expose_event", G_CALLBACK(gtk_glwindow_expose_callback), this);
255#endif
256 g_signal_connect(m_widget, "size_allocate", G_CALLBACK(gtk_glcanvas_size_callback), this);
257
258#if WXWIN_COMPATIBILITY_2_8
259 // if our parent window is already visible, we had been realized before we
260 // connected to the "realize" signal and hence our m_glContext hasn't been
261 // initialized yet and we have to do it now
262 if (gtk_widget_get_realized(m_wxwindow))
263 gtk_glwindow_realized_callback( m_wxwindow, this );
264#endif // WXWIN_COMPATIBILITY_2_8
265
266#ifndef __WXGTK3__
267 if (gtk_widget_get_mapped(m_wxwindow))
268 gtk_glwindow_map_callback( m_wxwindow, this );
269#endif
270
271 return true;
272}
273
274bool wxGLCanvas::SetBackgroundStyle(wxBackgroundStyle /* style */)
275{
276 return false;
277}
278
279Window wxGLCanvas::GetXWindow() const
280{
281 GdkWindow* window = GTKGetDrawingWindow();
282 return window ? GDK_WINDOW_XID(window) : 0;
283}
284
285void wxGLCanvas::OnInternalIdle()
286{
287 if (m_exposed)
288 {
289#ifdef __WXGTK3__
290 GTKSendPaintEvents(m_cairoPaintContext);
291 cairo_destroy(m_cairoPaintContext);
292 m_cairoPaintContext = NULL;
293#else
294 wxPaintEvent event( GetId() );
295 event.SetEventObject( this );
296 HandleWindowEvent( event );
297#endif
298
299 m_exposed = false;
300 GetUpdateRegion().Clear();
301 }
302
303 wxWindow::OnInternalIdle();
304}
305
306#if WXWIN_COMPATIBILITY_2_8
307
308void wxGLCanvas::GTKInitImplicitContext()
309{
310 if ( !m_glContext && m_createImplicitContext )
311 {
312 wxGLContext *share = m_sharedContext;
313 if ( !share && m_sharedContextOf )
314 share = m_sharedContextOf->m_glContext;
315
316 m_glContext = new wxGLContext(this, share);
317 }
318}
319
320#endif // WXWIN_COMPATIBILITY_2_8
321
322#endif // wxUSE_GLCANVAS