summaryrefslogtreecommitdiff
path: root/mate-volume-control/gvc-status-icon.c
diff options
context:
space:
mode:
Diffstat (limited to 'mate-volume-control/gvc-status-icon.c')
-rw-r--r--mate-volume-control/gvc-status-icon.c365
1 files changed, 365 insertions, 0 deletions
diff --git a/mate-volume-control/gvc-status-icon.c b/mate-volume-control/gvc-status-icon.c
new file mode 100644
index 0000000..0363888
--- /dev/null
+++ b/mate-volume-control/gvc-status-icon.c
@@ -0,0 +1,365 @@
+/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*-
+ *
+ * Copyright (C) 2008 Red Hat, Inc.
+ * Copyright (C) 2014 Michal Ratajsky <[email protected]>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program 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 General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301, USA.
+ *
+ */
+
+#include "config.h"
+
+#include <string.h>
+#include <glib.h>
+#include <glib/gi18n.h>
+#include <glib-object.h>
+#include <gtk/gtk.h>
+
+#include <libmatemixer/matemixer.h>
+
+#include "gvc-status-icon.h"
+#include "gvc-stream-status-icon.h"
+
+#define GVC_STATUS_ICON_GET_PRIVATE(o) (G_TYPE_INSTANCE_GET_PRIVATE ((o), GVC_TYPE_STATUS_ICON, GvcStatusIconPrivate))
+
+static const gchar *icon_names_output[] = {
+ "audio-volume-muted",
+ "audio-volume-low",
+ "audio-volume-medium",
+ "audio-volume-high",
+ NULL
+};
+
+static const gchar *icon_names_input[] = {
+ "audio-input-microphone-muted",
+ "audio-input-microphone-low",
+ "audio-input-microphone-medium",
+ "audio-input-microphone-high",
+ NULL
+};
+
+struct _GvcStatusIconPrivate
+{
+ GvcStreamStatusIcon *icon_input;
+ GvcStreamStatusIcon *icon_output;
+ gboolean running;
+ MateMixerContext *context;
+ MateMixerStream *input;
+};
+
+static void gvc_status_icon_class_init (GvcStatusIconClass *klass);
+static void gvc_status_icon_init (GvcStatusIcon *status_icon);
+
+G_DEFINE_TYPE (GvcStatusIcon, gvc_status_icon, G_TYPE_OBJECT)
+
+static void
+update_icon_input (GvcStatusIcon *status_icon)
+{
+ MateMixerStreamControl *control = NULL;
+ gboolean show = FALSE;
+
+ /* Enable the input icon in case there is an input stream present and there
+ * is a non-mixer application using the input */
+ if (status_icon->priv->input != NULL) {
+ const gchar *app_id;
+ const GList *inputs =
+ mate_mixer_stream_list_controls (status_icon->priv->input);
+
+ control = mate_mixer_stream_get_default_control (status_icon->priv->input);
+
+ const gchar *stream_name =
+ mate_mixer_stream_get_name (status_icon->priv->input);
+ g_debug ("Got stream name %s", stream_name);
+ if (g_str_has_suffix (stream_name, ".monitor")) {
+ inputs = NULL;
+ g_debug ("Stream is a monitor, ignoring");
+ }
+
+ while (inputs != NULL) {
+ MateMixerStreamControl *input =
+ MATE_MIXER_STREAM_CONTROL (inputs->data);
+ MateMixerStreamControlRole role =
+ mate_mixer_stream_control_get_role (input);
+
+ if (role == MATE_MIXER_STREAM_CONTROL_ROLE_APPLICATION) {
+ MateMixerAppInfo *app_info =
+ mate_mixer_stream_control_get_app_info (input);
+
+ app_id = mate_mixer_app_info_get_id (app_info);
+ if (app_id == NULL) {
+ /* A recording application which has no
+ * identifier set */
+ g_debug ("Found a recording application control %s",
+ mate_mixer_stream_control_get_label (input));
+
+ if G_UNLIKELY (control == NULL) {
+ /* In the unlikely case when there is no
+ * default input control, use the application
+ * control for the icon */
+ control = input;
+ }
+ show = TRUE;
+ break;
+ }
+
+ if (strcmp (app_id, "org.mate.VolumeControl") != 0 &&
+ strcmp (app_id, "org.gnome.VolumeControl") != 0 &&
+ strcmp (app_id, "org.PulseAudio.pavucontrol") != 0) {
+ g_debug ("Found a recording application %s", app_id);
+
+ if G_UNLIKELY (control == NULL)
+ control = input;
+
+ show = TRUE;
+ break;
+ }
+ }
+ inputs = inputs->next;
+ }
+
+ if (show == TRUE)
+ g_debug ("Input icon enabled");
+ else
+ g_debug ("There is no recording application, input icon disabled");
+ }
+
+ gvc_stream_status_icon_set_control (status_icon->priv->icon_input, control);
+
+ gtk_status_icon_set_visible (GTK_STATUS_ICON (status_icon->priv->icon_input), show);
+}
+
+static void
+update_icon_output (GvcStatusIcon *status_icon)
+{
+ MateMixerStream *stream;
+ MateMixerStreamControl *control = NULL;
+
+ stream = mate_mixer_context_get_default_output_stream (status_icon->priv->context);
+ if (stream != NULL)
+ control = mate_mixer_stream_get_default_control (stream);
+
+ gvc_stream_status_icon_set_control (status_icon->priv->icon_output, control);
+
+ if (control != NULL) {
+ g_debug ("Output icon enabled");
+ gtk_status_icon_set_visible (GTK_STATUS_ICON (status_icon->priv->icon_output),
+ TRUE);
+ }
+ else {
+ g_debug ("There is no output stream/control, output icon disabled");
+ gtk_status_icon_set_visible (GTK_STATUS_ICON (status_icon->priv->icon_output),
+ FALSE);
+ }
+}
+
+static void
+on_input_stream_control_added (MateMixerStream *stream,
+ const gchar *name,
+ GvcStatusIcon *status_icon)
+{
+ MateMixerStreamControl *control;
+
+ control = mate_mixer_stream_get_control (stream, name);
+ if G_LIKELY (control != NULL) {
+ MateMixerStreamControlRole role =
+ mate_mixer_stream_control_get_role (control);
+
+ /* Non-application input control doesn't affect the icon */
+ if (role != MATE_MIXER_STREAM_CONTROL_ROLE_APPLICATION)
+ return;
+ }
+
+ /* Either an application control has been added or we couldn't
+ * read the control, this shouldn't happen but let's revalidate the
+ * icon to be sure if it does */
+ update_icon_input (status_icon);
+}
+
+static void
+on_input_stream_control_removed (MateMixerStream *stream,
+ const gchar *name,
+ GvcStatusIcon *status_icon)
+{
+ /* The removed stream could be an application input, which may cause
+ * the input status icon to disappear */
+ update_icon_input (status_icon);
+}
+
+static gboolean
+update_default_input_stream (GvcStatusIcon *status_icon)
+{
+ MateMixerStream *stream;
+
+ stream = mate_mixer_context_get_default_input_stream (status_icon->priv->context);
+ if (stream == status_icon->priv->input)
+ return FALSE;
+
+ /* The input stream has changed */
+ if (status_icon->priv->input != NULL) {
+ g_signal_handlers_disconnect_by_data (G_OBJECT (status_icon->priv->input),
+ status_icon);
+ g_object_unref (status_icon->priv->input);
+ }
+
+ status_icon->priv->input = (stream == NULL) ? NULL : g_object_ref (stream);
+ if (status_icon->priv->input != NULL) {
+ g_signal_connect (G_OBJECT (status_icon->priv->input),
+ "control-added",
+ G_CALLBACK (on_input_stream_control_added),
+ status_icon);
+ g_signal_connect (G_OBJECT (status_icon->priv->input),
+ "control-removed",
+ G_CALLBACK (on_input_stream_control_removed),
+ status_icon);
+ }
+
+ /* Return TRUE if the default input stream has changed */
+ return TRUE;
+}
+
+static void
+on_context_state_notify (MateMixerContext *context,
+ GParamSpec *pspec,
+ GvcStatusIcon *status_icon)
+{
+ MateMixerState state = mate_mixer_context_get_state (context);
+
+ switch (state) {
+ case MATE_MIXER_STATE_FAILED:
+ g_warning ("Failed to connect to a sound system");
+ break;
+
+ case MATE_MIXER_STATE_READY:
+ update_default_input_stream (status_icon);
+
+ /* Each status change may affect the visibility of the icons */
+ update_icon_output (status_icon);
+ update_icon_input (status_icon);
+ break;
+ default:
+ break;
+ }
+}
+
+static void
+on_context_default_input_stream_notify (MateMixerContext *context,
+ GParamSpec *pspec,
+ GvcStatusIcon *status_icon)
+{
+ if (update_default_input_stream (status_icon) == FALSE)
+ return;
+
+ update_icon_input (status_icon);
+}
+
+static void
+on_context_default_output_stream_notify (MateMixerContext *control,
+ GParamSpec *pspec,
+ GvcStatusIcon *status_icon)
+{
+ update_icon_output (status_icon);
+}
+
+void
+gvc_status_icon_start (GvcStatusIcon *status_icon)
+{
+ g_return_if_fail (GVC_IS_STATUS_ICON (status_icon));
+
+ if G_UNLIKELY (status_icon->priv->running == TRUE)
+ return;
+
+ if G_UNLIKELY (mate_mixer_context_open (status_icon->priv->context) == FALSE) {
+ /* Normally this should never happen, in the worst case we
+ * should end up with the Null module */
+ g_warning ("Failed to connect to a sound system");
+ }
+
+ g_debug ("StatusIcon has been started");
+
+ status_icon->priv->running = TRUE;
+}
+
+static void
+gvc_status_icon_dispose (GObject *object)
+{
+ GvcStatusIcon *status_icon = GVC_STATUS_ICON (object);
+
+ if (status_icon->priv->input != NULL) {
+ g_signal_handlers_disconnect_by_data (G_OBJECT (status_icon->priv->input),
+ status_icon);
+ g_clear_object (&status_icon->priv->input);
+ }
+
+ g_clear_object (&status_icon->priv->context);
+ g_clear_object (&status_icon->priv->icon_input);
+ g_clear_object (&status_icon->priv->icon_output);
+
+ G_OBJECT_CLASS (gvc_status_icon_parent_class)->dispose (object);
+}
+
+static void
+gvc_status_icon_class_init (GvcStatusIconClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->dispose = gvc_status_icon_dispose;
+
+ g_type_class_add_private (klass, sizeof (GvcStatusIconPrivate));
+}
+
+static void
+gvc_status_icon_init (GvcStatusIcon *status_icon)
+{
+ status_icon->priv = GVC_STATUS_ICON_GET_PRIVATE (status_icon);
+
+ status_icon->priv->icon_input = gvc_stream_status_icon_new (NULL, icon_names_input);
+ status_icon->priv->icon_output = gvc_stream_status_icon_new (NULL, icon_names_output);
+
+ gvc_stream_status_icon_set_display_name (status_icon->priv->icon_input, _("Input"));
+ gvc_stream_status_icon_set_display_name (status_icon->priv->icon_output, _("Output"));
+
+ gtk_status_icon_set_title (GTK_STATUS_ICON (status_icon->priv->icon_input),
+ _("Microphone Volume"));
+ gtk_status_icon_set_title (GTK_STATUS_ICON (status_icon->priv->icon_output),
+ _("Sound Output Volume"));
+
+ status_icon->priv->context = mate_mixer_context_new ();
+
+ mate_mixer_context_set_app_name (status_icon->priv->context,
+ _("MATE Volume Control StatusIcon"));
+
+ mate_mixer_context_set_app_id (status_icon->priv->context, GVC_STATUS_ICON_DBUS_NAME);
+ mate_mixer_context_set_app_version (status_icon->priv->context, VERSION);
+ mate_mixer_context_set_app_icon (status_icon->priv->context, "multimedia-volume-control");
+
+ g_signal_connect (G_OBJECT (status_icon->priv->context),
+ "notify::state",
+ G_CALLBACK (on_context_state_notify),
+ status_icon);
+ g_signal_connect (G_OBJECT (status_icon->priv->context),
+ "notify::default-input-stream",
+ G_CALLBACK (on_context_default_input_stream_notify),
+ status_icon);
+ g_signal_connect (G_OBJECT (status_icon->priv->context),
+ "notify::default-output-stream",
+ G_CALLBACK (on_context_default_output_stream_notify),
+ status_icon);
+}
+
+GvcStatusIcon *
+gvc_status_icon_new (void)
+{
+ return g_object_new (GVC_TYPE_STATUS_ICON, NULL);
+}