31616a1dcd2ce3af3de03149a7d16093a35ebb2d
[psensor-pkg-debian.git] / src / ui_appindicator.c
1 /*
2  * Copyright (C) 2010-2013 jeanfi@gmail.com
3  *
4  * This program is free software; you can redistribute it and/or
5  * modify it under the terms of the GNU General Public License as
6  * published by the Free Software Foundation; either version 2 of the
7  * License, or (at your option) any later version.
8  *
9  * This program is distributed in the hope that it will be useful, but
10  * WITHOUT ANY WARRANTY; without even the implied warranty of
11  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
12  * General Public License for more details.
13  *
14  * You should have received a copy of the GNU General Public License
15  * along with this program; if not, write to the Free Software
16  * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
17  * 02110-1301 USA
18  */
19 #include <stdio.h>
20 #include <stdlib.h>
21 #include <string.h>
22
23 #include <gtk/gtk.h>
24 #include <libappindicator/app-indicator.h>
25
26 #include "cfg.h"
27 #include "psensor.h"
28 #include "ui.h"
29 #include "ui_appindicator.h"
30 #include "ui_sensorpref.h"
31 #include "ui_status.h"
32 #include "ui_pref.h"
33
34 static const char *ICON = "psensor_normal";
35 static const char *ATTENTION_ICON = "psensor_hot";
36
37 static struct psensor **sensors;
38 static GtkMenuItem **menu_items;
39 static int appindicator_supported = 1;
40 static AppIndicator *indicator;
41 static struct ui_psensor *ui_psensor;
42
43 static void cb_menu_show(GtkMenuItem *mi, gpointer data)
44 {
45         ui_window_show((struct ui_psensor *)data);
46 }
47
48 static void cb_menu_quit(GtkMenuItem *mi, gpointer data)
49 {
50         ui_psensor_quit(data);
51 }
52
53 static void cb_menu_preferences(GtkMenuItem *mi, gpointer data)
54 {
55 #ifdef HAVE_APPINDICATOR_029
56         gdk_threads_enter();
57 #endif
58
59         ui_pref_dialog_run((struct ui_psensor *)data);
60
61 #ifdef HAVE_APPINDICATOR_029
62         gdk_threads_leave();
63 #endif
64 }
65
66 static void cb_sensor_preferences(GtkMenuItem *mi, gpointer data)
67 {
68         struct ui_psensor *ui = data;
69
70 #ifdef HAVE_APPINDICATOR_029
71         gdk_threads_enter();
72 #endif
73
74         if (ui->sensors && *ui->sensors)
75                 ui_sensorpref_dialog_run(*ui->sensors, ui);
76
77 #ifdef HAVE_APPINDICATOR_029
78         gdk_threads_leave();
79 #endif
80 }
81
82 static void cb_about(GtkMenuItem *mi, gpointer data)
83 {
84         ui_show_about_dialog();
85 }
86
87 static const char *menu_desc =
88 "<ui>"
89 "  <popup name='MainMenu'>"
90 "      <menuitem name='Show' action='ShowAction' />"
91 "      <separator />"
92 "      <separator />"
93 "      <menuitem name='Preferences' action='PreferencesAction' />"
94 "      <menuitem name='SensorPreferences' action='SensorPreferencesAction' />"
95 "      <separator />"
96 "      <menuitem name='About' action='AboutAction' />"
97 "      <separator />"
98 "      <menuitem name='Quit' action='QuitAction' />"
99 "  </popup>"
100 "</ui>";
101
102 static GtkActionEntry entries[] = {
103         { "PsensorMenuAction", NULL, "_Psensor" },
104
105         { "ShowAction", NULL,
106           N_("_Show"), NULL,
107           N_("Show"),
108           G_CALLBACK(cb_menu_show) },
109
110         { "PreferencesAction", GTK_STOCK_PREFERENCES,
111           N_("_Preferences"), NULL,
112           N_("Preferences"),
113           G_CALLBACK(cb_menu_preferences) },
114
115         { "SensorPreferencesAction", GTK_STOCK_PREFERENCES,
116           N_("S_ensor Preferences"),
117           NULL,
118           N_("SensorPreferences"),
119           G_CALLBACK(cb_sensor_preferences) },
120
121         { "AboutAction", NULL,
122           N_("_About"),
123           NULL,
124           N_("About"),
125           G_CALLBACK(cb_about) },
126
127         { "QuitAction",
128           GTK_STOCK_QUIT,
129           N_("_Quit"),
130           NULL, N_("Quit"),
131           G_CALLBACK(cb_menu_quit) }
132 };
133 static guint n_entries = G_N_ELEMENTS(entries);
134
135 static void
136 update_menu_item(GtkMenuItem *item, struct psensor *s, int use_celcius)
137 {
138         gchar *str;
139         char *v;
140
141         v = psensor_current_value_to_str(s, use_celcius);
142
143         str = g_strdup_printf("%s: %s", s->name, v);
144
145         gtk_menu_item_set_label(item, str);
146
147         free(v);
148         g_free(str);
149 }
150
151 static void update_menu_items(int use_celcius)
152 {
153         struct psensor **s;
154         GtkMenuItem **m;
155
156         if (!sensors)
157                 return ;
158
159         for (s = sensors, m = menu_items; *s; s++, m++)
160                 update_menu_item(*m, *s, use_celcius);
161 }
162
163 static void
164 build_sensor_menu_items(const struct ui_psensor *ui,
165                         GtkMenu *menu)
166 {
167         int i, j, n, celcius;
168         const char *name;
169         struct psensor **sorted_sensors;
170
171         free(menu_items);
172
173         celcius  = ui->config->temperature_unit == CELCIUS;
174
175         sorted_sensors = ui_get_sensors_ordered_by_position(ui);
176         n = psensor_list_size(sorted_sensors);
177         menu_items = malloc(n * sizeof(GtkWidget *));
178         sensors = malloc((n + 1) * sizeof(struct psensor *));
179         for (i = 0, j = 0; i < n; i++) {
180                 if (config_is_appindicator_enabled(sorted_sensors[i]->id)) {
181                         sensors[j] = sorted_sensors[i];
182                         name = sensors[j]->name;
183
184                         menu_items[j] = GTK_MENU_ITEM
185                                 (gtk_menu_item_new_with_label(name));
186
187                         gtk_menu_shell_insert(GTK_MENU_SHELL(menu),
188                                               GTK_WIDGET(menu_items[j]),
189                                               j+2);
190
191                         update_menu_item(menu_items[j], sensors[j], celcius);
192
193                         j++;
194                 }
195         }
196
197         sensors[j] = NULL;
198
199         free(sorted_sensors);
200 }
201
202 static GtkWidget *get_menu(struct ui_psensor *ui)
203 {
204         GtkActionGroup *action_group;
205         GtkUIManager *menu_manager;
206         GError *error;
207         GtkMenu *menu;
208
209         action_group = gtk_action_group_new("PsensorActions");
210         gtk_action_group_set_translation_domain(action_group, PACKAGE);
211         menu_manager = gtk_ui_manager_new();
212
213         gtk_action_group_add_actions(action_group, entries, n_entries, ui);
214         gtk_ui_manager_insert_action_group(menu_manager, action_group, 0);
215
216         error = NULL;
217         gtk_ui_manager_add_ui_from_string(menu_manager, menu_desc, -1, &error);
218
219         if (error)
220                 g_error(_("building menus failed: %s"), error->message);
221
222         menu = GTK_MENU(gtk_ui_manager_get_widget(menu_manager, "/MainMenu"));
223
224         build_sensor_menu_items(ui, menu);
225
226         return GTK_WIDGET(menu);
227 }
228
229 void ui_appindicator_update(struct ui_psensor *ui, unsigned int attention)
230 {
231         AppIndicatorStatus status;
232
233         if (!indicator)
234                 return;
235
236         status = app_indicator_get_status(indicator);
237
238         if (!attention && status == APP_INDICATOR_STATUS_ATTENTION)
239                 app_indicator_set_status(indicator,
240                                          APP_INDICATOR_STATUS_ACTIVE);
241
242         if (attention && status == APP_INDICATOR_STATUS_ACTIVE)
243                 app_indicator_set_status(indicator,
244                                          APP_INDICATOR_STATUS_ATTENTION);
245
246         update_menu_items(ui->config->temperature_unit == CELCIUS);
247 }
248
249 static GtkStatusIcon *unity_fallback(AppIndicator *indicator)
250 {
251         GtkStatusIcon *ico;
252
253         log_debug("ui_appindicator.unity_fallback()");
254
255         appindicator_supported = 0;
256
257         ico = ui_status_get_icon(ui_psensor);
258
259         ui_status_set_visible(1);
260
261         return ico;
262 }
263
264 static void
265 unity_unfallback(AppIndicator *indicator, GtkStatusIcon *status_icon)
266 {
267         log_debug("ui_appindicator.unity_unfallback()");
268
269         ui_status_set_visible(0);
270
271         appindicator_supported = 1;
272 }
273
274 void ui_appindicator_update_menu(struct ui_psensor *ui)
275 {
276         GtkWidget *menu;
277
278         menu = get_menu(ui);
279         app_indicator_set_menu(indicator, GTK_MENU(menu));
280
281         gtk_widget_show_all(menu);
282 }
283
284 void ui_appindicator_init(struct ui_psensor *ui)
285 {
286         ui_psensor = ui;
287
288         indicator = app_indicator_new
289                 ("psensor",
290                  ICON,
291                  APP_INDICATOR_CATEGORY_APPLICATION_STATUS);
292
293         APP_INDICATOR_GET_CLASS(indicator)->fallback = unity_fallback;
294         APP_INDICATOR_GET_CLASS(indicator)->unfallback = unity_unfallback;
295
296         app_indicator_set_status(indicator, APP_INDICATOR_STATUS_ACTIVE);
297         app_indicator_set_attention_icon(indicator, ATTENTION_ICON);
298
299         ui_appindicator_update_menu(ui);
300 }
301
302 int is_appindicator_supported()
303 {
304         return appindicator_supported;
305 }
306
307 void ui_appindicator_cleanup()
308 {
309         free(sensors);
310         /* TODO: cleanup menu items. */
311 }