#if defined(Hiro_Window) namespace hiro { static auto Window_close(GtkWidget* widget, GdkEvent* event, pWindow* p) -> signed { if(p->state().onClose) { p->self().doClose(); } else { p->self().setVisible(false); } if(p->self().modal() && !p->self().visible()) p->self().setModal(false); return true; } //GTK3 draw: called into by GTK2 expose-event static auto Window_draw(GtkWidget* widget, cairo_t* context, pWindow* p) -> signed { if(auto color = p->state().backgroundColor) { double red = (double)color.red() / 255.0; double green = (double)color.green() / 255.0; double blue = (double)color.blue() / 255.0; double alpha = (double)color.alpha() / 255.0; if(gdk_screen_is_composited(gdk_screen_get_default()) && gdk_screen_get_rgba_visual(gdk_screen_get_default()) ) { cairo_set_source_rgba(context, red, green, blue, alpha); } else { cairo_set_source_rgb(context, red, green, blue); } cairo_set_operator(context, CAIRO_OPERATOR_SOURCE); cairo_paint(context); } else { #if HIRO_GTK==3 auto style = gtk_widget_get_style_context(widget); GtkAllocation allocation; gtk_widget_get_allocation(widget, &allocation); gtk_render_background(style, context, 0, 0, allocation.width, allocation.height); #endif } return false; } //GTK2 expose-event static auto Window_expose(GtkWidget* widget, GdkEvent* event, pWindow* p) -> signed { cairo_t* context = gdk_cairo_create(gtk_widget_get_window(widget)); Window_draw(widget, context, p); cairo_destroy(context); return false; } static auto Window_configure(GtkWidget* widget, GdkEvent* event, pWindow* p) -> signed { p->_synchronizeMargin(); return false; } static auto Window_drop(GtkWidget* widget, GdkDragContext* context, signed x, signed y, GtkSelectionData* data, unsigned type, unsigned timestamp, pWindow* p) -> void { if(!p->state().droppable) return; auto paths = DropPaths(data); if(!paths) return; p->self().doDrop(paths); } static auto Window_getPreferredWidth(GtkWidget* widget, int* minimalWidth, int* naturalWidth) -> void { if(auto p = (pWindow*)g_object_get_data(G_OBJECT(widget), "hiro::window")) { *minimalWidth = 1; *naturalWidth = p->state().geometry.width(); } } static auto Window_getPreferredHeight(GtkWidget* widget, int* minimalHeight, int* naturalHeight) -> void { if(auto p = (pWindow*)g_object_get_data(G_OBJECT(widget), "hiro::window")) { *minimalHeight = 1; *naturalHeight = p->state().geometry.height(); } } static auto Window_keyPress(GtkWidget* widget, GdkEventKey* event, pWindow* p) -> signed { if(auto key = pKeyboard::_translate(event->keyval)) { p->self().doKeyPress(key); } if(p->state().dismissable && event->keyval == GDK_KEY_Escape) { if(p->state().onClose) { p->self().doClose(); } else { p->self().setVisible(false); } if(p->state().modal && !p->pObject::state().visible) p->self().setModal(false); } return false; } static auto Window_keyRelease(GtkWidget* widget, GdkEventKey* event, pWindow* p) -> signed { if(auto key = pKeyboard::_translate(event->keyval)) { p->self().doKeyRelease(key); } return false; } static auto Window_realize(GtkWidget* widget, pWindow* p) -> void { } static auto Window_sizeAllocate(GtkWidget* widget, GtkAllocation* allocation, pWindow* p) -> void { p->_synchronizeState(); p->_synchronizeGeometry(); p->_synchronizeMargin(); return; } static auto Window_sizeRequest(GtkWidget* widget, GtkRequisition* requisition, pWindow* p) -> void { requisition->width = p->state().geometry.width(); requisition->height = p->state().geometry.height(); } static auto Window_stateEvent(GtkWidget* widget, GdkEvent* event, pWindow* p) -> void { p->_synchronizeState(); if(event->type == GDK_WINDOW_STATE) { auto windowStateEvent = (GdkEventWindowState*)event; if(windowStateEvent->changed_mask & GDK_WINDOW_STATE_MAXIMIZED) { p->state().maximized = windowStateEvent->new_window_state & GDK_WINDOW_STATE_MAXIMIZED; } if(windowStateEvent->changed_mask & GDK_WINDOW_STATE_ICONIFIED) { p->state().minimized = windowStateEvent->new_window_state & GDK_WINDOW_STATE_ICONIFIED; } } } static auto Window_unrealize(GtkWidget* widget, pWindow* p) -> void { } auto pWindow::construct() -> void { widget = gtk_window_new(GTK_WINDOW_TOPLEVEL); gtk_window_set_resizable(GTK_WINDOW(widget), true); //if program was given a name, try and set the window taskbar icon from one of the pixmaps folders if(!Application::state().name); else if(_setIcon({Path::user(), ".local/share/icons/"})); else if(_setIcon("/usr/local/share/pixmaps/")); else if(_setIcon("/usr/share/pixmaps/")); auto visual = gdk_screen_get_rgba_visual(gdk_screen_get_default()); if(!visual) visual = gdk_screen_get_system_visual(gdk_screen_get_default()); if(visual) gtk_widget_set_visual(widget, visual); gtk_widget_set_app_paintable(widget, true); gtk_widget_add_events(widget, GDK_CONFIGURE); #if HIRO_GTK==2 menuContainer = gtk_vbox_new(false, 0); #elif HIRO_GTK==3 menuContainer = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0); #endif gtk_container_add(GTK_CONTAINER(widget), menuContainer); gtk_widget_show(menuContainer); gtkMenu = gtk_menu_bar_new(); gtk_box_pack_start(GTK_BOX(menuContainer), gtkMenu, false, false, 0); formContainer = gtk_fixed_new(); gtk_box_pack_start(GTK_BOX(menuContainer), formContainer, true, true, 0); gtk_widget_show(formContainer); statusContainer = gtk_event_box_new(); gtkStatus = gtk_statusbar_new(); #if HIRO_GTK==2 gtk_statusbar_set_has_resize_grip(GTK_STATUSBAR(gtkStatus), true); #elif HIRO_GTK==3 gtk_window_set_has_resize_grip(GTK_WINDOW(widget), true); #endif gtk_container_add(GTK_CONTAINER(statusContainer), gtkStatus); gtk_box_pack_start(GTK_BOX(menuContainer), statusContainer, false, false, 0); gtk_widget_show(statusContainer); setBackgroundColor(state().backgroundColor); setDroppable(state().droppable); setGeometry(state().geometry); setResizable(state().resizable); setMaximized(state().maximized); setMinimized(state().minimized); setTitle(state().title); g_signal_connect(G_OBJECT(widget), "delete-event", G_CALLBACK(Window_close), (gpointer)this); #if HIRO_GTK==2 g_signal_connect(G_OBJECT(widget), "expose-event", G_CALLBACK(Window_expose), (gpointer)this); #elif HIRO_GTK==3 g_signal_connect(G_OBJECT(widget), "draw", G_CALLBACK(Window_draw), (gpointer)this); #endif g_signal_connect(G_OBJECT(widget), "configure-event", G_CALLBACK(Window_configure), (gpointer)this); g_signal_connect(G_OBJECT(widget), "drag-data-received", G_CALLBACK(Window_drop), (gpointer)this); g_signal_connect(G_OBJECT(widget), "key-press-event", G_CALLBACK(Window_keyPress), (gpointer)this); g_signal_connect(G_OBJECT(widget), "key-release-event", G_CALLBACK(Window_keyRelease), (gpointer)this); g_signal_connect(G_OBJECT(widget), "realize", G_CALLBACK(Window_realize), (gpointer)this); g_signal_connect(G_OBJECT(formContainer), "size-allocate", G_CALLBACK(Window_sizeAllocate), (gpointer)this); #if HIRO_GTK==2 g_signal_connect(G_OBJECT(formContainer), "size-request", G_CALLBACK(Window_sizeRequest), (gpointer)this); #elif HIRO_GTK==3 auto widgetClass = GTK_WIDGET_GET_CLASS(formContainer); widgetClass->get_preferred_width = Window_getPreferredWidth; widgetClass->get_preferred_height = Window_getPreferredHeight; #endif g_signal_connect(G_OBJECT(widget), "unrealize", G_CALLBACK(Window_unrealize), (gpointer)this); g_signal_connect(G_OBJECT(widget), "window-state-event", G_CALLBACK(Window_stateEvent), (gpointer)this); g_object_set_data(G_OBJECT(widget), "hiro::window", (gpointer)this); g_object_set_data(G_OBJECT(formContainer), "hiro::window", (gpointer)this); pApplication::state().windows.append(this); } auto pWindow::destruct() -> void { for(uint offset : range(pApplication::state().windows.size())) { if(pApplication::state().windows[offset] == this) { pApplication::state().windows.remove(offset); break; } } gtk_widget_destroy(widget); } auto pWindow::append(sMenuBar menuBar) -> void { _setMenuEnabled(menuBar->enabled(true)); _setMenuFont(menuBar->font(true)); _setMenuVisible(menuBar->visible(true)); } auto pWindow::append(sSizable sizable) -> void { } auto pWindow::append(sStatusBar statusBar) -> void { _setStatusEnabled(statusBar->enabled(true)); _setStatusFont(statusBar->font(true)); _setStatusText(statusBar->text()); _setStatusVisible(statusBar->visible(true)); } auto pWindow::focused() const -> bool { return gtk_window_is_active(GTK_WINDOW(widget)); } auto pWindow::frameMargin() const -> Geometry { if(state().fullScreen) return { 0, _menuHeight(), 0, _menuHeight() + _statusHeight() }; return { settings.geometry.frameX, settings.geometry.frameY + _menuHeight(), settings.geometry.frameWidth, settings.geometry.frameHeight + _menuHeight() + _statusHeight() }; } auto pWindow::handle() const -> uintptr { #if defined(DISPLAY_WINDOWS) return (uintptr)GDK_WINDOW_HWND(gtk_widget_get_window(widget)); #endif #if defined(DISPLAY_XORG) return GDK_WINDOW_XID(gtk_widget_get_window(widget)); #endif return (uintptr)nullptr; } auto pWindow::monitor() const -> uint { if(!gtk_widget_get_realized(widget)) return 0; auto window = gtk_widget_get_window(widget); return gdk_screen_get_monitor_at_window(gdk_screen_get_default(), window); } auto pWindow::remove(sMenuBar menuBar) -> void { _setMenuVisible(false); } auto pWindow::remove(sSizable sizable) -> void { } auto pWindow::remove(sStatusBar statusBar) -> void { _setStatusVisible(false); } auto pWindow::setBackgroundColor(Color color) -> void { GdkColor gdkColor = CreateColor(color); gtk_widget_modify_bg(widget, GTK_STATE_NORMAL, color ? &gdkColor : nullptr); } auto pWindow::setDismissable(bool dismissable) -> void { } auto pWindow::setDroppable(bool droppable) -> void { gtk_drag_dest_set(widget, GTK_DEST_DEFAULT_ALL, nullptr, 0, GDK_ACTION_COPY); if(droppable) gtk_drag_dest_add_uri_targets(widget); } auto pWindow::setEnabled(bool enabled) -> void { if(auto& menuBar = state().menuBar) { if(auto self = menuBar->self()) self->setEnabled(menuBar->enabled(true)); } if(auto& statusBar = state().statusBar) { if(auto self = statusBar->self()) self->setEnabled(statusBar->enabled(true)); } } auto pWindow::setFocused() -> void { gtk_window_present(GTK_WINDOW(widget)); } auto pWindow::setFullScreen(bool fullScreen) -> void { if(fullScreen) { gtk_window_fullscreen(GTK_WINDOW(widget)); } else { gtk_window_unfullscreen(GTK_WINDOW(widget)); } auto time = chrono::millisecond(); while(chrono::millisecond() - time < 20) Application::processEvents(); } auto pWindow::setGeometry(Geometry geometry) -> void { auto margin = frameMargin(); gtk_window_move(GTK_WINDOW(widget), geometry.x() - margin.x(), geometry.y() - margin.y()); setMaximumSize(state().maximumSize); setMinimumSize(state().minimumSize); auto time1 = chrono::millisecond(); while(chrono::millisecond() - time1 < 20) Application::processEvents(); gtk_window_resize(GTK_WINDOW(widget), geometry.width(), geometry.height() + _menuHeight() + _statusHeight()); auto time2 = chrono::millisecond(); while(chrono::millisecond() - time2 < 20) Application::processEvents(); } auto pWindow::setMaximized(bool maximized) -> void { auto lock = acquire(); if(maximized) { gtk_window_maximize(GTK_WINDOW(widget)); } else { gtk_window_unmaximize(GTK_WINDOW(widget)); } } auto pWindow::setMaximumSize(Size size) -> void { if(!state().resizable) size = state().geometry.size(); //TODO: this doesn't have any effect in GTK2 or GTK3 GdkGeometry geometry; if(size.height()) size.setHeight(size.height() + _menuHeight() + _statusHeight()); geometry.max_width = !state().resizable ? state().geometry.width() : size.width() ? size.width() : 32767; geometry.max_height = !state().resizable ? state().geometry.height() : size.height() ? size.height() : 32767; gtk_window_set_geometry_hints(GTK_WINDOW(widget), nullptr, &geometry, GDK_HINT_MAX_SIZE); } auto pWindow::setMinimized(bool minimized) -> void { auto lock = acquire(); if(minimized) { gtk_window_iconify(GTK_WINDOW(widget)); } else { gtk_window_deiconify(GTK_WINDOW(widget)); } } auto pWindow::setMinimumSize(Size size) -> void { if(!state().resizable) size = state().geometry.size(); //for GTK3 gtk_widget_set_size_request(formContainer, size.width(), size.height()); //for GTK2 GdkGeometry geometry; if(size.height()) size.setHeight(size.height() + _menuHeight() + _statusHeight()); geometry.min_width = !state().resizable ? state().geometry.width() : size.width() ? size.width() : 1; geometry.min_height = !state().resizable ? state().geometry.height() : size.height() ? size.height() : 1; gtk_window_set_geometry_hints(GTK_WINDOW(widget), nullptr, &geometry, GDK_HINT_MIN_SIZE); //for GTK2 } auto pWindow::setModal(bool modal) -> void { if(modal) { gtk_window_set_modal(GTK_WINDOW(widget), true); while(!Application::state().quit && state().modal) { if(Application::state().onMain) { Application::doMain(); } else { usleep(20 * 1000); } Application::processEvents(); } gtk_window_set_modal(GTK_WINDOW(widget), false); } } auto pWindow::setResizable(bool resizable) -> void { gtk_window_set_resizable(GTK_WINDOW(widget), resizable); #if HIRO_GTK==2 gtk_statusbar_set_has_resize_grip(GTK_STATUSBAR(gtkStatus), resizable); #elif HIRO_GTK==3 bool statusBarVisible = false; if(auto statusBar = state().statusBar) statusBarVisible = statusBar->visible(); gtk_window_set_has_resize_grip(GTK_WINDOW(widget), resizable && statusBarVisible); #endif setMaximumSize(state().maximumSize); setMinimumSize(state().minimumSize); } auto pWindow::setTitle(const string& title) -> void { gtk_window_set_title(GTK_WINDOW(widget), title ? title : " "); } auto pWindow::setVisible(bool visible) -> void { gtk_widget_set_visible(widget, visible); _synchronizeGeometry(); _synchronizeMargin(); } auto pWindow::_append(mWidget& widget) -> void { if(auto pWidget = widget.self()) { if(auto parent = widget.parentWidget(true)) { if(auto instance = parent->self()) { pWidget->gtkParent = instance->container(widget); } } else { pWidget->gtkParent = formContainer; } if(pWidget->gtkParent) { gtk_fixed_put(GTK_FIXED(pWidget->gtkParent), pWidget->gtkWidget, 0, 0); } } } auto pWindow::_append(mMenu& menu) -> void { if(auto pMenu = menu.self()) { gtk_menu_shell_append(GTK_MENU_SHELL(gtkMenu), pMenu->widget); } } auto pWindow::_menuHeight() const -> int { if(auto& menuBar = state().menuBar) { if(menuBar->visible()) { return settings.geometry.menuHeight + _menuTextHeight(); } } return 0; } auto pWindow::_menuTextHeight() const -> int { int height = 0; if(auto& menuBar = state().menuBar) { for(auto& menu : menuBar->state.menus) { height = max(height, menu->font(true).size(menu->text()).height()); } } return height; } auto pWindow::_setIcon(const string& pathname) -> bool { string filename; filename = {pathname, Application::state().name, ".svg"}; if(file::exists(filename)) { gtk_window_set_icon_from_file(GTK_WINDOW(widget), filename, nullptr); return true; } filename = {pathname, Application::state().name, ".png"}; if(file::exists(filename)) { //maximum image size GTK+ supports is 256x256; scale image down if necessary to prevent error image icon(filename); icon.scale(min(256u, icon.width()), min(256u, icon.height()), true); auto pixbuf = CreatePixbuf(icon); gtk_window_set_icon(GTK_WINDOW(widget), pixbuf); g_object_unref(G_OBJECT(pixbuf)); return true; } return false; } auto pWindow::_setMenuEnabled(bool enabled) -> void { gtk_widget_set_sensitive(gtkMenu, enabled); } auto pWindow::_setMenuFont(const Font& font) -> void { pFont::setFont(gtkMenu, font); } auto pWindow::_setMenuVisible(bool visible) -> void { gtk_widget_set_visible(gtkMenu, visible); } auto pWindow::_setStatusEnabled(bool enabled) -> void { gtk_widget_set_sensitive(gtkStatus, enabled); } auto pWindow::_setStatusFont(const Font& font) -> void { pFont::setFont(gtkStatus, font); } auto pWindow::_setStatusText(const string& text) -> void { gtk_statusbar_pop(GTK_STATUSBAR(gtkStatus), 1); gtk_statusbar_push(GTK_STATUSBAR(gtkStatus), 1, text); } auto pWindow::_setStatusVisible(bool visible) -> void { gtk_widget_set_visible(gtkStatus, visible); setResizable(self().resizable()); } auto pWindow::_statusHeight() const -> int { if(auto& statusBar = state().statusBar) { if(statusBar->visible()) { return settings.geometry.statusHeight + _statusTextHeight(); } } return 0; } auto pWindow::_statusTextHeight() const -> int { int height = 0; if(auto& statusBar = state().statusBar) { height = statusBar->font(true).size(statusBar->text()).height(); } return height; } //GTK is absolutely hopeless with window sizing //it will send size-allocate events during resizing of widgets during size-allocate events //instead of fighting with configure-event and size-allocate, just poll the state instead auto pWindow::_synchronizeGeometry() -> void { if(locked()) return; auto lock = acquire(); if(!gtk_widget_get_realized(widget)) return; if(!gtk_widget_get_visible(widget)) return; GtkAllocation allocation; gtk_widget_get_allocation(formContainer, &allocation); if(allocation.width != lastSize.width || allocation.height != lastSize.height) { auto size = self().geometry().size(); state().geometry.setSize({allocation.width, allocation.height}); if(auto& sizable = state().sizable) { sizable->setGeometry(self().geometry().setPosition()); } self().doSize(); //for GTK3: the window will not update after changing widget sizes until sending size-allocate //size-allocate will also call _synchronizeMargin() which is also important for window sizing //GtkAllocation is a typedef of GdkRectangle GtkAllocation rectangle; gtk_widget_get_allocation(widget, &rectangle); g_signal_emit_by_name(G_OBJECT(widget), "size-allocate", &rectangle, (gpointer)this, nullptr); } lastSize = allocation; auto gdkWindow = gtk_widget_get_window(widget); gdk_window_get_origin(gdkWindow, &allocation.x, &allocation.y); allocation.y += _menuHeight(); if(allocation.x != lastMove.x || allocation.y != lastMove.y) { state().geometry.setPosition({allocation.x, allocation.y}); self().doMove(); } lastMove = allocation; } auto pWindow::_synchronizeMargin() -> void { if(locked()) return; auto lock = acquire(); if(!gtk_widget_get_realized(widget)) return; if(!gtk_widget_get_visible(widget)) return; if(state().fullScreen || state().maximized || state().minimized) return; auto window = gtk_widget_get_window(widget); GdkRectangle border, client; gdk_window_get_frame_extents(window, &border); gdk_window_get_origin(window, &client.x, &client.y); #if HIRO_GTK==2 gdk_window_get_geometry(window, nullptr, nullptr, &client.width, &client.height, nullptr); #elif HIRO_GTK==3 gdk_window_get_geometry(window, nullptr, nullptr, &client.width, &client.height); #endif settings.geometry.frameX = client.x - border.x; settings.geometry.frameY = client.y - border.y; settings.geometry.frameWidth = border.width - client.width; settings.geometry.frameHeight = border.height - client.height; if(gtk_widget_get_visible(gtkMenu)) { GtkAllocation allocation; auto time = chrono::millisecond(); while(chrono::millisecond() - time < 20) { gtk_widget_get_allocation(gtkMenu, &allocation); if(allocation.height > 1) { settings.geometry.menuHeight = allocation.height - _menuTextHeight(); break; } } } if(gtk_widget_get_visible(gtkStatus)) { GtkAllocation allocation; auto time = chrono::millisecond(); while(chrono::millisecond() - time < 20) { gtk_widget_get_allocation(gtkStatus, &allocation); if(allocation.height > 1) { settings.geometry.statusHeight = allocation.height - _statusTextHeight(); break; } } } } //GTK doesn't add gtk_window_is_maximized() until 3.12; //and doesn't appear to have a companion gtk_window_is_(hidden,iconic,minimized); //so we have to do this the hard way auto pWindow::_synchronizeState() -> void { if(locked()) return; auto lock = acquire(); if(!gtk_widget_get_realized(widget)) return; #if defined(DISPLAY_WINDOWS) auto window = GDK_WINDOW_HWND(gtk_widget_get_window(widget)); bool maximized = IsZoomed(window); bool minimized = IsIconic(window); bool doSize = false; if(state().minimized != minimized) doSize = true; state().maximized = maximized; state().minimized = minimized; if(doSize) self().doSize(); #endif #if defined(DISPLAY_XORG) auto display = XOpenDisplay(nullptr); int screen = DefaultScreen(display); auto window = GDK_WINDOW_XID(gtk_widget_get_window(widget)); XlibAtom wmState = XInternAtom(display, "_NET_WM_STATE", XlibTrue); XlibAtom atom; int format; unsigned long items, after; unsigned char* data = nullptr; int result = XGetWindowProperty( display, window, wmState, 0, LONG_MAX, XlibFalse, AnyPropertyType, &atom, &format, &items, &after, &data ); auto atoms = (unsigned long*)data; if(result == Success) { bool maximizedHorizontal = false; bool maximizedVertical = false; bool minimized = false; for(auto index : range(items)) { auto memory = XGetAtomName(display, atoms[index]); auto name = string{memory}; if(name == "_NET_WM_STATE_MAXIMIZED_HORZ") maximizedHorizontal = true; if(name == "_NET_WM_STATE_MAXIMIZED_VERT") maximizedVertical = true; if(name == "_NET_WM_STATE_HIDDEN") minimized = true; XFree(memory); } bool doSize = false; //maximize sends size-allocate, which triggers doSize() if(state().minimized != minimized) doSize = true; //windows do not act bizarrely when maximized in only one direction //so for this reason, consider a window maximized only if it's in both directions state().maximized = maximizedHorizontal && maximizedVertical; state().minimized = minimized; if(doSize) self().doSize(); } XCloseDisplay(display); #endif } } #endif