/* * GNOME Logs - View and search logs * Copyright (C) 2013, 2014, 2015 Red Hat, Inc. * * This program 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. * * This program 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 this program. If not, see . */ #include "gl-window.h" #include #include "gl-categorylist.h" #include "gl-eventtoolbar.h" #include "gl-eventview.h" #include "gl-eventviewlist.h" #include "gl-enums.h" #include "gl-util.h" struct _GlWindow { /*< private >*/ GtkApplicationWindow parent_instance; }; typedef struct { GtkWidget *event_toolbar; GtkWidget *event; GtkWidget *info_bar; } GlWindowPrivate; G_DEFINE_TYPE_WITH_PRIVATE (GlWindow, gl_window, GTK_TYPE_APPLICATION_WINDOW) static void on_action_radio (GSimpleAction *action, GVariant *variant, gpointer user_data) { g_action_change_state (G_ACTION (action), variant); } static void on_action_toggle (GSimpleAction *action, GVariant *variant, gpointer user_data) { GVariant *variant_state; gboolean state; variant_state = g_action_get_state (G_ACTION (action)); state = g_variant_get_boolean (variant_state); g_action_change_state (G_ACTION (action), g_variant_new_boolean (!state)); } static void on_close (GSimpleAction *action, GVariant *variant, gpointer user_data) { GtkWindow *window; window = GTK_WINDOW (user_data); gtk_window_close (window); } static void on_toolbar_mode (GSimpleAction *action, GVariant *variant, gpointer user_data) { GlWindowPrivate *priv; const gchar *mode; GEnumClass *eclass; GEnumValue *evalue; GAction *search; priv = gl_window_get_instance_private (GL_WINDOW (user_data)); mode = g_variant_get_string (variant, NULL); eclass = g_type_class_ref (GL_TYPE_EVENT_TOOLBAR_MODE); evalue = g_enum_get_value_by_nick (eclass, mode); search = g_action_map_lookup_action (G_ACTION_MAP (user_data), "search"); if (evalue->value == GL_EVENT_TOOLBAR_MODE_LIST) { /* Switch the event view back to list mode if the back button is * clicked. */ GlEventView *view; view = GL_EVENT_VIEW (priv->event); gl_event_view_set_mode (view, GL_EVENT_VIEW_MODE_LIST); g_simple_action_set_enabled (G_SIMPLE_ACTION (search), TRUE); } else { g_simple_action_set_enabled (G_SIMPLE_ACTION (search), FALSE); g_action_change_state (search, g_variant_new_boolean (FALSE)); } g_simple_action_set_state (action, variant); g_type_class_unref (eclass); } static void on_view_mode (GSimpleAction *action, GVariant *variant, gpointer user_data) { GlWindowPrivate *priv; const gchar *mode; GlEventToolbar *toolbar; GlEventView *event; GEnumClass *eclass; GEnumValue *evalue; priv = gl_window_get_instance_private (GL_WINDOW (user_data)); mode = g_variant_get_string (variant, NULL); event = GL_EVENT_VIEW (priv->event); toolbar = GL_EVENT_TOOLBAR (priv->event_toolbar); eclass = g_type_class_ref (GL_TYPE_EVENT_VIEW_MODE); evalue = g_enum_get_value_by_nick (eclass, mode); switch (evalue->value) { case GL_EVENT_VIEW_MODE_LIST: gl_event_toolbar_set_mode (toolbar, GL_EVENT_TOOLBAR_MODE_LIST); break; case GL_EVENT_VIEW_MODE_DETAIL: gl_event_toolbar_set_mode (toolbar, GL_EVENT_TOOLBAR_MODE_DETAIL); gl_event_view_set_mode (event, GL_EVENT_VIEW_MODE_DETAIL); break; } g_simple_action_set_state (action, variant); g_type_class_unref (eclass); } static void on_search (GSimpleAction *action, GVariant *variant, gpointer user_data) { gboolean state; GlWindowPrivate *priv; GlEventView *view; state = g_variant_get_boolean (variant); priv = gl_window_get_instance_private (GL_WINDOW (user_data)); view = GL_EVENT_VIEW (priv->event); gl_event_view_set_search_mode (view, state); g_simple_action_set_state (action, variant); } static void on_export (GSimpleAction *action, GVariant *variant, gpointer user_data) { GlWindowPrivate *priv; GlEventView *event; GtkFileChooser *file_chooser; GtkWidget *dialog; gint res; priv = gl_window_get_instance_private (GL_WINDOW (user_data)); event = GL_EVENT_VIEW (priv->event); dialog = gtk_file_chooser_dialog_new (_("Save logs"), GTK_WINDOW (user_data), GTK_FILE_CHOOSER_ACTION_SAVE, _("_Cancel"), GTK_RESPONSE_CANCEL, _("_Save"), GTK_RESPONSE_ACCEPT, NULL); file_chooser = GTK_FILE_CHOOSER (dialog); gtk_file_chooser_set_do_overwrite_confirmation (file_chooser, TRUE); gtk_file_chooser_set_current_name (file_chooser, _("log messages")); res = gtk_dialog_run (GTK_DIALOG (dialog)); if (res == GTK_RESPONSE_ACCEPT) { gboolean have_error = FALSE; gchar *file_content; GFile *output_file; GFileOutputStream *file_ostream; GError *error = NULL; GtkWidget *error_dialog; file_content = gl_event_view_get_output_logs (event); output_file = gtk_file_chooser_get_file (file_chooser); file_ostream = g_file_replace (output_file, NULL, TRUE, G_FILE_CREATE_NONE, NULL, &error); if (error != NULL) { have_error = TRUE; g_warning ("Error while replacing exported log messages file: %s", error->message); g_clear_error (&error); } g_output_stream_write (G_OUTPUT_STREAM (file_ostream), file_content, strlen (file_content), NULL, &error); if (error != NULL) { have_error = TRUE; g_warning ("Error while replacing exported log messages file: %s", error->message); g_clear_error (&error); } g_output_stream_close (G_OUTPUT_STREAM (file_ostream), NULL, &error); if (error != NULL) { have_error = TRUE; g_warning ("Error while replacing exported log messages file: %s", error->message); g_clear_error (&error); } if (have_error == TRUE) { error_dialog = gtk_message_dialog_new (GTK_WINDOW (user_data), GTK_DIALOG_MODAL, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "%s", _("Unable to export log messages to a file")); gtk_dialog_run (GTK_DIALOG (error_dialog)); gtk_widget_destroy (error_dialog); } g_free (file_content); g_object_unref (file_ostream); g_object_unref (output_file); } gtk_widget_destroy (dialog); } static void on_view_boot (GSimpleAction *action, GVariant *variant, gpointer user_data) { GlWindowPrivate *priv; GlEventView *event; GlEventToolbar *toolbar; gchar *current_boot; const gchar *boot_match; priv = gl_window_get_instance_private (GL_WINDOW (user_data)); event = GL_EVENT_VIEW (priv->event); toolbar = GL_EVENT_TOOLBAR (priv->event_toolbar); boot_match = g_variant_get_string (variant, NULL); gl_event_view_view_boot (event, boot_match); current_boot = gl_event_view_get_current_boot_time (event, boot_match); if (current_boot == NULL) { g_debug ("Error fetching the time using boot_match"); } gl_event_toolbar_change_current_boot (toolbar, current_boot); g_simple_action_set_state (action, variant); g_free (current_boot); } static gboolean on_gl_window_key_press_event (GlWindow *window, GdkEvent *event, gpointer user_data) { GlWindowPrivate *priv; GlEventView *view; GAction *action; GVariant *variant; const gchar *mode; GEnumClass *eclass; GEnumValue *evalue; priv = gl_window_get_instance_private (window); action = g_action_map_lookup_action (G_ACTION_MAP (window), "search"); view = GL_EVENT_VIEW (priv->event); if (gl_event_view_handle_search_event (view, action, event) == GDK_EVENT_STOP) { return GDK_EVENT_STOP; } action = g_action_map_lookup_action (G_ACTION_MAP (window), "view-mode"); variant = g_action_get_state (action); mode = g_variant_get_string (variant, NULL); eclass = g_type_class_ref (GL_TYPE_EVENT_VIEW_MODE); evalue = g_enum_get_value_by_nick (eclass, mode); g_variant_unref (variant); switch (evalue->value) { case GL_EVENT_VIEW_MODE_LIST: break; /* Ignored, as there is no back button. */ case GL_EVENT_VIEW_MODE_DETAIL: if (gl_event_toolbar_handle_back_button_event (GL_EVENT_TOOLBAR (priv->event_toolbar), (GdkEventKey*)event) == GDK_EVENT_STOP) { GlEventView *view; view = GL_EVENT_VIEW (priv->event); gl_event_view_set_mode (view, GL_EVENT_VIEW_MODE_LIST); g_type_class_unref (eclass); return GDK_EVENT_STOP; } break; } g_type_class_unref (eclass); return GDK_EVENT_PROPAGATE; } static void on_help_button_clicked (GlWindow *window, gint response_id, gpointer user_data) { GlWindowPrivate *priv; GtkWindow *parent; GError *error = NULL; parent = GTK_WINDOW (window); priv = gl_window_get_instance_private (GL_WINDOW (window)); gtk_show_uri (gtk_window_get_screen (parent), "help:gnome-logs/permissions", GDK_CURRENT_TIME, &error); if (error) { g_debug ("Error while opening help: %s", error->message); g_error_free (error); } gtk_widget_hide (priv->info_bar); } void gl_window_set_sort_order (GlWindow *window, GlSortOrder sort_order) { GlWindowPrivate *priv; GlEventView *event; g_return_if_fail (GL_WINDOW (window)); priv = gl_window_get_instance_private (window); event = GL_EVENT_VIEW (priv->event); gl_event_view_set_sort_order (event, sort_order); } static GActionEntry actions[] = { { "view-mode", on_action_radio, "s", "'list'", on_view_mode }, { "toolbar-mode", on_action_radio, "s", "'list'", on_toolbar_mode }, { "search", on_action_toggle, NULL, "false", on_search }, { "view-boot", on_action_radio, "s", "''", on_view_boot }, { "export", on_export }, { "close", on_close } }; static void gl_window_class_init (GlWindowClass *klass) { GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass); gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/Logs/gl-window.ui"); gtk_widget_class_bind_template_child_private (widget_class, GlWindow, event_toolbar); gtk_widget_class_bind_template_child_private (widget_class, GlWindow, event); gtk_widget_class_bind_template_child_private (widget_class, GlWindow, info_bar); gtk_widget_class_bind_template_callback (widget_class, on_gl_window_key_press_event); gtk_widget_class_bind_template_callback (widget_class, on_help_button_clicked); } static void gl_window_init (GlWindow *window) { GtkCssProvider *provider; GdkScreen *screen; GlWindowPrivate *priv; GlEventToolbar *toolbar; GlEventView *event; GAction *action_view_boot; GArray *boot_ids; GlJournalBootID *boot_id; gchar *boot_match; GVariant *variant; gtk_widget_init_template (GTK_WIDGET (window)); priv = gl_window_get_instance_private (window); event = GL_EVENT_VIEW (priv->event); toolbar = GL_EVENT_TOOLBAR (priv->event_toolbar); boot_ids = gl_event_view_get_boot_ids (event); boot_id = &g_array_index (boot_ids, GlJournalBootID, boot_ids->len - 1); boot_match = boot_id->boot_match; gl_event_toolbar_add_boots (toolbar, boot_ids); g_action_map_add_action_entries (G_ACTION_MAP (window), actions, G_N_ELEMENTS (actions), window); action_view_boot = g_action_map_lookup_action (G_ACTION_MAP (window), "view-boot"); variant = g_variant_new_string (boot_match); g_action_change_state (action_view_boot, variant); provider = gtk_css_provider_new (); g_signal_connect (provider, "parsing-error", G_CALLBACK (gl_util_on_css_provider_parsing_error), NULL); gtk_css_provider_load_from_resource (provider, "/org/gnome/Logs/gl-style.css"); screen = gdk_screen_get_default (); gtk_style_context_add_provider_for_screen (screen, GTK_STYLE_PROVIDER (provider), GTK_STYLE_PROVIDER_PRIORITY_APPLICATION); if (!gl_util_can_read_system_journal ()) { GtkWidget *message_label; GtkWidget *content_area; message_label = gtk_label_new (_("Unable to read system logs")); gtk_widget_set_hexpand (GTK_WIDGET (message_label), TRUE); gtk_widget_show (message_label); content_area = gtk_info_bar_get_content_area (GTK_INFO_BAR (priv->info_bar)); gtk_container_add (GTK_CONTAINER (content_area), message_label); gtk_widget_show (priv->info_bar); } if (!gl_util_can_read_user_journal ()) { GtkWidget *message_label; GtkWidget *content_area; message_label = gtk_label_new (_("Unable to read user logs")); gtk_widget_set_hexpand (GTK_WIDGET (message_label), TRUE); gtk_widget_show (message_label); content_area = gtk_info_bar_get_content_area (GTK_INFO_BAR (priv->info_bar)); gtk_container_add (GTK_CONTAINER (content_area), message_label); gtk_widget_show (priv->info_bar); } g_object_unref (provider); } GtkWidget * gl_window_new (GtkApplication *application) { g_return_val_if_fail (GTK_APPLICATION (application), NULL); return g_object_new (GL_TYPE_WINDOW, "application", application, NULL); }