// +build !windows,!darwin // 7 july 2014 package ui import ( "unsafe" "fmt" ) // #include "gtk_unix.h" // extern gboolean windowClosing(GtkWidget *, GdkEvent *, gpointer); // extern void windowResizing(GtkWidget *, GdkRectangle *, gpointer); import "C" type window struct { widget *C.GtkWidget wc *C.GtkContainer bin *C.GtkBin window *C.GtkWindow layoutw *C.GtkWidget layoutc *C.GtkContainer layout *C.GtkLayout closing *event *container } func newWindow(title string, width int, height int, control Control) *window { widget := C.gtk_window_new(C.GTK_WINDOW_TOPLEVEL) ctitle := togstr(title) defer freegstr(ctitle) layoutw := C.gtk_layout_new(nil, nil) w := &window{ widget: widget, wc: (*C.GtkContainer)(unsafe.Pointer(widget)), bin: (*C.GtkBin)(unsafe.Pointer(widget)), window: (*C.GtkWindow)(unsafe.Pointer(widget)), layoutw: layoutw, layoutc: (*C.GtkContainer)(unsafe.Pointer(layoutw)), layout: (*C.GtkLayout)(unsafe.Pointer(layoutw)), closing: newEvent(), container: new(container), } C.gtk_window_set_title(w.window, ctitle) g_signal_connect( C.gpointer(unsafe.Pointer(w.window)), "delete-event", C.GCallback(C.windowClosing), C.gpointer(unsafe.Pointer(w))) // we connect to the layout's size-allocate, not to the window's configure-event // this allows us to handle client-side decoration-based configurations (such as GTK+ on Wayland) properly // also see commitResize() in sizing_unix.go for additional notes // thanks to many people in irc.gimp.net/#gtk+ for help (including tristan for suggesting g_signal_connect_after()) g_signal_connect_after( C.gpointer(unsafe.Pointer(layoutw)), "size-allocate", C.GCallback(C.windowResizing), C.gpointer(unsafe.Pointer(w))) C.gtk_window_resize(w.window, C.gint(width), C.gint(height)) C.gtk_container_add(w.wc, layoutw) w.child = control w.child.setParent(&controlParent{w.layoutc}) return w } func (w *window) Title() string { return fromgstr(C.gtk_window_get_title(w.window)) } func (w *window) SetTitle(title string) { ctitle := togstr(title) defer freegstr(ctitle) C.gtk_window_set_title(w.window, ctitle) } func (w *window) Show() { C.gtk_widget_show_all(w.widget) } func (w *window) Hide() { C.gtk_widget_hide(w.widget) } func (w *window) Close() { C.gtk_widget_destroy(w.widget) } func (w *window) OnClosing(e func() bool) { w.closing.setbool(e) } //export windowClosing func windowClosing(wid *C.GtkWidget, e *C.GdkEvent, data C.gpointer) C.gboolean { w := (*window)(unsafe.Pointer(data)) close := w.closing.fire() if close { return C.GDK_EVENT_PROPAGATE // will do gtk_widget_destroy(), which is what we want (thanks ebassi in irc.gimp.net/#gtk+) } return C.GDK_EVENT_STOP // keeps window alive } //export windowResizing func windowResizing(wid *C.GtkWidget, r *C.GdkRectangle, data C.gpointer) { w := (*window)(unsafe.Pointer(data)) // the origin of the window's content area is always (0, 0) w.resize(0, 0, int(r.width), int(r.height)) fmt.Printf("new size %d x %d\n", r.width, r.height) }