aboutsummaryrefslogtreecommitdiffstats
path: root/src/empathy-chat-manager.c
diff options
context:
space:
mode:
Diffstat (limited to 'src/empathy-chat-manager.c')
-rw-r--r--src/empathy-chat-manager.c168
1 files changed, 168 insertions, 0 deletions
diff --git a/src/empathy-chat-manager.c b/src/empathy-chat-manager.c
index 7c66a509b..2de258924 100644
--- a/src/empathy-chat-manager.c
+++ b/src/empathy-chat-manager.c
@@ -17,8 +17,12 @@
* Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
*/
+#include <telepathy-glib/telepathy-glib.h>
+
#include <libempathy/empathy-dispatcher.h>
+#include "empathy-chat-window.h"
+
#define DEBUG_FLAG EMPATHY_DEBUG_OTHER
#include <libempathy/empathy-debug.h>
@@ -40,6 +44,8 @@ struct _EmpathyChatManagerPriv
{
/* Queue of (ChatData *) representing the closed chats */
GQueue *queue;
+
+ TpBaseClient *handler;
};
#define GET_PRIV(o) \
@@ -88,11 +94,171 @@ chat_data_free (ChatData *data)
}
static void
+process_tp_chat (EmpathyTpChat *tp_chat,
+ TpAccount *account,
+ gint64 user_action_time)
+{
+ EmpathyChat *chat = NULL;
+ const gchar *id;
+
+ id = empathy_tp_chat_get_id (tp_chat);
+ if (!tp_str_empty (id))
+ {
+ chat = empathy_chat_window_find_chat (account, id);
+ }
+
+ if (chat != NULL)
+ {
+ empathy_chat_set_tp_chat (chat, tp_chat);
+ }
+ else
+ {
+ chat = empathy_chat_new (tp_chat);
+ /* empathy_chat_new returns a floating reference as EmpathyChat is
+ * a GtkWidget. This reference will be taken by a container
+ * (a GtkNotebook) when we'll call empathy_chat_window_present_chat */
+ }
+
+ empathy_chat_window_present_chat (chat, user_action_time);
+
+ if (empathy_tp_chat_is_invited (tp_chat, NULL))
+ {
+ /* We have been invited to the room. Add ourself as member as this
+ * channel has been approved. */
+ empathy_tp_chat_join (tp_chat);
+ }
+
+ g_object_unref (tp_chat);
+}
+
+typedef struct
+{
+ EmpathyTpChat *tp_chat;
+ TpAccount *account;
+ gint64 user_action_time;
+ gulong sig_id;
+} chat_ready_ctx;
+
+static chat_ready_ctx *
+chat_ready_ctx_new (EmpathyTpChat *tp_chat,
+ TpAccount *account,
+ gint64 user_action_time)
+{
+ chat_ready_ctx *ctx = g_slice_new0 (chat_ready_ctx);
+
+ ctx->tp_chat = g_object_ref (tp_chat);
+ ctx->account = g_object_ref (account);
+ ctx->user_action_time = user_action_time;
+ return ctx;
+}
+
+static void
+chat_ready_ctx_free (chat_ready_ctx *ctx)
+{
+ g_object_unref (ctx->tp_chat);
+ g_object_unref (ctx->account);
+
+ if (ctx->sig_id != 0)
+ g_signal_handler_disconnect (ctx->tp_chat, ctx->sig_id);
+
+ g_slice_free (chat_ready_ctx, ctx);
+}
+
+static void
+tp_chat_ready_cb (GObject *object,
+ GParamSpec *spec,
+ gpointer user_data)
+{
+ EmpathyTpChat *tp_chat = EMPATHY_TP_CHAT (object);
+ chat_ready_ctx *ctx = user_data;
+
+ if (!empathy_tp_chat_is_ready (tp_chat))
+ return;
+
+ process_tp_chat (tp_chat, ctx->account, ctx->user_action_time);
+
+ chat_ready_ctx_free (ctx);
+}
+
+static void
+handle_channels (TpSimpleHandler *handler,
+ TpAccount *account,
+ TpConnection *connection,
+ GList *channels,
+ GList *requests_satisfied,
+ gint64 user_action_time,
+ TpHandleChannelsContext *context,
+ gpointer user_data)
+{
+ GList *l;
+
+ for (l = channels; l != NULL; l = g_list_next (l))
+ {
+ TpChannel *channel = l->data;
+ EmpathyTpChat *tp_chat;
+
+ tp_chat = empathy_tp_chat_new (channel);
+
+ if (empathy_tp_chat_is_ready (tp_chat))
+ {
+ process_tp_chat (tp_chat, account, user_action_time);
+ }
+ else
+ {
+ chat_ready_ctx *ctx = chat_ready_ctx_new (tp_chat, account,
+ user_action_time);
+
+ ctx->sig_id = g_signal_connect (tp_chat, "notify::ready",
+ G_CALLBACK (tp_chat_ready_cb), ctx);
+ }
+ }
+
+ tp_handle_channels_context_accept (context);
+}
+
+static void
empathy_chat_manager_init (EmpathyChatManager *self)
{
EmpathyChatManagerPriv *priv = GET_PRIV (self);
+ TpDBusDaemon *dbus;
+ GError *error = NULL;
priv->queue = g_queue_new ();
+
+ dbus = tp_dbus_daemon_dup (&error);
+ if (dbus == NULL)
+ {
+ g_critical ("Failed to get D-Bus daemon: %s", error->message);
+ g_error_free (error);
+ return;
+ }
+
+ /* Text channels handler */
+ priv->handler = tp_simple_handler_new (dbus, FALSE, FALSE, "Empathy", FALSE,
+ handle_channels, self, NULL);
+
+ g_object_unref (dbus);
+
+ tp_base_client_take_handler_filter (priv->handler, tp_asv_new (
+ TP_PROP_CHANNEL_CHANNEL_TYPE, G_TYPE_STRING, TP_IFACE_CHANNEL_TYPE_TEXT,
+ TP_PROP_CHANNEL_TARGET_HANDLE_TYPE, G_TYPE_UINT, TP_HANDLE_TYPE_CONTACT,
+ NULL));
+
+ tp_base_client_take_handler_filter (priv->handler, tp_asv_new (
+ TP_PROP_CHANNEL_CHANNEL_TYPE, G_TYPE_STRING, TP_IFACE_CHANNEL_TYPE_TEXT,
+ TP_PROP_CHANNEL_TARGET_HANDLE_TYPE, G_TYPE_UINT, TP_HANDLE_TYPE_ROOM,
+ NULL));
+
+ tp_base_client_take_handler_filter (priv->handler, tp_asv_new (
+ TP_PROP_CHANNEL_CHANNEL_TYPE, G_TYPE_STRING, TP_IFACE_CHANNEL_TYPE_TEXT,
+ TP_PROP_CHANNEL_TARGET_HANDLE_TYPE, G_TYPE_UINT, TP_HANDLE_TYPE_NONE,
+ NULL));
+
+ if (!tp_base_client_register (priv->handler, &error))
+ {
+ g_critical ("Failed to register text handler: %s", error->message);
+ g_error_free (error);
+ }
}
static void
@@ -108,6 +274,8 @@ empathy_chat_manager_finalize (GObject *object)
priv->queue = NULL;
}
+ tp_clear_object (&priv->handler);
+
G_OBJECT_CLASS (empathy_chat_manager_parent_class)->finalize (object);
}