aboutsummaryrefslogtreecommitdiffstats
path: root/libempathy/empathy-tp-call.c
diff options
context:
space:
mode:
Diffstat (limited to 'libempathy/empathy-tp-call.c')
-rw-r--r--libempathy/empathy-tp-call.c655
1 files changed, 655 insertions, 0 deletions
diff --git a/libempathy/empathy-tp-call.c b/libempathy/empathy-tp-call.c
new file mode 100644
index 000000000..e0347ba9a
--- /dev/null
+++ b/libempathy/empathy-tp-call.c
@@ -0,0 +1,655 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
+/*
+ * Copyright (C) 2007 Elliot Fairweather
+ * Copyright (C) 2007 Collabora Ltd.
+ *
+ * This library is free software; you can redistribute it and/or
+ * modify it under the terms of the GNU Lesser General Public
+ * License as published by the Free Software Foundation; either
+ * version 2.1 of the License, or (at your option) any later version.
+ *
+ * This library is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ * Lesser General Public License for more details.
+ *
+ * You should have received a copy of the GNU Lesser General Public
+ * License along with this library; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
+ *
+ * Authors: Elliot Fairweather <elliot.fairweather@collabora.co.uk>
+ * Xavier Claessens <xclaesse@gmail.com>
+ */
+
+#include "config.h"
+
+#include <libtelepathy/tp-chan-type-streamed-media-gen.h>
+#include <libtelepathy/tp-helpers.h>
+#include <libtelepathy/tp-conn.h>
+
+#include <libmissioncontrol/mission-control.h>
+
+#include "empathy-tp-call.h"
+#include "empathy-tp-group.h"
+#include "empathy-utils.h"
+#include "empathy-debug.h"
+#include "empathy-enum-types.h"
+#include "tp-stream-engine-gen.h"
+
+#define DEBUG_DOMAIN "TpCall"
+
+#define GET_PRIV(obj) (G_TYPE_INSTANCE_GET_PRIVATE ((obj), EMPATHY_TYPE_TP_CALL, EmpathyTpCallPriv))
+
+#define STREAM_ENGINE_BUS_NAME "org.freedesktop.Telepathy.StreamEngine"
+#define STREAM_ENGINE_OBJECT_PATH "/org/freedesktop/Telepathy/StreamEngine"
+#define STREAM_ENGINE_INTERFACE "org.freedesktop.Telepathy.StreamEngine"
+#define CHANNEL_HANDLER_INTERFACE "org.freedesktop.Telepathy.ChannelHandler"
+
+typedef struct _EmpathyTpCallPriv EmpathyTpCallPriv;
+
+struct _EmpathyTpCallPriv {
+ TpChan *tp_chan;
+ DBusGProxy *streamed_iface;
+ DBusGProxy *se_ch_proxy;
+ DBusGProxy *se_proxy;
+ McAccount *account;
+ EmpathyTpGroup *group;
+ EmpathyContact *contact;
+ EmpathyTpCallStatus status;
+ gboolean is_incoming;
+ guint audio_stream;
+ guint video_stream;
+};
+
+static void empathy_tp_call_class_init (EmpathyTpCallClass *klass);
+static void empathy_tp_call_init (EmpathyTpCall *call);
+
+enum {
+ PROP_0,
+ PROP_ACCOUNT,
+ PROP_TP_CHAN,
+ PROP_STATUS
+};
+
+enum {
+ DESTROY,
+ LAST_SIGNAL
+};
+
+static guint signals[LAST_SIGNAL];
+
+G_DEFINE_TYPE (EmpathyTpCall, empathy_tp_call, G_TYPE_OBJECT)
+
+static void
+tp_call_set_status (EmpathyTpCall *call,
+ EmpathyTpCallStatus status)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ priv->status = status;
+ g_object_notify (G_OBJECT (call), "status");
+}
+
+static void
+tp_call_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (object);
+
+ switch (prop_id) {
+ case PROP_ACCOUNT:
+ priv->account = g_object_ref (g_value_get_object (value));
+ break;
+ case PROP_TP_CHAN:
+ priv->tp_chan = g_object_ref (g_value_get_object (value));
+ break;
+ case PROP_STATUS:
+ tp_call_set_status (EMPATHY_TP_CALL (object),
+ g_value_get_enum (value));
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+tp_call_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (object);
+
+ switch (prop_id) {
+ case PROP_ACCOUNT:
+ g_value_set_object (value, priv->account);
+ break;
+ case PROP_TP_CHAN:
+ g_value_set_object (value, priv->tp_chan);
+ break;
+ case PROP_STATUS:
+ g_value_set_enum (value, priv->status);
+ break;
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ break;
+ }
+}
+
+static void
+tp_call_destroy_cb (TpChan *call_chan,
+ EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ empathy_debug (DEBUG_DOMAIN, "Channel Closed or CM crashed");
+
+ g_object_unref (priv->tp_chan);
+ priv->tp_chan = NULL;
+ priv->streamed_iface = NULL;
+
+ g_signal_emit (call, signals[DESTROY], 0);
+}
+
+static void
+tp_call_closed_cb (TpChan *call_chan,
+ EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ /* The channel is closed, do just like if the proxy was destroyed */
+ g_signal_handlers_disconnect_by_func (priv->tp_chan,
+ tp_call_destroy_cb,
+ call);
+ tp_call_destroy_cb (call_chan, call);
+}
+
+static void
+tp_call_stream_added_cb (DBusGProxy *streamed_iface,
+ guint stream_id,
+ guint contact_handle,
+ guint stream_type,
+ EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ empathy_debug (DEBUG_DOMAIN, "Stream added: id=%d, stream_type=%d",
+ stream_id, stream_type);
+
+ switch (stream_type) {
+ case TP_MEDIA_STREAM_TYPE_AUDIO:
+ priv->audio_stream = stream_id;
+ break;
+ case TP_MEDIA_STREAM_TYPE_VIDEO:
+ priv->video_stream = stream_id;
+ break;
+ default:
+ empathy_debug (DEBUG_DOMAIN, "Unknown stream type: %d", stream_type);
+ }
+}
+
+
+static void
+tp_call_stream_removed_cb (DBusGProxy *streamed_iface,
+ guint stream_id,
+ EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ empathy_debug (DEBUG_DOMAIN, "Stream removed: %d", stream_id);
+
+ if (stream_id == priv->audio_stream) {
+ priv->audio_stream = 0;
+ }
+ else if (stream_id == priv->video_stream) {
+ priv->video_stream = 0;
+ }
+}
+
+static void
+tp_call_list_streams_cb (DBusGProxy *proxy,
+ GPtrArray *streams,
+ GError *error,
+ gpointer user_data)
+{
+ guint i;
+
+ if (error) {
+ empathy_debug (DEBUG_DOMAIN, "Failed to list streams: %s",
+ error->message);
+ return;
+ }
+
+ for (i = 0; i < streams->len; i++) {
+ GValueArray *values;
+ guint stream_id;
+ guint contact_handle;
+ guint stream_type;
+
+ values = g_ptr_array_index (streams, i);
+ stream_id = g_value_get_uint (g_value_array_get_nth (values, 0));
+ contact_handle = g_value_get_uint (g_value_array_get_nth (values, 1));
+ stream_type = g_value_get_uint (g_value_array_get_nth (values, 2));
+
+ tp_call_stream_added_cb (proxy,
+ stream_id,
+ contact_handle,
+ stream_type,
+ user_data);
+ }
+}
+
+static void
+tp_call_member_added_cb (EmpathyTpGroup *group,
+ EmpathyContact *contact,
+ EmpathyContact *actor,
+ guint reason,
+ const gchar *message,
+ EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ empathy_debug (DEBUG_DOMAIN, "Members added %s (%d)",
+ empathy_contact_get_id (contact),
+ empathy_contact_get_handle (contact));
+
+ if (!priv->contact) {
+ if (!empathy_contact_is_user (contact)) {
+ priv->is_incoming = TRUE;
+ priv->contact = g_object_ref (contact);
+ tp_call_set_status (call, EMPATHY_TP_CALL_STATUS_RINGING);
+ }
+ return;
+ }
+
+ /* We already have the other contact, that means we now have 2 members,
+ * so we can start the call */
+ tp_call_set_status (call, EMPATHY_TP_CALL_STATUS_RUNNING);
+}
+
+static void
+tp_call_remote_pending_cb (EmpathyTpGroup *group,
+ EmpathyContact *contact,
+ EmpathyContact *actor,
+ guint reason,
+ const gchar *message,
+ EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ empathy_debug (DEBUG_DOMAIN, "Remote pending: %s (%d)",
+ empathy_contact_get_id (contact),
+ empathy_contact_get_handle (contact));
+
+ if (!priv->contact) {
+ priv->is_incoming = FALSE;
+ priv->contact = g_object_ref (contact);
+ tp_call_set_status (call, EMPATHY_TP_CALL_STATUS_RINGING);
+ }
+}
+
+static void
+tp_call_async_cb (DBusGProxy *proxy,
+ GError *error,
+ gpointer user_data)
+{
+ if (error) {
+ empathy_debug (DEBUG_DOMAIN, "Failed to %s: %s",
+ user_data,
+ error->message);
+ }
+}
+
+static GObject *
+tp_call_constructor (GType type,
+ guint n_props,
+ GObjectConstructParam *props)
+{
+ GObject *call;
+ EmpathyTpCallPriv *priv;
+ TpConn *tp_conn;
+ MissionControl *mc;
+
+ call = G_OBJECT_CLASS (empathy_tp_call_parent_class)->constructor (type, n_props, props);
+ priv = GET_PRIV (call);
+
+ priv->group = empathy_tp_group_new (priv->account, priv->tp_chan);
+ priv->streamed_iface = tp_chan_get_interface (priv->tp_chan,
+ TELEPATHY_CHAN_IFACE_STREAMED_QUARK);
+
+ /* Connect signals */
+ dbus_g_proxy_connect_signal (priv->streamed_iface, "StreamAdded",
+ G_CALLBACK (tp_call_stream_added_cb),
+ call, NULL);
+ dbus_g_proxy_connect_signal (priv->streamed_iface, "StreamRemoved",
+ G_CALLBACK (tp_call_stream_removed_cb),
+ call, NULL);
+ dbus_g_proxy_connect_signal (DBUS_G_PROXY (priv->tp_chan), "Closed",
+ G_CALLBACK (tp_call_closed_cb),
+ call, NULL);
+ g_signal_connect (priv->tp_chan, "destroy",
+ G_CALLBACK (tp_call_destroy_cb),
+ call);
+ g_signal_connect (priv->group, "member-added",
+ G_CALLBACK (tp_call_member_added_cb),
+ call);
+ g_signal_connect (priv->group, "remote-pending",
+ G_CALLBACK (tp_call_remote_pending_cb),
+ call);
+
+ /* Start stream engine */
+ mc = empathy_mission_control_new ();
+ tp_conn = mission_control_get_connection (mc, priv->account, NULL);
+ priv->se_ch_proxy = dbus_g_proxy_new_for_name (tp_get_bus (),
+ STREAM_ENGINE_BUS_NAME,
+ STREAM_ENGINE_OBJECT_PATH,
+ CHANNEL_HANDLER_INTERFACE);
+ priv->se_proxy = dbus_g_proxy_new_for_name (tp_get_bus (),
+ STREAM_ENGINE_BUS_NAME,
+ STREAM_ENGINE_OBJECT_PATH,
+ STREAM_ENGINE_INTERFACE);
+ org_freedesktop_Telepathy_ChannelHandler_handle_channel_async (priv->se_ch_proxy,
+ dbus_g_proxy_get_bus_name (DBUS_G_PROXY (tp_conn)),
+ dbus_g_proxy_get_path (DBUS_G_PROXY (tp_conn)),
+ priv->tp_chan->type,
+ dbus_g_proxy_get_path (DBUS_G_PROXY (priv->tp_chan)),
+ priv->tp_chan->handle_type,
+ priv->tp_chan->handle,
+ tp_call_async_cb,
+ "handle channel");
+ g_object_unref (tp_conn);
+ g_object_unref (mc);
+
+ /* Get streams */
+ tp_chan_type_streamed_media_list_streams_async (priv->streamed_iface,
+ tp_call_list_streams_cb,
+ call);
+
+ return call;
+}
+
+static void
+tp_call_finalize (GObject *object)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (object);
+
+ empathy_debug (DEBUG_DOMAIN, "Finalizing: %p", object);
+
+ if (priv->tp_chan) {
+ GError *error;
+
+ g_signal_handlers_disconnect_by_func (priv->tp_chan,
+ tp_call_destroy_cb,
+ object);
+ empathy_debug (DEBUG_DOMAIN, "Closing channel...");
+ if (!tp_chan_close (DBUS_G_PROXY (priv->tp_chan), &error)) {
+ empathy_debug (DEBUG_DOMAIN,
+ "Error closing text channel: %s",
+ error ? error->message : "No error given");
+ g_clear_error (&error);
+ }
+ g_object_unref (priv->tp_chan);
+ }
+
+ g_object_unref (priv->group);
+ g_object_unref (priv->contact);
+ g_object_unref (priv->account);
+ g_object_unref (priv->se_ch_proxy);
+ g_object_unref (priv->se_proxy);
+
+ G_OBJECT_CLASS (empathy_tp_call_parent_class)->finalize (object);
+}
+
+static void
+empathy_tp_call_class_init (EmpathyTpCallClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->constructor = tp_call_constructor;
+ object_class->finalize = tp_call_finalize;
+ object_class->set_property = tp_call_set_property;
+ object_class->get_property = tp_call_get_property;
+
+ /* Construct-only properties */
+ g_object_class_install_property (object_class,
+ PROP_ACCOUNT,
+ g_param_spec_object ("account",
+ "channel Account",
+ "The account associated with the channel",
+ MC_TYPE_ACCOUNT,
+ G_PARAM_READWRITE |
+ G_PARAM_CONSTRUCT_ONLY));
+ g_object_class_install_property (object_class,
+ PROP_TP_CHAN,
+ g_param_spec_object ("tp-chan",
+ "telepathy channel",
+ "The media channel for the call",
+ TELEPATHY_CHAN_TYPE,
+ G_PARAM_READWRITE |
+ G_PARAM_CONSTRUCT_ONLY));
+
+ /* Normal properties */
+ g_object_class_install_property (object_class,
+ PROP_STATUS,
+ g_param_spec_enum ("status",
+ "call status",
+ "The status of the call",
+ EMPATHY_TYPE_TP_CALL_STATUS,
+ EMPATHY_TP_CALL_STATUS_PREPARING,
+ G_PARAM_READABLE));
+
+ /* Signals */
+ signals[DESTROY] =
+ g_signal_new ("destroy",
+ G_TYPE_FROM_CLASS (klass),
+ G_SIGNAL_RUN_LAST,
+ 0,
+ NULL, NULL,
+ g_cclosure_marshal_VOID__VOID,
+ G_TYPE_NONE,
+ 0);
+
+
+ g_type_class_add_private (klass, sizeof (EmpathyTpCallPriv));
+}
+
+static void
+empathy_tp_call_init (EmpathyTpCall *call)
+{
+}
+
+EmpathyTpCall *
+empathy_tp_call_new (McAccount *account, TpChan *channel)
+{
+ return g_object_new (EMPATHY_TYPE_TP_CALL,
+ "account", account,
+ "tp_chan", channel,
+ NULL);
+}
+
+gboolean
+empathy_tp_call_is_incoming (EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ return priv->is_incoming;
+}
+
+EmpathyTpCallStatus
+empathy_tp_call_get_status (EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ return priv->status;
+}
+
+EmpathyContact *
+empathy_tp_call_get_contact (EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ return priv->contact;
+}
+
+void
+empathy_tp_call_accept (EmpathyTpCall *call)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+ EmpathyContact *contact;
+
+ contact = empathy_tp_group_get_self_contact (priv->group);
+ empathy_tp_group_add_member (priv->group, contact, "");
+}
+
+void
+empathy_tp_call_invite (EmpathyTpCall *call,
+ EmpathyContact *contact)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ empathy_tp_group_add_member (priv->group, contact, "you're welcome");
+}
+
+void
+empathy_tp_call_request_streams (EmpathyTpCall *call,
+ gboolean audio,
+ gboolean video)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+ GArray *stream_types;
+ guint handle;
+ guint type;
+
+ empathy_debug (DEBUG_DOMAIN, "Requesting streams for audio=%s video=%s",
+ audio ? "Yes" : "No",
+ video ? "Yes" : "No");
+
+ stream_types = g_array_new (FALSE, FALSE, sizeof (guint));
+ if (audio) {
+ type = TP_MEDIA_STREAM_TYPE_AUDIO;
+ g_array_append_val (stream_types, type);
+ }
+ if (video) {
+ type = TP_MEDIA_STREAM_TYPE_VIDEO;
+ g_array_append_val (stream_types, type);
+ }
+
+ handle = empathy_contact_get_handle (priv->contact);
+ tp_chan_type_streamed_media_request_streams_async (priv->streamed_iface,
+ handle,
+ stream_types,
+ tp_call_list_streams_cb,
+ call);
+
+ g_array_free (stream_types, TRUE);
+}
+
+void
+empathy_tp_call_send_video (EmpathyTpCall *call,
+ gboolean send)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+ guint new_direction;
+
+ if (!priv->video_stream) {
+ return;
+ }
+
+ if (send) {
+ new_direction = TP_MEDIA_STREAM_DIRECTION_BIDIRECTIONAL;
+ } else {
+ new_direction = TP_MEDIA_STREAM_DIRECTION_RECEIVE;
+ }
+
+ tp_chan_type_streamed_media_request_stream_direction_async (priv->streamed_iface,
+ priv->video_stream,
+ new_direction,
+ tp_call_async_cb,
+ "request stream direction");
+}
+
+void
+empathy_tp_call_add_preview_window (EmpathyTpCall *call,
+ guint socket_id)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ org_freedesktop_Telepathy_StreamEngine_add_preview_window_async (priv->se_proxy,
+ socket_id,
+ tp_call_async_cb,
+ "add preview window");
+}
+
+void
+empathy_tp_call_remove_preview_window (EmpathyTpCall *call,
+ guint socket_id)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ org_freedesktop_Telepathy_StreamEngine_remove_preview_window_async (priv->se_proxy,
+ socket_id,
+ tp_call_async_cb,
+ "remove preview window");
+}
+
+void
+empathy_tp_call_set_output_window (EmpathyTpCall *call,
+ guint socket_id)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ org_freedesktop_Telepathy_StreamEngine_set_output_window_async (priv->se_proxy,
+ dbus_g_proxy_get_path (DBUS_G_PROXY (priv->tp_chan)),
+ priv->video_stream,
+ socket_id,
+ tp_call_async_cb,
+ "set output window");
+}
+
+void
+empathy_tp_call_set_output_volume (EmpathyTpCall *call,
+ guint volume)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ org_freedesktop_Telepathy_StreamEngine_set_output_volume_async (priv->se_proxy,
+ dbus_g_proxy_get_path (DBUS_G_PROXY (priv->tp_chan)),
+ priv->audio_stream,
+ volume,
+ tp_call_async_cb,
+ "set output volume");
+}
+
+
+void
+empathy_tp_call_mute_output (EmpathyTpCall *call,
+ gboolean is_muted)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ org_freedesktop_Telepathy_StreamEngine_mute_output_async (priv->se_proxy,
+ dbus_g_proxy_get_path (DBUS_G_PROXY (priv->tp_chan)),
+ priv->audio_stream,
+ is_muted,
+ tp_call_async_cb,
+ "mute output");
+}
+
+
+void
+empathy_tp_call_mute_input (EmpathyTpCall *call,
+ gboolean is_muted)
+{
+ EmpathyTpCallPriv *priv = GET_PRIV (call);
+
+ org_freedesktop_Telepathy_StreamEngine_mute_input_async (priv->se_proxy,
+ dbus_g_proxy_get_path (DBUS_G_PROXY (priv->tp_chan)),
+ priv->audio_stream,
+ is_muted,
+ tp_call_async_cb,
+ "mute output");
+}
+