/* * Main window * * Copyright (C) 2017 Patrick McDermott * * This file is part of Marquee. * * Marquee is free software: you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Marquee is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with Marquee. If not, see . */ #include "window.h" #include #include #include #include "application.h" #include "notebook.h" #include "tab-page.h" struct _MqWindow { GtkWindow parent_instance; MqApplication *application; const gchar **uris; MqConfig *config; GtkWidget *notebook; MqTabPage *root_tab; guint current_tab; gboolean fullscreen; }; enum { PROP_APPLICATION = 1, PROP_URIS, N_PROPERTIES }; static GParamSpec *obj_properties[N_PROPERTIES] = {NULL,}; struct _MqWindowClass { GtkWindowClass parent_class; }; G_DEFINE_TYPE(MqWindow, mq_window, GTK_TYPE_WINDOW) static void is_maximized_cb(MqWindow *window, GParamSpec G_GNUC_UNUSED *param_spec) { mq_config_set_boolean(window->config, "window.maximized", gtk_window_is_maximized(GTK_WINDOW(window))); mq_config_save(window->config); } static void configure_event_cb(MqWindow G_GNUC_UNUSED *window, GdkEventConfigure *event) { mq_config_set_integer(window->config, "window.width", event->width); mq_config_set_integer(window->config, "window.height", event->height); mq_config_save(window->config); } static gboolean window_state_event_cb(MqWindow *window, GdkEventWindowState *event) { window->fullscreen = event->new_window_state & GDK_WINDOW_STATE_FULLSCREEN; return FALSE; } static void close_confirm_warn_check_button_toggled_cb(GtkToggleButton *toggle_button, MqWindow *window) { mq_config_set_boolean(window->config, "tabs.warn-on-close", gtk_toggle_button_get_active(toggle_button)); mq_config_save(window->config); } static void close_confirm_response_cb(GtkWidget *dialog, gint response_id, MqWindow *window) { gtk_widget_destroy(dialog); if (response_id == GTK_RESPONSE_OK) { gtk_widget_destroy(GTK_WIDGET(window)); } } static gboolean delete_event_cb(MqWindow *window, GdkEvent G_GNUC_UNUSED *event) { guint num_tabs; gchar *message; GtkWidget *message_label; GtkWidget *check_button; GtkWidget *vbox; GtkWidget *hbox; GtkWidget *dialog; num_tabs = mq_window_get_num_tabs(window); if (num_tabs == 1) { return FALSE; } if (!mq_config_get_boolean(window->config, "tabs.warn-on-close")) { return FALSE; } /* Message */ message = g_strdup_printf("You are about to close %d tabs. " "Are you sure you want to continue?", num_tabs); message_label = gtk_label_new(message); g_free(message); /* Check button */ check_button = gtk_check_button_new_with_mnemonic( "_Warn When Closing Multiple Tabs or Windows"); gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(check_button), mq_config_get_boolean(window->config, "tabs.warn-on-close")); /* Connect signal after setting initial toggle status, to avoid a * spurious signal. */ g_signal_connect(check_button, "toggled", G_CALLBACK(close_confirm_warn_check_button_toggled_cb), window); /* Right vertical box (message and check button) */ vbox = gtk_box_new(GTK_ORIENTATION_VERTICAL, 0); gtk_box_pack_start(GTK_BOX(vbox), message_label, FALSE, FALSE, 0); gtk_box_pack_start(GTK_BOX(vbox), check_button, FALSE, FALSE, 0); /* Horizontal box (icon and right vertical box) */ hbox = gtk_box_new(GTK_ORIENTATION_HORIZONTAL, 0); gtk_box_pack_start(GTK_BOX(hbox), gtk_image_new_from_icon_name( "dialog-question", GTK_ICON_SIZE_DIALOG), FALSE, FALSE, 0); gtk_box_pack_start(GTK_BOX(hbox), vbox, FALSE, FALSE, 0); /* Dialog */ dialog = gtk_dialog_new_with_buttons("Confirm Close", GTK_WINDOW(window), GTK_DIALOG_DESTROY_WITH_PARENT, "_Cancel", GTK_RESPONSE_CANCEL, "Cl_ose Tabs", GTK_RESPONSE_OK, NULL); gtk_container_add( GTK_CONTAINER(gtk_dialog_get_content_area(GTK_DIALOG(dialog))), hbox); g_signal_connect(dialog, "response", G_CALLBACK(close_confirm_response_cb), window); gtk_widget_grab_focus(gtk_dialog_get_widget_for_response( GTK_DIALOG(dialog), GTK_RESPONSE_OK)); gtk_widget_show_all(dialog); return TRUE; } static void destroy_cb(MqWindow *window) { mq_application_delete_window(window->application, window); } static void set_title(MqWindow *window, const gchar *title) { gchar *window_title; window_title = g_strdup_printf("%s - Marquee", title); gtk_window_set_title(GTK_WINDOW(window), window_title); g_free(window_title); } static void switch_page_cb(GtkNotebook G_GNUC_UNUSED *notebook, GtkWidget G_GNUC_UNUSED *page, guint page_num, MqWindow *window) { window->current_tab = ++page_num; /* TODO: Use MqNotebook function. */ set_title(window, mq_tab_page_get_title( MQ_TAB_PAGE(gtk_notebook_get_nth_page( GTK_NOTEBOOK(window->notebook), page_num)))); } static void update_positions(GtkNotebook G_GNUC_UNUSED *notebook, GtkWidget G_GNUC_UNUSED *child, guint G_GNUC_UNUSED page_num, MqWindow *window) { /* TODO: After MqTab becomes MqTabPage, derived from GtkBox, call * mq_tab_page_update_positions() on child or * mq_tab_page_seek(page_num + 1), whichever has the lower position. */ /* TODO: Should this also update the tabs data structure? Probably. */ /* Temporarily "use" window. */ window = window; } static void constructed(GObject *object) { MqWindow *window; gsize i; window = MQ_WINDOW(object); if (mq_config_get_boolean(window->config, "window.maximized")) { gtk_window_maximize(GTK_WINDOW(window)); } else { gtk_window_unmaximize(GTK_WINDOW(window)); } gtk_window_set_default_size(GTK_WINDOW(window), mq_config_get_integer(window->config, "window.width"), mq_config_get_integer(window->config, "window.height")); if (window->uris && window->uris[0]) { for (i = 0; window->uris && window->uris[i]; ++i) { mq_tab_page_new(window, window->uris[i]); } } else { mq_tab_page_new(window, NULL); } gtk_widget_show_all(GTK_WIDGET(window)); } static void get_property(GObject *object, guint property_id, GValue *value, GParamSpec *param_spec) { MqWindow *window; window = MQ_WINDOW(object); switch (property_id) { case PROP_APPLICATION: g_value_set_pointer(value, window->application); break; case PROP_URIS: g_value_set_pointer(value, window->uris); break; ; default: G_OBJECT_WARN_INVALID_PROPERTY_ID(object, property_id, param_spec); break; } } static void set_property(GObject *object, guint property_id, const GValue *value, GParamSpec *param_spec) { MqWindow *window; window = MQ_WINDOW(object); switch (property_id) { case PROP_APPLICATION: window->application = g_value_get_pointer(value); window->config = mq_application_get_config( window->application); break; case PROP_URIS: window->uris = g_value_get_pointer(value); break; ; default: G_OBJECT_WARN_INVALID_PROPERTY_ID(object, property_id, param_spec); break; } } static void mq_window_class_init(MqWindowClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS(klass); object_class->constructed = constructed; object_class->get_property = get_property; object_class->set_property = set_property; obj_properties[PROP_APPLICATION] = g_param_spec_pointer( "application", "Application", "The parent MqApplication instance", G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB); obj_properties[PROP_URIS] = g_param_spec_pointer( "uris", "URIs", "A NULL-terminated string array of URIs to load", G_PARAM_WRITABLE | G_PARAM_CONSTRUCT_ONLY | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB); g_object_class_install_properties(object_class, N_PROPERTIES, obj_properties); } static void mq_window_init(MqWindow *window) { window->fullscreen = FALSE; g_signal_connect(window, "notify::is-maximized", G_CALLBACK(is_maximized_cb), NULL); g_signal_connect(window, "configure-event", G_CALLBACK(configure_event_cb), NULL); g_signal_connect(window, "window-state-event", G_CALLBACK(window_state_event_cb), NULL); g_signal_connect(window, "delete-event", G_CALLBACK(delete_event_cb), NULL); g_signal_connect(window, "destroy", G_CALLBACK(destroy_cb), NULL); window->notebook = mq_notebook_new(window); gtk_container_add(GTK_CONTAINER(window), window->notebook); g_signal_connect(window->notebook, "switch-page", G_CALLBACK(switch_page_cb), window); g_signal_connect(window->notebook, "page-reordered", G_CALLBACK(update_positions), window); } MqWindow * mq_window_new(MqApplication *application, const gchar **uris) { return g_object_new(MQ_TYPE_WINDOW, "type", GTK_WINDOW_TOPLEVEL, "application", application, "uris", uris, NULL); } void mq_window_quit(MqWindow *window) { mq_application_quit(window->application, GTK_WINDOW(window)); } MqApplication * mq_window_get_application(MqWindow *window) { return window->application; } void mq_window_toggle_fullscreen(MqWindow *window) { if (!window->fullscreen) { gtk_window_fullscreen(GTK_WINDOW(window)); } else { gtk_window_unfullscreen(GTK_WINDOW(window)); } } void mq_window_insert_tab(MqWindow *window, GtkWidget *tab_page, GtkWidget *tab_label, gint position) { gtk_notebook_insert_page(GTK_NOTEBOOK(window->notebook), tab_page, tab_label, position - 1); gtk_notebook_set_tab_reorderable(GTK_NOTEBOOK(window->notebook), tab_page, TRUE); gtk_notebook_set_tab_detachable(GTK_NOTEBOOK(window->notebook), tab_page, TRUE); gtk_widget_show_all(tab_page); gtk_widget_show_all(tab_label); } void mq_window_set_current_tab(MqWindow *window, guint tab) { window->current_tab = tab; gtk_notebook_set_current_page(GTK_NOTEBOOK(window->notebook), tab - 1); } guint mq_window_get_current_tab(MqWindow *window) { return window->current_tab; } guint mq_window_get_num_tabs(MqWindow *window) { /* TODO: Use MqNotebook function. */ return gtk_notebook_get_n_pages(GTK_NOTEBOOK(window->notebook)); } void mq_window_update_tab_title(MqWindow *window, guint position, const gchar *title) { if (position == window->current_tab) { set_title(window, title); } } void mq_window_scroll_tab_labels(MqWindow *window) { mq_tab_page_scroll_tab_labels(window->root_tab); } void mq_window_begin_scrolling_tab_labels(MqWindow *window) { mq_tab_page_begin_scrolling_tab_labels(window->root_tab); } void mq_window_end_scrolling_tab_labels(MqWindow *window) { mq_tab_page_end_scrolling_tab_labels(window->root_tab); }