Fix crash in wxGTK wxPopupWindow when creating it without parent.
[wxWidgets.git] / src / gtk / glcanvas.cpp
CommitLineData
8b089c5e 1/////////////////////////////////////////////////////////////////////////////
9b5f1895 2// Name: src/gtk/glcanvas.cpp
77ffb593 3// Purpose: wxGLCanvas, for using OpenGL/Mesa with wxWidgets and GTK
8b089c5e
JS
4// Author: Robert Roebling
5// Modified by:
6// Created: 17/08/98
7// RCS-ID: $Id$
8// Copyright: (c) Robert Roebling
65571936 9// Licence: wxWindows licence
8b089c5e
JS
10/////////////////////////////////////////////////////////////////////////////
11
14f355c2
VS
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
8b089c5e
JS
15#if wxUSE_GLCANVAS
16
17#include "wx/glcanvas.h"
18
a1abca32
PC
19#include <gtk/gtk.h>
20#include <gdk/gdkx.h>
86b8b3ac 21#include "wx/gtk/private/gtk2-compat.h"
8b089c5e 22
dc3065a5
VZ
23#if WXWIN_COMPATIBILITY_2_8
24
8b089c5e 25//-----------------------------------------------------------------------------
dc3065a5 26// "realize" from m_wxwindow: used to create m_glContext implicitly
8b089c5e
JS
27//-----------------------------------------------------------------------------
28
865bb325 29extern "C" {
144ae5bb 30static void
34a34b02 31gtk_glwindow_realized_callback( GtkWidget *WXUNUSED(widget), wxGLCanvas *win )
8b089c5e 32{
dc3065a5 33 win->GTKInitImplicitContext();
8b089c5e 34}
865bb325 35}
8b089c5e 36
dc3065a5
VZ
37#endif // WXWIN_COMPATIBILITY_2_8
38
8b089c5e
JS
39//-----------------------------------------------------------------------------
40// "map" from m_wxwindow
41//-----------------------------------------------------------------------------
42
9dc44eff 43#ifndef __WXGTK3__
865bb325 44extern "C" {
144ae5bb 45static void
8b089c5e
JS
46gtk_glwindow_map_callback( GtkWidget * WXUNUSED(widget), wxGLCanvas *win )
47{
dc3065a5
VZ
48 wxPaintEvent event( win->GetId() );
49 event.SetEventObject( win );
937013e0 50 win->HandleWindowEvent( event );
8b089c5e 51
dc3065a5
VZ
52 win->m_exposed = false;
53 win->GetUpdateRegion().Clear();
8b089c5e 54}
865bb325 55}
9dc44eff 56#endif
8b089c5e
JS
57
58//-----------------------------------------------------------------------------
59// "expose_event" of m_wxwindow
60//-----------------------------------------------------------------------------
61
865bb325 62extern "C" {
9dc44eff
PC
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
6d727f6c 78static gboolean
8b089c5e
JS
79gtk_glwindow_expose_callback( GtkWidget *WXUNUSED(widget), GdkEventExpose *gdk_event, wxGLCanvas *win )
80{
670f9935 81 win->m_exposed = true;
8b089c5e
JS
82
83 win->GetUpdateRegion().Union( gdk_event->area.x,
84 gdk_event->area.y,
85 gdk_event->area.width,
86 gdk_event->area.height );
6d727f6c 87 return false;
8b089c5e 88}
9dc44eff 89#endif
865bb325 90}
8b089c5e 91
8b089c5e
JS
92//-----------------------------------------------------------------------------
93// "size_allocate" of m_wxwindow
94//-----------------------------------------------------------------------------
95
865bb325 96extern "C" {
2b5f62a0 97static void
e0d1fd7f
VZ
98gtk_glcanvas_size_callback(GtkWidget *WXUNUSED(widget),
99 GtkAllocation * WXUNUSED(alloc),
100 wxGLCanvas *win)
8b089c5e 101{
8b089c5e
JS
102 wxSizeEvent event( wxSize(win->m_width,win->m_height), win->GetId() );
103 event.SetEventObject( win );
937013e0 104 win->HandleWindowEvent( event );
8b089c5e 105}
865bb325 106}
8b089c5e 107
3780e252
PC
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);
9dc44eff
PC
125#ifdef __WXGTK3__
126 gtk_widget_set_visual(win->m_wxwindow, visual);
127#else
3780e252
PC
128 GdkColormap* colormap = gdk_colormap_new(visual, false);
129 gtk_widget_set_colormap(win->m_wxwindow, colormap);
130 g_object_unref(colormap);
9dc44eff 131#endif
3780e252
PC
132 }
133 // remove hook
134 return false;
135 }
136 return true;
137}
138}
139
8b089c5e
JS
140//---------------------------------------------------------------------------
141// wxGlCanvas
142//---------------------------------------------------------------------------
143
4660d7e5 144IMPLEMENT_CLASS(wxGLCanvas, wxWindow)
8b089c5e 145
dc3065a5
VZ
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
b7ea712c 155 : m_createImplicitContext(false)
dc3065a5 156#endif
b7ea712c 157{
dc3065a5 158 Create(parent, id, pos, size, style, name, attribList, palette);
b7ea712c
RR
159}
160
dc3065a5
VZ
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)
b7ea712c 171 : m_createImplicitContext(true)
8b089c5e 172{
21562ad2
VZ
173 m_sharedContext = NULL;
174 m_sharedContextOf = NULL;
175
dc3065a5 176 Create(parent, id, pos, size, style, name, attribList, palette);
8b089c5e
JS
177}
178
dc3065a5
VZ
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)
b7ea712c 188 : m_createImplicitContext(true)
2b5f62a0 189{
5c33522f 190 m_sharedContext = const_cast<wxGLContext *>(shared);
dc3065a5
VZ
191
192 Create(parent, id, pos, size, style, name, attribList, palette);
8b089c5e
JS
193}
194
dc3065a5
VZ
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 )
b7ea712c 202 : m_createImplicitContext(true)
2b5f62a0 203{
21562ad2 204 m_sharedContext = NULL;
5c33522f 205 m_sharedContextOf = const_cast<wxGLCanvas *>(shared);
dc3065a5
VZ
206
207 Create(parent, id, pos, size, style, name, attribList, palette);
8b089c5e
JS
208}
209
dc3065a5 210#endif // WXWIN_COMPATIBILITY_2_8
2b5f62a0 211
dc3065a5
VZ
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,
9ce5cf09 219 const wxPalette& palette)
dc3065a5 220{
9ce5cf09 221#if wxUSE_PALETTE
9a83f860 222 wxASSERT_MSG( !palette.IsOk(), wxT("palettes not supported") );
9ce5cf09
VZ
223#endif // wxUSE_PALETTE
224 wxUnusedVar(palette); // Unused when wxDEBUG_LEVEL==0
e0d1fd7f 225
670f9935
WS
226 m_exposed = false;
227 m_noExpose = true;
228 m_nativeSizeEvent = true;
9dc44eff
PC
229#ifdef __WXGTK3__
230 m_cairoPaintContext = NULL;
231 m_backgroundStyle = wxBG_STYLE_PAINT;
232#endif
34a34b02 233
498ace9e
VZ
234 if ( !InitVisual(attribList) )
235 return false;
2b5f62a0 236
3780e252
PC
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);
2b5f62a0 242
ff654490 243 wxWindow::Create( parent, id, pos, size, style, name );
fee7a683 244
144ae5bb 245 gtk_widget_set_double_buffered(m_wxwindow, false);
2b5b9325 246
dc3065a5 247#if WXWIN_COMPATIBILITY_2_8
b7ea712c 248 g_signal_connect(m_wxwindow, "realize", G_CALLBACK(gtk_glwindow_realized_callback), this);
dc3065a5 249#endif // WXWIN_COMPATIBILITY_2_8
9dc44eff
PC
250#ifdef __WXGTK3__
251 g_signal_connect(m_wxwindow, "draw", G_CALLBACK(draw), this);
252#else
b7ea712c
RR
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);
9dc44eff 255#endif
b7ea712c 256 g_signal_connect(m_widget, "size_allocate", G_CALLBACK(gtk_glcanvas_size_callback), this);
a6f5aa49 257
dc3065a5 258#if WXWIN_COMPATIBILITY_2_8
bc869971
VZ
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
fc9ab22a 262 if (gtk_widget_get_realized(m_wxwindow))
bc869971 263 gtk_glwindow_realized_callback( m_wxwindow, this );
dc3065a5 264#endif // WXWIN_COMPATIBILITY_2_8
bc869971 265
9dc44eff 266#ifndef __WXGTK3__
fc9ab22a 267 if (gtk_widget_get_mapped(m_wxwindow))
bc869971 268 gtk_glwindow_map_callback( m_wxwindow, this );
9dc44eff 269#endif
bc869971 270
670f9935 271 return true;
a6f5aa49
VZ
272}
273
9dc44eff
PC
274bool wxGLCanvas::SetBackgroundStyle(wxBackgroundStyle /* style */)
275{
276 return false;
277}
278
498ace9e 279Window wxGLCanvas::GetXWindow() const
8b089c5e 280{
f089940f 281 GdkWindow* window = GTKGetDrawingWindow();
9dc44eff 282 return window ? GDK_WINDOW_XID(window) : 0;
8b089c5e
JS
283}
284
8b089c5e
JS
285void wxGLCanvas::OnInternalIdle()
286{
dc3065a5 287 if (m_exposed)
8b089c5e 288 {
9dc44eff
PC
289#ifdef __WXGTK3__
290 GTKSendPaintEvents(m_cairoPaintContext);
291 cairo_destroy(m_cairoPaintContext);
292 m_cairoPaintContext = NULL;
293#else
8b089c5e
JS
294 wxPaintEvent event( GetId() );
295 event.SetEventObject( this );
937013e0 296 HandleWindowEvent( event );
9dc44eff 297#endif
8b089c5e 298
670f9935 299 m_exposed = false;
8b089c5e
JS
300 GetUpdateRegion().Clear();
301 }
2b5f62a0 302
8b089c5e
JS
303 wxWindow::OnInternalIdle();
304}
305
dc3065a5
VZ
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}
a6f5aa49 319
dc3065a5 320#endif // WXWIN_COMPATIBILITY_2_8
a6f5aa49 321
dc3065a5 322#endif // wxUSE_GLCANVAS