/* -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ #include #include #include #include #include #include #include #include "application.h" #include "procdialogs.h" #include "prefsdialog.h" #include "interface.h" #include "proctable.h" #include "load-graph.h" #include "settings-keys.h" #include "argv.h" #include "util.h" #include "lsof.h" #include "disks.h" static void cb_solaris_mode_changed (Gio::Settings& settings, Glib::ustring key, GsmApplication* app) { app->config.solaris_mode = settings.get_boolean(key); app->cpu_graph->clear_background(); if (app->timeout) { proctable_update (app); } } static void cb_draw_stacked_changed (Gio::Settings& settings, Glib::ustring key, GsmApplication* app) { app->config.draw_stacked = settings.get_boolean(key); app->cpu_graph->clear_background(); load_graph_reset(app->cpu_graph); } static void cb_draw_smooth_changed (Gio::Settings& settings, Glib::ustring key, GsmApplication* app) { app->config.draw_smooth = settings.get_boolean(key); app->cpu_graph->clear_background(); load_graph_reset(app->cpu_graph); } static void cb_network_in_bits_changed (Gio::Settings& settings, Glib::ustring key, GsmApplication* app) { app->config.network_in_bits = settings.get_boolean(key); // force scale to be redrawn app->net_graph->clear_background(); } static void cb_timeouts_changed (Gio::Settings& settings, Glib::ustring key, GsmApplication* app) { if (key == GSM_SETTING_PROCESS_UPDATE_INTERVAL) { app->config.update_interval = settings.get_int (key); app->smooth_refresh->reset(); if (app->timeout) { proctable_reset_timeout (app); } } else if (key == GSM_SETTING_GRAPH_UPDATE_INTERVAL) { app->config.graph_update_interval = settings.get_int (key); load_graph_change_speed(app->cpu_graph, app->config.graph_update_interval); load_graph_change_speed(app->mem_graph, app->config.graph_update_interval); load_graph_change_speed(app->net_graph, app->config.graph_update_interval); } else if (key == GSM_SETTING_DISKS_UPDATE_INTERVAL) { app->config.disks_update_interval = settings.get_int (key); disks_reset_timeout (app); } } static void apply_cpu_color_settings(Gio::Settings& settings, GsmApplication* app) { GVariant *cpu_colors_var = g_settings_get_value (settings.gobj (), GSM_SETTING_CPU_COLORS); gsize n = g_variant_n_children(cpu_colors_var); gchar *color; // Create builder to add the new colors if user has more than the number of cores with defaults. GVariantBuilder builder; GVariant* child; GVariant* full; g_variant_builder_init(&builder, G_VARIANT_TYPE_ARRAY); for (guint i = 0; i < static_cast(app->config.num_cpus); i++) { if(i < n) { child = g_variant_get_child_value ( cpu_colors_var, i ); g_variant_get_child( child, 1, "s", &color); g_variant_builder_add_value ( &builder, child); g_variant_unref (child); } else { color = g_strdup ("#f25915e815e8"); g_variant_builder_add(&builder, "(us)", i, color); } gdk_rgba_parse(&app->config.cpu_color[i], color); g_free (color); } full = g_variant_builder_end(&builder); // if the user has more cores than colors stored in the gsettings, store the newly built gvariant in gsettings if (n < static_cast(app->config.num_cpus)) { g_settings_set_value(settings.gobj (), GSM_SETTING_CPU_COLORS, full); } else { g_variant_unref(full); } g_variant_unref(cpu_colors_var); } static void cb_color_changed (Gio::Settings& settings, Glib::ustring key, GsmApplication* app) { if (key == GSM_SETTING_CPU_COLORS) { apply_cpu_color_settings(settings, app); for (int i = 0; i < app->config.num_cpus; i++) { if(!gdk_rgba_equal(&app->cpu_graph->colors[i], &app->config.cpu_color[i])) { app->cpu_graph->colors[i] = app->config.cpu_color[i]; break; } } return; } auto color = settings.get_string (key); if (key == GSM_SETTING_MEM_COLOR) { gdk_rgba_parse (&app->config.mem_color, color.c_str ()); app->mem_graph->colors.at(0) = app->config.mem_color; } else if (key == GSM_SETTING_SWAP_COLOR) { gdk_rgba_parse (&app->config.swap_color, color.c_str ()); app->mem_graph->colors.at(1) = app->config.swap_color; } else if (key == GSM_SETTING_NET_IN_COLOR) { gdk_rgba_parse (&app->config.net_in_color, color.c_str ()); app->net_graph->colors.at(0) = app->config.net_in_color; } else if (key == GSM_SETTING_NET_OUT_COLOR) { gdk_rgba_parse (&app->config.net_out_color, color.c_str ()); app->net_graph->colors.at(1) = app->config.net_out_color; } } void GsmApplication::load_settings() { glibtop_cpu cpu; this->settings = Gio::Settings::create (GSM_GSETTINGS_SCHEMA); config.solaris_mode = this->settings->get_boolean (GSM_SETTING_SOLARIS_MODE); this->settings->signal_changed(GSM_SETTING_SOLARIS_MODE).connect ([this](const Glib::ustring& key) { cb_solaris_mode_changed (*this->settings.operator->(), key, this); }); config.draw_stacked = this->settings->get_boolean (GSM_SETTING_DRAW_STACKED); this->settings->signal_changed(GSM_SETTING_DRAW_STACKED).connect ([this](const Glib::ustring& key) { cb_draw_stacked_changed (*this->settings.operator->(), key, this); }); config.draw_smooth = this->settings->get_boolean (GSM_SETTING_DRAW_SMOOTH); this->settings->signal_changed(GSM_SETTING_DRAW_SMOOTH).connect ([this](const Glib::ustring& key) { cb_draw_smooth_changed (*this->settings.operator->(), key, this); }); config.network_in_bits = this->settings->get_boolean (GSM_SETTING_NETWORK_IN_BITS); this->settings->signal_changed (GSM_SETTING_NETWORK_IN_BITS).connect ([this](const Glib::ustring& key) { cb_network_in_bits_changed (*this->settings.operator->(), key, this); }); auto cbtc = [this](const Glib::ustring& key) { cb_timeouts_changed(*this->settings.operator->(), key, this); }; config.update_interval = this->settings->get_int (GSM_SETTING_PROCESS_UPDATE_INTERVAL); this->settings->signal_changed (GSM_SETTING_PROCESS_UPDATE_INTERVAL).connect (cbtc); config.graph_update_interval = this->settings->get_int (GSM_SETTING_GRAPH_UPDATE_INTERVAL); this->settings->signal_changed (GSM_SETTING_GRAPH_UPDATE_INTERVAL).connect (cbtc); config.disks_update_interval = this->settings->get_int (GSM_SETTING_DISKS_UPDATE_INTERVAL); this->settings->signal_changed (GSM_SETTING_DISKS_UPDATE_INTERVAL).connect (cbtc); glibtop_get_cpu (&cpu); frequency = cpu.frequency; config.num_cpus = glibtop_get_sysinfo()->ncpu; // or server->ncpu + 1 apply_cpu_color_settings (*this->settings.operator->(), this); auto mem_color = this->settings->get_string (GSM_SETTING_MEM_COLOR); gdk_rgba_parse (&config.mem_color, mem_color.empty () ? "#000000ff0082" : mem_color.c_str ()); auto swap_color = this->settings->get_string (GSM_SETTING_SWAP_COLOR); gdk_rgba_parse (&config.swap_color, swap_color.empty () ? "#00b6000000ff" : swap_color.c_str ()); auto net_in_color = this->settings->get_string (GSM_SETTING_NET_IN_COLOR); gdk_rgba_parse (&config.net_in_color, net_in_color.empty () ? "#000000f200f2" : net_in_color.c_str ()); auto net_out_color = this->settings->get_string (GSM_SETTING_NET_OUT_COLOR); gdk_rgba_parse (&config.net_out_color, net_out_color.empty () ? "#00f2000000c1" : net_out_color.c_str ()); auto cbcc = [this](const Glib::ustring& key) { cb_color_changed(*this->settings.operator->(), key, this); }; for (auto k : {GSM_SETTING_CPU_COLORS, GSM_SETTING_MEM_COLOR, GSM_SETTING_SWAP_COLOR, GSM_SETTING_NET_IN_COLOR, GSM_SETTING_NET_OUT_COLOR}) { this->settings->signal_changed (k).connect(cbcc); } } GsmApplication::GsmApplication() : Gtk::Application("org.gnome.SystemMonitor", Gio::APPLICATION_HANDLES_COMMAND_LINE), tree(NULL), proc_actionbar_revealer(NULL), popup_menu(NULL), disk_list(NULL), stack(NULL), refresh_button(NULL), process_menu_button(NULL), window_menu_button(NULL), end_process_button(NULL), search_button(NULL), search_entry(NULL), search_bar(NULL), config(), cpu_graph(NULL), mem_graph(NULL), net_graph(NULL), cpu_label_fixed_width(0), net_label_fixed_width(0), selection(NULL), timeout(0U), disk_timeout(0U), top_of_tree(NULL), last_vscroll_max(0.0), last_vscroll_value(0.0), pretty_table(NULL), settings(NULL), main_window(NULL), frequency(0U), smooth_refresh(NULL), cpu_total_time(0ULL), cpu_total_time_last(0ULL) { Glib::set_application_name(_("System Monitor")); } Glib::RefPtr GsmApplication::get () { static Glib::RefPtr singleton; if (!singleton) { singleton = Glib::RefPtr(new GsmApplication()); } return singleton; } void GsmApplication::on_activate() { gtk_window_present (GTK_WINDOW (main_window)); } void GsmApplication::save_config () { int width, height, xpos, ypos; gboolean maximized; gtk_window_get_size (GTK_WINDOW (main_window), &width, &height); gtk_window_get_position (GTK_WINDOW (main_window), &xpos, &ypos); maximized = gdk_window_get_state (gtk_widget_get_window (GTK_WIDGET (main_window))) & GDK_WINDOW_STATE_MAXIMIZED; g_settings_set (settings->gobj (), GSM_SETTING_WINDOW_STATE, "(iiii)", width, height, xpos, ypos); settings->set_boolean (GSM_SETTING_MAXIMIZED, maximized); } int GsmApplication::on_command_line(const Glib::RefPtr& command_line) { int argc = 0; char** argv = command_line->get_arguments(argc); Glib::OptionContext context; context.set_summary(_("A simple process and system monitor.")); context.set_ignore_unknown_options(true); procman::OptionGroup option_group; context.set_main_group(option_group); try { context.parse(argc, argv); } catch (const Glib::Error& ex) { g_error("Arguments parse error : %s", ex.what().c_str()); } g_strfreev(argv); if (option_group.print_version) { g_print("%s %s\n", _("GNOME System Monitor"), VERSION); exit (EXIT_SUCCESS); } if (option_group.show_processes_tab) this->settings->set_string (GSM_SETTING_CURRENT_TAB, "processes"); else if (option_group.show_resources_tab) this->settings->set_string (GSM_SETTING_CURRENT_TAB, "resources"); else if (option_group.show_file_systems_tab) this->settings->set_string (GSM_SETTING_CURRENT_TAB, "disks"); else if (option_group.print_version) on_activate (); return 0; } void GsmApplication::on_help_activate(const Glib::VariantBase&) { GError* error = 0; if (!g_app_info_launch_default_for_uri("help:gnome-system-monitor", NULL, &error)) { g_warning("Could not display help : %s", error->message); g_error_free(error); } } void GsmApplication::on_lsof_activate(const Glib::VariantBase&) { procman_lsof(this); } void GsmApplication::on_preferences_activate(const Glib::VariantBase&) { create_preferences_dialog (this); } void GsmApplication::on_quit_activate(const Glib::VariantBase&) { shutdown (); } void GsmApplication::shutdown() { save_config (); if (timeout) g_source_remove (timeout); if (disk_timeout) g_source_remove (disk_timeout); proctable_free_table (this); delete smooth_refresh; delete pretty_table; glibtop_close(); quit(); } void GsmApplication::on_startup() { Gtk::Application::on_startup(); load_resources (); Glib::RefPtr action; action = Gio::SimpleAction::create("quit"); action->signal_activate().connect(sigc::mem_fun(*this, &GsmApplication::on_quit_activate)); add_action(action); action = Gio::SimpleAction::create("help"); action->signal_activate().connect(sigc::mem_fun(*this, &GsmApplication::on_help_activate)); add_action(action); action = Gio::SimpleAction::create("lsof"); action->signal_activate().connect(sigc::mem_fun(*this, &GsmApplication::on_lsof_activate)); add_action(action); action = Gio::SimpleAction::create("preferences"); action->signal_activate().connect(sigc::mem_fun(*this, &GsmApplication::on_preferences_activate)); add_action(action); Glib::RefPtr builder = Gtk::Builder::create_from_resource("/org/gnome/gnome-system-monitor/data/menus.ui"); Glib::RefPtr menu = Glib::RefPtr::cast_static(builder->get_object ("app-menu")); set_app_menu (menu); add_accelerator("d", "win.show-dependencies", NULL); add_accelerator("q", "app.quit", NULL); add_accelerator("s", "win.send-signal-stop", g_variant_new_int32 (SIGSTOP)); add_accelerator("c", "win.send-signal-cont", g_variant_new_int32 (SIGCONT)); add_accelerator("e", "win.send-signal-end", g_variant_new_int32 (SIGTERM)); add_accelerator("k", "win.send-signal-kill", g_variant_new_int32 (SIGKILL)); add_accelerator("m", "win.memory-maps", NULL); add_accelerator("o", "win.open-files", NULL); add_accelerator("Return", "win.process-properties", NULL); add_accelerator("f", "win.search", g_variant_new_boolean (TRUE)); add_accelerator("F1", "app.help", NULL); Gtk::Window::set_default_icon_name ("org.gnome.SystemMonitor"); glibtop_init (); load_settings (); pretty_table = new PrettyTable(); smooth_refresh = new SmoothRefresh(settings); create_main_window (this); add_accelerator ("1", "win.show-page", g_variant_new_string ("processes")); add_accelerator ("2", "win.show-page", g_variant_new_string ("resources")); add_accelerator ("3", "win.show-page", g_variant_new_string ("disks")); add_accelerator ("r", "win.refresh", NULL); gtk_widget_show (GTK_WIDGET (main_window)); } void GsmApplication::load_resources() { auto res = Gio::Resource::create_from_file(GSM_RESOURCE_FILE); res->register_global(); }