1 /* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
3 * Copyright (C) 2007-2008 Collabora Ltd.
5 * This library is free software; you can redistribute it and/or
6 * modify it under the terms of the GNU Lesser General Public
7 * License as published by the Free Software Foundation; either
8 * version 2.1 of the License, or (at your option) any later version.
10 * This library is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
13 * Lesser General Public License for more details.
15 * You should have received a copy of the GNU Lesser General Public
16 * License along with this library; if not, write to the Free Software
17 * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
19 * Authors: Xavier Claessens <xclaesse@gmail.com>
26 #include <glib/gi18n.h>
27 #include <dbus/dbus-glib.h>
29 #include <telepathy-glib/dbus.h>
30 #include <libmissioncontrol/mc-enum-types.h>
32 #include "empathy-idle.h"
33 #include "empathy-utils.h"
35 #define DEBUG_FLAG EMPATHY_DEBUG_OTHER
36 #include "empathy-debug.h"
38 /* Number of seconds before entering extended autoaway. */
39 #define EXT_AWAY_TIME (30*60)
41 #define GET_PRIV(obj) EMPATHY_GET_PRIV (obj, EmpathyIdle)
49 McPresence flash_state;
53 gboolean away_reset_status;
54 McPresence away_saved_state;
55 gboolean nm_reset_status;
56 McPresence nm_saved_state;
59 gboolean nm_connected;
60 guint ext_away_timeout;
80 G_DEFINE_TYPE (EmpathyIdle, empathy_idle, G_TYPE_OBJECT)
83 idle_ext_away_cb (EmpathyIdle *idle)
85 EmpathyIdlePriv *priv;
87 priv = GET_PRIV (idle);
89 DEBUG ("Going to extended autoaway");
90 empathy_idle_set_state (idle, MC_PRESENCE_EXTENDED_AWAY);
91 priv->ext_away_timeout = 0;
97 idle_ext_away_stop (EmpathyIdle *idle)
99 EmpathyIdlePriv *priv;
101 priv = GET_PRIV (idle);
103 if (priv->ext_away_timeout) {
104 g_source_remove (priv->ext_away_timeout);
105 priv->ext_away_timeout = 0;
110 idle_ext_away_start (EmpathyIdle *idle)
112 EmpathyIdlePriv *priv;
114 priv = GET_PRIV (idle);
116 idle_ext_away_stop (idle);
117 priv->ext_away_timeout = g_timeout_add_seconds (EXT_AWAY_TIME,
118 (GSourceFunc) idle_ext_away_cb,
123 idle_presence_changed_cb (MissionControl *mc,
128 EmpathyIdlePriv *priv;
130 priv = GET_PRIV (idle);
132 DEBUG ("Presence changed to '%s' (%d)", status, state);
134 g_free (priv->status);
137 if (!G_STR_EMPTY (status)) {
138 priv->status = g_strdup (status);
141 g_object_notify (G_OBJECT (idle), "state");
142 g_object_notify (G_OBJECT (idle), "status");
146 idle_session_idle_changed_cb (DBusGProxy *gs_proxy,
150 EmpathyIdlePriv *priv;
152 priv = GET_PRIV (idle);
154 DEBUG ("Session idle state changed, %s -> %s",
155 priv->is_idle ? "yes" : "no",
156 is_idle ? "yes" : "no");
158 if (priv->state <= MC_PRESENCE_OFFLINE ||
159 priv->state == MC_PRESENCE_HIDDEN ||
161 /* We are not online or we don't want to go auto away,
162 * nothing to do here */
163 priv->is_idle = is_idle;
167 if (is_idle && !priv->is_idle) {
168 McPresence new_state;
169 /* We are now idle */
171 if (priv->state == MC_PRESENCE_AWAY ||
172 priv->state == MC_PRESENCE_EXTENDED_AWAY) {
173 /* User set away manually, when coming back we restore
174 * default presence. */
175 new_state = priv->state;
176 priv->away_saved_state = MC_PRESENCE_AVAILABLE;
177 priv->away_reset_status = TRUE;
179 new_state = MC_PRESENCE_AWAY;
180 priv->away_saved_state = priv->state;
181 priv->away_reset_status = FALSE;
184 DEBUG ("Going to autoaway");
185 empathy_idle_set_state (idle, new_state);
187 idle_ext_away_start (idle);
188 } else if (!is_idle && priv->is_idle) {
189 /* We are no more idle, restore state */
190 idle_ext_away_stop (idle);
192 DEBUG ("Restoring state to %d, reset status: %s",
193 priv->away_saved_state,
194 priv->away_reset_status ? "Yes" : "No");
196 if (priv->nm_connected) {
197 empathy_idle_set_presence (idle,
198 priv->away_saved_state,
199 priv->away_reset_status ? NULL : priv->status);
201 /* We can't restore state now, will do when NM gets
203 priv->nm_saved_state = priv->away_saved_state;
204 priv->nm_reset_status = priv->away_reset_status;
207 priv->away_saved_state = MC_PRESENCE_UNSET;
208 priv->away_reset_status = FALSE;
211 priv->is_idle = is_idle;
215 idle_nm_state_change_cb (DBusGProxy *proxy,
219 EmpathyIdlePriv *priv;
220 gboolean old_nm_connected;
221 gboolean new_nm_connected;
223 priv = GET_PRIV (idle);
225 DEBUG ("New network state (%d), in use = %s",
226 state, priv->use_nm ? "Yes" : "No");
232 old_nm_connected = priv->nm_connected;
233 new_nm_connected = !(state == NM_STATE_CONNECTING ||
234 state == NM_STATE_DISCONNECTED);
235 priv->nm_connected = TRUE; /* To be sure _set_state will work */
237 if (old_nm_connected && !new_nm_connected) {
238 /* We are no more connected */
239 idle_ext_away_stop (idle);
241 priv->nm_saved_state = priv->state;
242 empathy_idle_set_state (idle, MC_PRESENCE_OFFLINE);
244 else if (!old_nm_connected && new_nm_connected) {
245 /* We are now connected */
246 empathy_idle_set_presence (idle,
247 priv->nm_saved_state,
248 priv->nm_reset_status ? NULL : priv->status);
249 priv->nm_saved_state = MC_PRESENCE_UNSET;
250 priv->nm_reset_status = FALSE;
253 priv->nm_connected = new_nm_connected;
257 idle_finalize (GObject *object)
259 EmpathyIdlePriv *priv;
261 priv = GET_PRIV (object);
263 g_free (priv->status);
264 g_object_unref (priv->mc);
266 if (priv->gs_proxy) {
267 g_object_unref (priv->gs_proxy);
270 idle_ext_away_stop (EMPATHY_IDLE (object));
274 idle_get_property (GObject *object,
279 EmpathyIdlePriv *priv;
282 priv = GET_PRIV (object);
283 idle = EMPATHY_IDLE (object);
287 g_value_set_enum (value, empathy_idle_get_state (idle));
290 g_value_set_string (value, empathy_idle_get_status (idle));
292 case PROP_FLASH_STATE:
293 g_value_set_enum (value, empathy_idle_get_flash_state (idle));
296 g_value_set_boolean (value, empathy_idle_get_auto_away (idle));
299 g_value_set_boolean (value, empathy_idle_get_use_nm (idle));
302 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
308 idle_set_property (GObject *object,
313 EmpathyIdlePriv *priv;
316 priv = GET_PRIV (object);
317 idle = EMPATHY_IDLE (object);
321 empathy_idle_set_state (idle, g_value_get_enum (value));
324 empathy_idle_set_status (idle, g_value_get_string (value));
326 case PROP_FLASH_STATE:
327 empathy_idle_set_flash_state (idle, g_value_get_enum (value));
330 empathy_idle_set_auto_away (idle, g_value_get_boolean (value));
333 empathy_idle_set_use_nm (idle, g_value_get_boolean (value));
336 G_OBJECT_WARN_INVALID_PROPERTY_ID (object, param_id, pspec);
342 empathy_idle_class_init (EmpathyIdleClass *klass)
344 GObjectClass *object_class = G_OBJECT_CLASS (klass);
346 object_class->finalize = idle_finalize;
347 object_class->get_property = idle_get_property;
348 object_class->set_property = idle_set_property;
350 g_object_class_install_property (object_class,
352 g_param_spec_enum ("state",
356 MC_PRESENCE_AVAILABLE,
358 g_object_class_install_property (object_class,
360 g_param_spec_string ("status",
365 g_object_class_install_property (object_class,
367 g_param_spec_enum ("flash-state",
374 g_object_class_install_property (object_class,
376 g_param_spec_boolean ("auto-away",
377 "Automatic set presence to away",
378 "Should it set presence to away if inactive",
382 g_object_class_install_property (object_class,
384 g_param_spec_boolean ("use-nm",
385 "Use Network Manager",
386 "Set presence according to Network Manager",
390 g_type_class_add_private (object_class, sizeof (EmpathyIdlePriv));
394 empathy_idle_init (EmpathyIdle *idle)
396 DBusGConnection *system_bus;
397 GError *error = NULL;
398 EmpathyIdlePriv *priv = G_TYPE_INSTANCE_GET_PRIVATE (idle,
399 EMPATHY_TYPE_IDLE, EmpathyIdlePriv);
402 priv->is_idle = FALSE;
403 priv->mc = empathy_mission_control_new ();
404 priv->state = mission_control_get_presence_actual (priv->mc, NULL);
405 priv->status = mission_control_get_presence_message_actual (priv->mc, NULL);
406 if (G_STR_EMPTY (priv->status)) {
407 g_free (priv->status);
411 dbus_g_proxy_connect_signal (DBUS_G_PROXY (priv->mc),
413 G_CALLBACK (idle_presence_changed_cb),
416 priv->gs_proxy = dbus_g_proxy_new_for_name (tp_get_bus (),
417 "org.gnome.ScreenSaver",
418 "/org/gnome/ScreenSaver",
419 "org.gnome.ScreenSaver");
420 if (priv->gs_proxy) {
421 dbus_g_proxy_add_signal (priv->gs_proxy, "SessionIdleChanged",
424 dbus_g_proxy_connect_signal (priv->gs_proxy, "SessionIdleChanged",
425 G_CALLBACK (idle_session_idle_changed_cb),
428 DEBUG ("Failed to get gs proxy");
432 system_bus = dbus_g_bus_get (DBUS_BUS_SYSTEM, &error);
434 DEBUG ("Failed to get system bus: %s",
435 error ? error->message : "No error given");
437 priv->nm_proxy = dbus_g_proxy_new_for_name (system_bus,
438 "org.freedesktop.NetworkManager",
439 "/org/freedesktop/NetworkManager",
440 "org.freedesktop.NetworkManager");
442 if (priv->nm_proxy) {
443 dbus_g_proxy_add_signal (priv->nm_proxy, "StateChange",
444 G_TYPE_UINT, G_TYPE_INVALID);
445 dbus_g_proxy_connect_signal (priv->nm_proxy, "StateChange",
446 G_CALLBACK (idle_nm_state_change_cb),
449 DEBUG ("Failed to get nm proxy");
452 priv->nm_connected = TRUE;
456 empathy_idle_new (void)
458 static EmpathyIdle *idle = NULL;
461 idle = g_object_new (EMPATHY_TYPE_IDLE, NULL);
462 g_object_add_weak_pointer (G_OBJECT (idle), (gpointer) &idle);
471 empathy_idle_get_state (EmpathyIdle *idle)
473 EmpathyIdlePriv *priv;
475 priv = GET_PRIV (idle);
481 empathy_idle_set_state (EmpathyIdle *idle,
484 EmpathyIdlePriv *priv;
486 priv = GET_PRIV (idle);
488 empathy_idle_set_presence (idle, state, priv->status);
492 empathy_idle_get_status (EmpathyIdle *idle)
494 EmpathyIdlePriv *priv;
496 priv = GET_PRIV (idle);
499 return empathy_presence_get_default_message (priv->state);
506 empathy_idle_set_status (EmpathyIdle *idle,
509 EmpathyIdlePriv *priv;
511 priv = GET_PRIV (idle);
513 empathy_idle_set_presence (idle, priv->state, status);
517 empathy_idle_get_flash_state (EmpathyIdle *idle)
519 EmpathyIdlePriv *priv;
521 priv = GET_PRIV (idle);
523 return priv->flash_state;
527 empathy_idle_set_flash_state (EmpathyIdle *idle,
530 EmpathyIdlePriv *priv;
532 priv = GET_PRIV (idle);
534 priv->flash_state = state;
536 if (state == MC_PRESENCE_UNSET) {
539 g_object_notify (G_OBJECT (idle), "flash-state");
543 empathy_idle_set_presence (EmpathyIdle *idle,
547 EmpathyIdlePriv *priv;
548 const gchar *default_status;
550 priv = GET_PRIV (idle);
552 DEBUG ("Changing presence to %s (%d)",
555 if (!priv->nm_connected) {
556 DEBUG ("NM not connected");
560 /* Do not set translated default messages */
561 default_status = empathy_presence_get_default_message (state);
562 if (status && strcmp (status, default_status) == 0) {
566 mission_control_set_presence (priv->mc,
573 empathy_idle_get_auto_away (EmpathyIdle *idle)
575 EmpathyIdlePriv *priv = GET_PRIV (idle);
577 return priv->auto_away;
581 empathy_idle_set_auto_away (EmpathyIdle *idle,
584 EmpathyIdlePriv *priv = GET_PRIV (idle);
586 priv->auto_away = auto_away;
588 g_object_notify (G_OBJECT (idle), "auto-away");
592 empathy_idle_get_use_nm (EmpathyIdle *idle)
594 EmpathyIdlePriv *priv = GET_PRIV (idle);
600 empathy_idle_set_use_nm (EmpathyIdle *idle,
603 EmpathyIdlePriv *priv = GET_PRIV (idle);
605 if (!priv->nm_proxy || use_nm == priv->use_nm) {
609 priv->use_nm = use_nm;
613 GError *error = NULL;
615 dbus_g_proxy_call (priv->nm_proxy, "state",
618 G_TYPE_UINT, &nm_status,
622 DEBUG ("Couldn't get NM state: %s", error->message);
623 g_clear_error (&error);
624 nm_status = NM_STATE_ASLEEP;
627 idle_nm_state_change_cb (priv->nm_proxy, nm_status, idle);
629 if (!priv->nm_connected) {
630 empathy_idle_set_state (idle, priv->nm_saved_state);
632 priv->nm_connected = TRUE;
633 priv->nm_saved_state = MC_PRESENCE_UNSET;
636 g_object_notify (G_OBJECT (idle), "use-nm");