/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- * * Copyright (C) 2018 Canonical Ltd. * * 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 * Lesser General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, see . */ #include #include #include #include "cc-stream-list-box.h" #include "cc-stream-row.h" #include "cc-sound-enums.h" struct _CcStreamListBox { GtkListBox parent_instance; GtkSizeGroup *label_size_group; GvcMixerControl *mixer_control; CcStreamType stream_type; guint stream_added_handler_id; guint stream_removed_handler_id; }; G_DEFINE_TYPE (CcStreamListBox, cc_stream_list_box, GTK_TYPE_LIST_BOX) enum { PROP_0, PROP_LABEL_SIZE_GROUP }; static gint sort_cb (GtkListBoxRow *row1, GtkListBoxRow *row2, gpointer user_data) { CcStreamListBox *self = user_data; GvcMixerStream *stream1, *stream2, *event_sink; g_autofree gchar *name1 = NULL; g_autofree gchar *name2 = NULL; stream1 = cc_stream_row_get_stream (CC_STREAM_ROW (row1)); stream2 = cc_stream_row_get_stream (CC_STREAM_ROW (row2)); /* Put the system sound events control at the top */ event_sink = gvc_mixer_control_get_event_sink_input (self->mixer_control); if (stream1 == event_sink) return -1; else if (stream2 == event_sink) return 1; name1 = g_utf8_casefold (gvc_mixer_stream_get_name (stream1), -1); name2 = g_utf8_casefold (gvc_mixer_stream_get_name (stream2), -1); return g_strcmp0 (name1, name2); } static void stream_added_cb (CcStreamListBox *self, guint id) { GvcMixerStream *stream; const gchar *app_id; CcStreamRow *row; stream = gvc_mixer_control_lookup_stream_id (self->mixer_control, id); if (stream == NULL) return; app_id = gvc_mixer_stream_get_application_id (stream); /* Skip master volume controls */ if (g_strcmp0 (app_id, "org.gnome.VolumeControl") == 0 || g_strcmp0 (app_id, "org.PulseAudio.pavucontrol") == 0) { return; } /* Skip streams that aren't volume controls */ if (GVC_IS_MIXER_SOURCE (stream) || GVC_IS_MIXER_SINK (stream) || gvc_mixer_stream_is_virtual (stream) || gvc_mixer_stream_is_event_stream (stream)) { return; } row = cc_stream_row_new (self->label_size_group, stream, id, self->stream_type, self->mixer_control); gtk_widget_show (GTK_WIDGET (row)); gtk_list_box_row_set_activatable (GTK_LIST_BOX_ROW (row), FALSE); gtk_container_add (GTK_CONTAINER (self), GTK_WIDGET (row)); } static CcStreamRow * find_row (CcStreamListBox *self, guint id) { g_autoptr(GList) children = NULL; GList *link; children = gtk_container_get_children (GTK_CONTAINER (self)); for (link = children; link; link = link->next) { CcStreamRow *row = link->data; if (!CC_IS_STREAM_ROW (row)) continue; if (id == cc_stream_row_get_id (row)) return row; } return NULL; } static void stream_removed_cb (CcStreamListBox *self, guint id) { CcStreamRow *row; row = find_row (self, id); if (row != NULL) gtk_container_remove (GTK_CONTAINER (self), GTK_WIDGET (row)); } static void cc_stream_list_box_set_property (GObject *object, guint property_id, const GValue *value, GParamSpec *pspec) { CcStreamListBox *self = CC_STREAM_LIST_BOX (object); switch (property_id) { case PROP_LABEL_SIZE_GROUP: self->label_size_group = g_value_dup_object (value); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } } static void cc_stream_list_box_get_property (GObject *object, guint property_id, GValue *value, GParamSpec *pspec) { CcStreamListBox *self = CC_STREAM_LIST_BOX (object); switch (property_id) { case PROP_LABEL_SIZE_GROUP: g_value_set_object (value, self->label_size_group); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); } } static void cc_stream_list_box_dispose (GObject *object) { CcStreamListBox *self = CC_STREAM_LIST_BOX (object); g_clear_object (&self->mixer_control); G_OBJECT_CLASS (cc_stream_list_box_parent_class)->dispose (object); } void cc_stream_list_box_class_init (CcStreamListBoxClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->set_property = cc_stream_list_box_set_property; object_class->get_property = cc_stream_list_box_get_property; object_class->dispose = cc_stream_list_box_dispose; g_object_class_install_property (object_class, PROP_LABEL_SIZE_GROUP, g_param_spec_object ("label-size-group", NULL, NULL, GTK_TYPE_SIZE_GROUP, G_PARAM_READWRITE | G_PARAM_CONSTRUCT)); } void cc_stream_list_box_init (CcStreamListBox *self) { gtk_list_box_set_selection_mode (GTK_LIST_BOX (self), GTK_SELECTION_NONE); gtk_list_box_set_sort_func (GTK_LIST_BOX (self), sort_cb, self, NULL); } void cc_stream_list_box_set_mixer_control (CcStreamListBox *self, GvcMixerControl *mixer_control) { g_return_if_fail (CC_IS_STREAM_LIST_BOX (self)); if (self->mixer_control != NULL) { g_signal_handler_disconnect (self->mixer_control, self->stream_added_handler_id); self->stream_added_handler_id = 0; g_signal_handler_disconnect (self->mixer_control, self->stream_removed_handler_id); self->stream_removed_handler_id = 0; } g_clear_object (&self->mixer_control); self->mixer_control = g_object_ref (mixer_control); self->stream_added_handler_id = g_signal_connect_object (self->mixer_control, "stream-added", G_CALLBACK (stream_added_cb), self, G_CONNECT_SWAPPED); self->stream_removed_handler_id = g_signal_connect_object (self->mixer_control, "stream-removed", G_CALLBACK (stream_removed_cb), self, G_CONNECT_SWAPPED); } void cc_stream_list_box_set_stream_type (CcStreamListBox *self, CcStreamType stream_type) { g_return_if_fail (CC_IS_STREAM_LIST_BOX (self)); self->stream_type = stream_type; }