]> git.llucax.com Git - software/sadba.git/blob - src/lib-display-blanking-status-menu-widget.c
49f838245a0714e6fb544c38f533c0068b754fb2
[software/sadba.git] / src / lib-display-blanking-status-menu-widget.c
1  /***********************************************************************************
2  *  Display blanking status area plugin
3  *  Copyright (C) 2012 Leandro Lucarella
4  *  Based on status-area-orientationlock-applet by Mohammad Abu-Garbeyyeh.
5  *
6  *  This program is free software; you can redistribute it and/or modify
7  *  it under the terms of the GNU General Public License as published by
8  *  the Free Software Foundation; either version 2 of the License, or
9  *  (at your option) any later version.
10  *
11  *  This program is distributed in the hope that it will be useful,
12  *  but WITHOUT ANY WARRANTY; without even the implied warranty of
13  *  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
14  *  GNU General Public License for more details.
15  *
16  *  You should have received a copy of the GNU General Public License
17  *  along with this program; if not, write to the Free Software
18  *  Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
19  *
20  ***********************************************************************************/
21
22 #include <stdio.h>
23 #include <stdlib.h>
24 #include <string.h>
25 #include <libintl.h>
26 #include <gtk/gtk.h>
27 #include <hildon/hildon.h>
28 #include <libhildondesktop/libhildondesktop.h>
29 #include <gconf/gconf-client.h>
30 #include <dbus/dbus.h>
31 #include <mce/dbus-names.h>
32
33
34 #define TYPE_DISPLAY_BLANKING_STATUS_PLUGIN (display_blanking_status_plugin_get_type ())
35
36 #define DISPLAY_BLANKING_STATUS_PLUGIN(obj)            (G_TYPE_CHECK_INSTANCE_CAST ((obj), \
37                                     TYPE_DISPLAY_BLANKING_STATUS_PLUGIN, DisplayBlankingStatusPlugin))
38
39 #define DISPLAY_BLANKING_STATUS_PLUGIN_CLASS(klass)    (G_TYPE_CHECK_CLASS_CAST ((klass), \
40                                 TYPE_DISPLAY_BLANKING_STATUS_PLUGIN, DisplayBlankingStatusPluginClass))
41
42 #define IS_DISPLAY_BLANKING_STATUS_PLUGIN(obj)         (G_TYPE_CHECK_INSTANCE_TYPE ((obj), \
43                                                     TYPE_DISPLAY_BLANKING_STATUS_PLUGIN))
44
45 #define IS_DISPLAY_BLANKING_STATUS_PLUGIN_CLASS(klass) (G_TYPE_CHECK_CLASS_TYPE ((klass), \
46                                                     TYPE_DISPLAY_BLANKING_STATUS_PLUGIN))
47
48 #define DISPLAY_BLANKING_STATUS_PLUGIN_GET_CLASS(obj)  (G_TYPE_INSTANCE_GET_CLASS ((obj), \
49                             TYPE_DISPLAY_BLANKING_STATUS_PLUGIN, DisplayBlankingStatusPluginClass))
50
51 #define STATUS_AREA_DISPLAY_BLANKING_ICON_SIZE 18
52
53 typedef struct _DisplayBlankingStatusPlugin        DisplayBlankingStatusPlugin;
54 typedef struct _DisplayBlankingStatusPluginClass   DisplayBlankingStatusPluginClass;
55 typedef struct _DisplayBlankingStatusPluginPrivate DisplayBlankingStatusPluginPrivate;
56
57 struct _DisplayBlankingStatusPlugin
58 {
59     HDStatusMenuItem parent;
60
61     DisplayBlankingStatusPluginPrivate *priv;
62 };
63
64 struct _DisplayBlankingStatusPluginClass
65 {
66     HDStatusMenuItemClass parent;
67 };
68
69 GType display_blanking_status_plugin_get_type (void);
70
71 #define DISPLAY_BLANKING_STATUS_PLUGIN_GET_PRIVATE(obj) \
72         (G_TYPE_INSTANCE_GET_PRIVATE (obj, \
73                 TYPE_DISPLAY_BLANKING_STATUS_PLUGIN, \
74                 DisplayBlankingStatusPluginPrivate))
75
76 #define MODE_GCONF_ROOT "/system/osso/dsm/display"
77 #define MODE_GCONF_KEY  MODE_GCONF_ROOT "/inhibit_blank_mode"
78
79 #define INHIBIT_MSG_INTERVAL 30 // in seconds
80
81 #define GETTEXT_DOM "status-area-displayblanking-applet"
82 #define gettext_noop(str) (str)
83
84 // Undocumented blanking modes as reported by David Weinehall from Nokia:
85 // http://www.gossamer-threads.com/lists/maemo/developers/61201#61201
86 #define BLANKING_MODES 5
87 static const char *_DisplayBlankingDescription[BLANKING_MODES] =
88 {
89     gettext_noop ("Both enabled"),
90     gettext_noop ("Both only on battery"),
91     gettext_noop ("Blanking only on battery"),
92     gettext_noop ("Both disabled"),
93     gettext_noop ("Only dimming")
94 };
95 static const char *mode_icon_name[BLANKING_MODES] =
96 {
97     "display-blanking-icon.0",
98     "display-blanking-icon.1",
99     "display-blanking-icon.2",
100     "display-blanking-icon.3",
101     "display-blanking-icon.4",
102 };
103 #define INHIBIT_ICON_NAME "display-blanking-inhibit-icon"
104
105 struct _DisplayBlankingStatusPluginPrivate
106 {
107     GConfClient *gconf_client;
108     DBusConnection* dbus_conn;
109     DBusMessage* dbus_msg;
110     GtkWidget *mode_button;
111     GtkWidget *mode_dialog;
112     gint inhibit_timer_id; // if == 0, no timer is set
113     gpointer data;
114 };
115
116 HD_DEFINE_PLUGIN_MODULE (DisplayBlankingStatusPlugin,
117         display_blanking_status_plugin, HD_TYPE_STATUS_MENU_ITEM)
118
119 static void
120 display_blanking_status_plugin_class_finalize (
121         DisplayBlankingStatusPluginClass *klass)
122 {
123 }
124
125 static void
126 display_blanking_status_plugin_class_init (DisplayBlankingStatusPluginClass *c)
127 {
128     g_type_class_add_private (c, sizeof (DisplayBlankingStatusPluginPrivate));
129 }
130
131 static void
132 update_mode_gui (gint mode, DisplayBlankingStatusPluginPrivate *priv)
133 {
134     GtkWidget *icon = gtk_image_new_from_icon_name (mode_icon_name[mode],
135             GTK_ICON_SIZE_DIALOG);
136     gtk_button_set_image (GTK_BUTTON (priv->mode_button), icon);
137 }
138
139 static gboolean
140 on_timeout (DisplayBlankingStatusPluginPrivate *priv)
141 {
142     dbus_bool_t ok = dbus_connection_send (priv->dbus_conn, priv->dbus_msg,
143             NULL);
144     g_assert (ok == TRUE);
145
146     return TRUE;
147 }
148
149 static void
150 on_mode_dialog_button_clicked (GtkWidget *button, GtkDialog *dialog)
151 {
152     const gchar *title = hildon_button_get_title (HILDON_BUTTON (button));
153
154     gint *mode = (gint *) g_object_get_data (G_OBJECT(dialog), "mode");
155     g_assert (mode != NULL);
156
157     for (*mode = 0; *mode < BLANKING_MODES; (*mode)++) {
158         if (strcmp (title, _DisplayBlankingDescription[*mode]) == 0)
159             break;
160     }
161     g_assert (*mode < BLANKING_MODES);
162
163     gtk_dialog_response (dialog, GTK_RESPONSE_OK);
164 }
165
166 static void
167 on_mode_button_clicked (GtkWidget *button,
168         DisplayBlankingStatusPluginPrivate *priv)
169 {
170     GtkWidget *parent = gtk_widget_get_ancestor (GTK_WIDGET (priv->mode_button),
171             GTK_TYPE_WINDOW);
172     gtk_widget_hide (parent);
173
174     g_assert (priv->mode_dialog == NULL);
175     priv->mode_dialog = gtk_dialog_new ();
176     gtk_window_set_modal (GTK_WINDOW (priv->mode_dialog), TRUE);
177     gtk_window_set_title (GTK_WINDOW (priv->mode_dialog),
178             dgettext (GETTEXT_DOM, "Select display blanking mode"));
179
180     GtkWidget *pan_area = hildon_pannable_area_new ();
181     g_assert (pan_area != NULL);
182
183     GtkWidget *vbox = gtk_vbox_new (FALSE, 0);
184     g_assert (vbox != NULL);
185
186     hildon_pannable_area_add_with_viewport (HILDON_PANNABLE_AREA (pan_area),
187             vbox);
188     gtk_box_pack_start (GTK_BOX (GTK_DIALOG (priv->mode_dialog)->vbox),
189             pan_area, TRUE, TRUE, 0);
190
191     gtk_widget_set_size_request (pan_area, -1, MIN (350, BLANKING_MODES * 70));
192
193     gint mode = BLANKING_MODES;
194     for (int i = 0; i < BLANKING_MODES; i++) {
195         GtkWidget *button =
196                 hildon_button_new_with_text (HILDON_SIZE_FINGER_HEIGHT,
197                     HILDON_BUTTON_ARRANGEMENT_VERTICAL,
198                     dgettext (GETTEXT_DOM, (_DisplayBlankingDescription[i])),
199                     NULL);
200         hildon_button_set_style (HILDON_BUTTON (button),
201             HILDON_BUTTON_STYLE_PICKER);
202         GtkWidget *icon = gtk_image_new_from_icon_name (mode_icon_name[i],
203                 GTK_ICON_SIZE_DIALOG);
204         hildon_button_set_image (HILDON_BUTTON (button), icon);
205         gtk_button_set_alignment (GTK_BUTTON (button), 0.0f, 0.5f);
206         gtk_box_pack_start (GTK_BOX (vbox), button, FALSE, FALSE, 0);
207         g_signal_connect (button, "clicked",
208                 G_CALLBACK (on_mode_dialog_button_clicked), priv->mode_dialog);
209     }
210
211     gtk_widget_show_all (priv->mode_dialog);
212
213     g_object_set_data (G_OBJECT (priv->mode_dialog), "mode", &mode);
214     gtk_dialog_run (GTK_DIALOG (priv->mode_dialog));
215
216     if (mode != BLANKING_MODES) {
217         // will trigger the gconf notify signal
218         GError *error = NULL;
219         gconf_client_set_int (priv->gconf_client, MODE_GCONF_KEY, mode, &error);
220         g_assert (error == NULL);
221     }
222
223     gtk_widget_destroy (priv->mode_dialog);
224     priv->mode_dialog = NULL;
225 }
226
227 static void
228 on_inhibit_button_clicked (GtkWidget *button,
229         DisplayBlankingStatusPluginPrivate *priv)
230 {
231     GtkWidget *parent = gtk_widget_get_ancestor (GTK_WIDGET (priv->mode_button),
232             GTK_TYPE_WINDOW);
233     gtk_widget_hide (parent);
234
235     if (priv->inhibit_timer_id) {
236         gboolean ok = g_source_remove (priv->inhibit_timer_id);
237         g_assert (ok == TRUE);
238         priv->inhibit_timer_id = 0;
239     }
240     else {
241         priv->inhibit_timer_id = g_timeout_add_seconds (INHIBIT_MSG_INTERVAL,
242                 (GSourceFunc) on_timeout, priv);
243         g_assert (priv->inhibit_timer_id > 0);
244     }
245 }
246
247 static void
248 on_gconf_notify (GConfClient* client, guint cnxn_id, GConfEntry* entry,
249         DisplayBlankingStatusPluginPrivate* priv)
250 {
251     const gchar* key = gconf_entry_get_key (entry);
252     g_assert (key != NULL);
253
254     // Ignore notification about keys we don't care about
255     if (strcmp (key, MODE_GCONF_KEY) != 0)
256         return;
257
258     const GConfValue* value = gconf_entry_get_value (entry);
259     g_assert (value != NULL);
260     g_assert (GCONF_VALUE_TYPE_VALID (value->type));
261     g_assert (value->type == GCONF_VALUE_INT);
262
263     gint mode = gconf_value_get_int (value);
264     update_mode_gui (mode, priv);
265 }
266
267 static void
268 init_gconf (DisplayBlankingStatusPluginPrivate *priv)
269 {
270     GError* error = NULL;
271
272     priv->gconf_client = gconf_client_get_default ();
273     g_assert (GCONF_IS_CLIENT (priv->gconf_client));
274
275     gconf_client_add_dir (priv->gconf_client, MODE_GCONF_ROOT,
276             GCONF_CLIENT_PRELOAD_NONE, &error);
277     g_assert (error == NULL);
278
279     gconf_client_notify_add (priv->gconf_client, MODE_GCONF_KEY,
280             (GConfClientNotifyFunc) &on_gconf_notify, priv, NULL, &error);
281     g_assert (error == NULL);
282 }
283
284 static void
285 init_dbus (DisplayBlankingStatusPluginPrivate *priv)
286 {
287     DBusError error;
288     dbus_error_init (&error);
289
290     priv->dbus_conn = dbus_bus_get (DBUS_BUS_SYSTEM, &error);
291     g_assert (!dbus_error_is_set (&error));
292     g_assert (priv->dbus_conn != NULL);
293
294     priv->dbus_msg = dbus_message_new_method_call (MCE_SERVICE,
295             MCE_REQUEST_PATH, MCE_REQUEST_IF, MCE_PREVENT_BLANK_REQ);
296     g_assert (priv->dbus_msg != NULL);
297     dbus_message_set_no_reply (priv->dbus_msg, TRUE);
298 }
299
300 static void
301 init_mode_gui (DisplayBlankingStatusPluginPrivate *priv)
302 {
303     priv->mode_dialog = NULL;
304     priv->mode_button = hildon_gtk_button_new (HILDON_SIZE_FINGER_HEIGHT |
305                 HILDON_SIZE_AUTO_WIDTH);
306
307     GError* error = NULL;
308     gint mode = gconf_client_get_int (priv->gconf_client, MODE_GCONF_KEY,
309             &error);
310     g_assert (error == NULL);
311     update_mode_gui (mode, priv);
312
313     g_signal_connect (priv->mode_button, "clicked",
314             G_CALLBACK (on_mode_button_clicked), priv);
315 }
316
317 static void
318 display_blanking_status_plugin_init (DisplayBlankingStatusPlugin *plugin)
319 {
320     DisplayBlankingStatusPluginPrivate *priv;
321
322     priv = DISPLAY_BLANKING_STATUS_PLUGIN_GET_PRIVATE (plugin);
323     plugin->priv = priv;
324
325     init_gconf (priv);
326     init_dbus (priv);
327     init_mode_gui (priv);
328
329     priv->inhibit_timer_id = 0;
330     GtkWidget *inhibit_button = hildon_gtk_toggle_button_new (
331             HILDON_SIZE_FINGER_HEIGHT | HILDON_SIZE_AUTO_WIDTH);
332     gtk_toggle_button_set_mode (GTK_TOGGLE_BUTTON(inhibit_button), FALSE);
333     GtkWidget *icon = gtk_image_new_from_icon_name (INHIBIT_ICON_NAME,
334             GTK_ICON_SIZE_DIALOG);
335     gtk_button_set_image (GTK_BUTTON (inhibit_button), icon);
336     g_signal_connect (inhibit_button, "clicked",
337             G_CALLBACK (on_inhibit_button_clicked), priv);
338
339     GtkWidget *hbbox = gtk_hbutton_box_new ();
340     g_assert (hbbox != NULL);
341
342     gtk_container_add (GTK_CONTAINER (hbbox), priv->mode_button);
343     gtk_container_add (GTK_CONTAINER (hbbox), inhibit_button);
344
345     gtk_container_add (GTK_CONTAINER (plugin), hbbox);
346
347     gtk_widget_show_all (GTK_WIDGET (plugin));
348 }
349