/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 2 -*- */ /* MATE Volume Applet * Copyright (C) 2004 Ronald Bultje <rbultje@ronald.bitfreak.net> * * applet.c: the main applet * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 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 * Library General Public License for more details. * * You should have received a copy of the GNU Library 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. */ #ifdef HAVE_CONFIG_H #include "config.h" #endif /* this is for lrint */ #define _ISOC99_SOURCE #include <math.h> #include <string.h> #include <glib-object.h> #include <gdk/gdkkeysyms.h> #include <gtk/gtk.h> #include <gio/gio.h> #include "applet.h" #include "keys.h" #include "preferences.h" #define IS_PANEL_HORIZONTAL(o) \ (o == MATE_PANEL_APPLET_ORIENT_UP || o == MATE_PANEL_APPLET_ORIENT_DOWN) /* This is defined is load.c, we're doing this instead of creating a load.h file * because nothing else is exported. */ GList * mate_volume_applet_create_mixer_collection (void); static void mate_volume_applet_class_init (MateVolumeAppletClass *klass); static void mate_volume_applet_init (MateVolumeApplet *applet); static void mate_volume_applet_dispose (GObject *object); static void mate_volume_applet_size_allocate (GtkWidget *widget, GtkAllocation *allocation); static void mate_volume_applet_popup_dock (MateVolumeApplet *applet); static void mate_volume_applet_popdown_dock (MateVolumeApplet *applet); /* This function and mate_volume_applet_key are not static so we can * inject external events into the applet. Its to work around a GTK+ * misfeature. See dock.c for details. */ gboolean mate_volume_applet_scroll (GtkWidget *widget, GdkEventScroll *event); static gboolean mate_volume_applet_button (GtkWidget *widget, GdkEventButton *event); gboolean mate_volume_applet_key (GtkWidget *widget, GdkEventKey *event); static gdouble mate_volume_applet_get_volume (GstMixer *mixer, GstMixerTrack *track); static void mate_volume_applet_background (MatePanelApplet *mate_panel_applet, MatePanelAppletBackgroundType type, GdkColor *colour, GdkPixmap *pixmap); static void mate_volume_applet_orientation (MatePanelApplet *applet, MatePanelAppletOrient orient); static gboolean mate_volume_applet_refresh (MateVolumeApplet *applet, gboolean force_refresh, gdouble volume, gint mute); static void cb_notify_message (GstBus *bus, GstMessage *message, gpointer data); static gboolean cb_check (gpointer data); static void cb_volume (GtkAdjustment *adj, gpointer data); static void cb_gsettings (GSettings *settings, gchar *key, gpointer data); static void cb_verb (GtkAction *action, gpointer data); static void cb_theme_change (GtkIconTheme *icon_theme, gpointer data); static void cb_stop_scroll_events (GtkWidget *widget, GdkEvent *event); static MatePanelAppletClass *parent_class = NULL; G_DEFINE_TYPE (MateVolumeApplet, mate_volume_applet, PANEL_TYPE_APPLET) static void init_pixbufs (MateVolumeApplet *applet) { static const gchar *pix_filenames[] = { "audio-volume-muted", "audio-volume-low", "audio-volume-medium", "audio-volume-high", NULL }; gint n; for (n = 0; pix_filenames[n] != NULL; n++) { if (applet->pix[n]) { g_object_unref (applet->pix[n]); applet->pix[n] = NULL; // mate_icon_theme_load_icon can call us // recursively, so we have to be careful. } applet->pix[n] = gtk_icon_theme_load_icon (gtk_icon_theme_get_default (), pix_filenames[n], applet->panel_size - 4, 0, NULL); if (applet->pix[n] != NULL && gtk_widget_get_default_direction () == GTK_TEXT_DIR_RTL) { GdkPixbuf *temp; temp = gdk_pixbuf_flip (applet->pix[n], TRUE); g_object_unref (G_OBJECT (applet->pix[n])); applet->pix[n] = temp; } } } static void mate_volume_applet_class_init (MateVolumeAppletClass *klass) { GObjectClass *gobject_class = G_OBJECT_CLASS (klass); GtkWidgetClass *gtkwidget_class = GTK_WIDGET_CLASS (klass); MatePanelAppletClass *matepanelapplet_class = MATE_PANEL_APPLET_CLASS (klass); parent_class = g_type_class_ref (PANEL_TYPE_APPLET); gobject_class->dispose = mate_volume_applet_dispose; gtkwidget_class->key_press_event = mate_volume_applet_key; gtkwidget_class->button_press_event = mate_volume_applet_button; gtkwidget_class->scroll_event = mate_volume_applet_scroll; gtkwidget_class->size_allocate = mate_volume_applet_size_allocate; matepanelapplet_class->change_orient = mate_volume_applet_orientation; matepanelapplet_class->change_background = mate_volume_applet_background; /* FIXME: * - style-set. */ } static void mate_volume_applet_init (MateVolumeApplet *applet) { GtkWidget *image; AtkObject *ao; applet->timeout = 0; applet->elements = NULL; applet->settings = mate_panel_applet_settings_new (MATE_PANEL_APPLET (applet), "org.mate.panel.applet.mixer"); applet->mixer = NULL; applet->tracks = NULL; applet->lock = FALSE; applet->state = -1; applet->prefs = NULL; applet->dock = NULL; applet->adjustment = NULL; applet->panel_size = 24; g_set_application_name (_("Volume Applet")); /* init pixbufs */ init_pixbufs (applet); /* icon (our main UI) */ image = gtk_image_new (); applet->image = GTK_IMAGE (image); gtk_container_add (GTK_CONTAINER (applet), image); gtk_widget_show (image); gtk_window_set_default_icon_name ("multimedia-volume-control"); /* dock window (expanded UI) */ applet->pop = FALSE; /* tooltip over applet */ gtk_widget_set_tooltip_text (GTK_WIDGET (applet), _("Volume Control")); /* prevent scroll events from reaching the tooltip */ g_signal_connect (G_OBJECT (applet), "event-after", G_CALLBACK (cb_stop_scroll_events), NULL); /* handle icon theme changes */ g_signal_connect (gtk_icon_theme_get_default (), "changed", G_CALLBACK (cb_theme_change), applet); /* other stuff */ mate_panel_applet_set_flags (MATE_PANEL_APPLET (applet), MATE_PANEL_APPLET_EXPAND_MINOR); /* i18n */ ao = gtk_widget_get_accessible (GTK_WIDGET (applet)); atk_object_set_name (ao, _("Volume Control")); /* Watch for signals from GST. */ applet->bus = gst_bus_new (); gst_bus_add_signal_watch (applet->bus); g_signal_connect (G_OBJECT (applet->bus), "message::element", (GCallback) cb_notify_message, applet); } /* Parse the list of tracks that are stored in GSettings */ static char ** parse_track_list (const char *track_list) { if (track_list) return g_strsplit (track_list, ":", 0); else return NULL; } static GList * select_tracks (GstElement *element, const char *active_track_names, gboolean reset_state) { const GList *tracks, *l; GstMixerTrack *track_fallback; GList *active_tracks; char **active_track_name_list; active_tracks = NULL; track_fallback = NULL; active_track_name_list = NULL; if (reset_state) { gst_element_set_state (element, GST_STATE_READY); if (gst_element_get_state(element, NULL, NULL, -1) != GST_STATE_CHANGE_SUCCESS) return NULL; } tracks = gst_mixer_list_tracks (GST_MIXER (element)); if (active_track_names) active_track_name_list = parse_track_list (active_track_names); for (l = tracks; l; l = l->next) { GstMixerTrack *track = l->data; gint i; if (!track->num_channels) continue; if (!track_fallback) track_fallback = track; if (GST_MIXER_TRACK_HAS_FLAG (track, GST_MIXER_TRACK_MASTER)) track_fallback = track; if (active_track_name_list) { for (i = 0; active_track_name_list[i] != NULL; i++) { gchar *track_test = active_track_name_list[i]; if (!strcmp (track_test, track->label)) active_tracks = g_list_append (active_tracks, track); } } } /* if the list had no matches and we've got a fallback track, * then use it. */ if (!active_tracks && track_fallback) active_tracks = g_list_append (active_tracks, track_fallback); if (!active_tracks && reset_state) { gst_element_set_state (element, GST_STATE_NULL); } g_strfreev (active_track_name_list); return active_tracks; } static gboolean select_element_and_track (MateVolumeApplet *applet, GList *elements, const char *active_element_name, const char *active_track_names) { GstElement *active_element; GList *active_tracks, *l; applet->elements = elements; active_element = NULL; active_tracks = NULL; if (active_element_name) { for (l = elements; l; l = l->next) { GstElement *element = l->data; const char *element_name; element_name = g_object_get_data (G_OBJECT (element), "mate-volume-applet-name"); if (!strcmp (element_name, active_element_name)) { active_element = element; break; } } } if (active_element) active_tracks = select_tracks (active_element, active_track_names, TRUE); if (!active_tracks) { active_element = NULL; for (l = elements; l; l = l->next) { GstElement *element = l->data; if ((active_tracks = select_tracks (element, active_track_names, TRUE))) { active_element = element; break; } } } if (!active_element) return FALSE; applet->mixer = g_object_ref (active_element); gst_element_set_bus (GST_ELEMENT (applet->mixer), applet->bus); applet->tracks = active_tracks; g_list_foreach (applet->tracks, (GFunc) g_object_ref, NULL); return TRUE; } static void mate_volume_applet_setup_timeout (MateVolumeApplet *applet) { gboolean need_timeout = TRUE; need_timeout = ((gst_mixer_get_mixer_flags (GST_MIXER (applet->mixer)) & GST_MIXER_FLAG_AUTO_NOTIFICATIONS) == 0); if (need_timeout) { if (applet->timeout == 0) { applet->timeout = g_timeout_add (100, cb_check, applet); } } else { if (applet->timeout != 0) { g_source_remove (applet->timeout); applet->timeout = 0; } } } gboolean mate_volume_applet_setup (MateVolumeApplet *applet, GList *elements) { GtkObject *adj; static const GtkActionEntry actions[] = { { "RunMixer", NULL, N_("_Open Volume Control"), NULL, NULL, G_CALLBACK (cb_verb) }, { "Help", GTK_STOCK_HELP, N_("_Help"), NULL, NULL, G_CALLBACK (cb_verb) }, { "About", GTK_STOCK_ABOUT, N_("_About"), NULL, NULL, G_CALLBACK (cb_verb) }, { "Pref", GTK_STOCK_PROPERTIES, N_("_Preferences"), NULL, NULL, G_CALLBACK (cb_verb) } }; static const GtkToggleActionEntry toggle_actions[] = { { "Mute", NULL, N_("Mu_te"), NULL, NULL, G_CALLBACK (cb_verb), FALSE } }; gchar *active_element_name; gchar *active_track_name; gchar *ui_path; GstMixerTrack *first_track; gboolean res; active_element_name = g_settings_get_string (applet->settings, MATE_VOLUME_APPLET_KEY_ACTIVE_ELEMENT); active_track_name = g_settings_get_string (applet->settings, MATE_VOLUME_APPLET_KEY_ACTIVE_TRACK); res = select_element_and_track (applet, elements, active_element_name, active_track_name); g_free (active_element_name); g_free (active_track_name); if (res) { first_track = g_list_first (applet->tracks)->data; applet->adjustment = GTK_ADJUSTMENT (gtk_adjustment_new (50, 0, 100, 4, 10, 0)); /* We want a reference from the applet as well as from the dock it * will be attached to. */ g_object_ref_sink (applet->adjustment); g_signal_connect (applet->adjustment, "value-changed", G_CALLBACK (cb_volume), applet); gtk_adjustment_set_value (applet->adjustment, mate_volume_applet_get_volume (applet->mixer, first_track)); } mate_volume_applet_orientation (MATE_PANEL_APPLET (applet), mate_panel_applet_get_orient (MATE_PANEL_APPLET (applet))); /* menu */ applet->action_group = gtk_action_group_new ("Mixer Applet Actions"); gtk_action_group_set_translation_domain (applet->action_group, GETTEXT_PACKAGE); gtk_action_group_add_actions (applet->action_group, actions, G_N_ELEMENTS (actions), applet); gtk_action_group_add_toggle_actions (applet->action_group, toggle_actions, G_N_ELEMENTS (toggle_actions), applet); ui_path = g_build_filename (MIXER_MENU_UI_DIR, "mixer-applet-menu.xml", NULL); mate_panel_applet_setup_menu_from_file (MATE_PANEL_APPLET (applet), ui_path, applet->action_group); g_free (ui_path); mate_volume_applet_refresh (applet, TRUE, -1, -1); if (res) { mate_volume_applet_setup_timeout (applet); /* gsettings */ g_signal_connect (applet->settings, "changed::" MATE_VOLUME_APPLET_KEY_ACTIVE_ELEMENT, G_CALLBACK (cb_gsettings), applet); g_signal_connect (applet->settings, "changed::" MATE_VOLUME_APPLET_KEY_ACTIVE_TRACK, G_CALLBACK (cb_gsettings), applet); } gtk_widget_show (GTK_WIDGET (applet)); return TRUE; } static void mate_volume_applet_dispose (GObject *object) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (object); gint n; mate_volume_applet_popdown_dock (applet); if (applet->action_group) { g_object_unref (applet->action_group); applet->action_group = NULL; } if (applet->elements) { GList *item; for (item = applet->elements; item != NULL; item = item->next) { GstElement *element = GST_ELEMENT (item->data); gst_element_set_state (element, GST_STATE_NULL); gst_object_unref (GST_OBJECT (element)); } g_list_free (applet->elements); applet->elements = NULL; } if (applet->tracks) { g_list_foreach (applet->tracks, (GFunc) g_object_unref, NULL); g_list_free (applet->tracks); applet->tracks = NULL; } if (applet->mixer) { gst_object_unref (GST_OBJECT (applet->mixer)); applet->mixer = NULL; } if (applet->timeout) { g_source_remove (applet->timeout); applet->timeout = 0; } if (applet->dock) { g_object_unref (applet->dock); applet->dock = NULL; } if (applet->adjustment) { g_object_unref (applet->adjustment); applet->adjustment = NULL; } for (n = 0; n < 5; n++) { if (applet->pix[n] != NULL) { g_object_unref (G_OBJECT (applet->pix[n])); applet->pix[n] = NULL; } } if (applet->bus) { gst_bus_remove_signal_watch (applet->bus); gst_object_unref (applet->bus); applet->bus = NULL; } G_OBJECT_CLASS (parent_class)->dispose (object); } /* * Show a dialog (once) when no mixer is available. */ static void show_no_mixer_dialog (MateVolumeApplet *applet) { static gboolean shown = FALSE; GtkWidget *dialog; if (shown) return; shown = TRUE; dialog = gtk_message_dialog_new (NULL, 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, "%s\n\n%s", _("The volume control did not find any elements and/or " "devices to control. This means either that you don't " "have the right GStreamer plugins installed, or that you " "don't have a sound card configured."), _("You can remove the volume control from the panel by " "right-clicking the speaker icon on the panel and " "selecting \"Remove From Panel\" from the menu.")); gtk_widget_show (dialog); g_signal_connect (dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); } /* * get position that the dock should get based on applet position. */ static void mate_volume_applet_get_dock_position (MateVolumeApplet *applet, gint *_x, gint *_y) { GtkWidget *widget = GTK_WIDGET (applet); GtkAllocation widget_allocation, dock_allocation; gint x, y; gtk_widget_get_allocation (GTK_WIDGET (applet->dock), &dock_allocation); gtk_widget_get_allocation (widget, &widget_allocation); gdk_window_get_origin (gtk_widget_get_window (widget), &x, &y); switch (mate_panel_applet_get_orient (MATE_PANEL_APPLET (applet))) { case MATE_PANEL_APPLET_ORIENT_DOWN: x += widget_allocation.x; x -= (dock_allocation.width - widget_allocation.width) / 2; y += widget_allocation.height + widget_allocation.y; break; case MATE_PANEL_APPLET_ORIENT_UP: x += widget_allocation.x; x -= (dock_allocation.width - widget_allocation.width) / 2; y += widget_allocation.y; y -= dock_allocation.height; break; case MATE_PANEL_APPLET_ORIENT_RIGHT: x += widget_allocation.width + widget_allocation.x; y += widget_allocation.y; y -= (dock_allocation.height - widget_allocation.height) / 2; break; case MATE_PANEL_APPLET_ORIENT_LEFT: x += widget_allocation.x; x -= dock_allocation.width; y += widget_allocation.y; y -= (dock_allocation.height - widget_allocation.height) / 2; break; default: g_assert_not_reached (); } *_x = x; *_y = y; } /* * popup (show) or popdown (hide) the dock. */ static void mate_volume_applet_popup_dock (MateVolumeApplet *applet) { GtkWidget *widget = GTK_WIDGET (applet); gint x, y; /* Get it in just about the right position so that it * doesn't flicker to obviously when we reposition it. */ mate_volume_applet_get_dock_position (applet, &x, &y); gtk_window_move (GTK_WINDOW (applet->dock), x, y); gtk_widget_show_all (GTK_WIDGET (applet->dock)); /* Reposition the window now that we know its actual size * and can center it. */ mate_volume_applet_get_dock_position (applet, &x, &y); gtk_window_move (GTK_WINDOW (applet->dock), x, y); /* Set the keyboard focus in the correct place. */ mate_volume_applet_dock_set_focus (applet->dock); /* set menu item as active */ gtk_widget_set_state (GTK_WIDGET (applet), GTK_STATE_SELECTED); /* keep state */ applet->pop = TRUE; } static void mate_volume_applet_popdown_dock (MateVolumeApplet *applet) { GtkWidget *widget = GTK_WIDGET (applet); if (!applet->pop) return; /* hide */ gtk_widget_hide_all (GTK_WIDGET (applet->dock)); /* set menu item as active */ gtk_widget_set_state (GTK_WIDGET (applet), GTK_STATE_NORMAL); /* keep state */ applet->pop = FALSE; } static void mate_volume_applet_pop_dock (MateVolumeApplet *applet) { if (applet->pop) { mate_volume_applet_popdown_dock (applet); } else { mate_volume_applet_popup_dock (applet); } } static void mate_volume_applet_update_mute_action (MateVolumeApplet *applet, gboolean newmute) { GtkAction *action; if (!applet->action_group) return; action = gtk_action_group_get_action (applet->action_group, "Mute"); if (newmute == gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action))) return; gtk_action_block_activate (action); gtk_toggle_action_set_active (GTK_TOGGLE_ACTION (action), newmute); gtk_action_unblock_activate (action); } gboolean mixer_is_muted (MateVolumeApplet *applet) { return applet->state & 1; } /* * Toggle mute. */ void mate_volume_applet_toggle_mute (MateVolumeApplet *applet) { gboolean mute = mixer_is_muted (applet); gboolean newmute = !mute; GList *tracks; for (tracks = g_list_first (applet->tracks); tracks; tracks = tracks->next) gst_mixer_set_mute (applet->mixer, tracks->data, !mute); if (mute) { /* sync back actual volume */ cb_volume (applet->adjustment, applet); } /* update menu */ mate_volume_applet_update_mute_action (applet, newmute); /* update graphic - this should happen automagically after the next * idle call, but apparently doesn't for some people... */ mate_volume_applet_refresh (applet, TRUE, -1, newmute); } /* * Run g-v-c. */ void mate_volume_applet_run_mixer (MateVolumeApplet *applet) { GError *error = NULL; gdk_spawn_command_line_on_screen (gtk_widget_get_screen (GTK_WIDGET (applet)), "mate-volume-control", &error); if (error) { GtkWidget *dialog; dialog = gtk_message_dialog_new (NULL, 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, _("Failed to start Volume Control: %s"), error->message); g_signal_connect (dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); gtk_widget_show (dialog); g_error_free (error); } } /* * Control events, change volume and so on. */ /* This is not static so we can inject external events * into the applet. Its to work around a GTK+ misfeature. See dock.c * for details. */ gboolean mate_volume_applet_scroll (GtkWidget *widget, GdkEventScroll *event) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (widget); if (!applet->mixer) { show_no_mixer_dialog (applet); return TRUE; } if (event->type == GDK_SCROLL) { switch (event->direction) { case GDK_SCROLL_UP: case GDK_SCROLL_DOWN: { gdouble volume = gtk_adjustment_get_value (applet->adjustment); if (event->direction == GDK_SCROLL_UP) { volume += gtk_adjustment_get_step_increment (applet->adjustment); if (volume > gtk_adjustment_get_upper (applet->adjustment)) volume = gtk_adjustment_get_upper (applet->adjustment); } else { volume -= gtk_adjustment_get_step_increment (applet->adjustment); if (volume < gtk_adjustment_get_lower (applet->adjustment)) volume = gtk_adjustment_get_lower (applet->adjustment); } gtk_adjustment_set_value (applet->adjustment, volume); return TRUE; } default: break; } } if (GTK_WIDGET_CLASS (parent_class)->scroll_event) return GTK_WIDGET_CLASS (parent_class)->scroll_event (widget, event); else return FALSE; } static gboolean mate_volume_applet_button (GtkWidget *widget, GdkEventButton *event) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (widget); if (event->window != gtk_widget_get_window (GTK_WIDGET (applet)) && event->type == GDK_BUTTON_PRESS) { mate_volume_applet_popdown_dock (applet); return TRUE; } else if (event->window == gtk_widget_get_window (GTK_WIDGET (applet))) { switch (event->button) { case 1: switch (event->type) { case GDK_BUTTON_PRESS: if (!applet->mixer) { show_no_mixer_dialog (applet); } else { mate_volume_applet_pop_dock (applet); } return TRUE; case GDK_2BUTTON_PRESS: if (applet->mixer) { mate_volume_applet_popdown_dock (applet); } mate_volume_applet_toggle_mute (applet); return TRUE; default: break; } break; case 2: /* mute */ mate_volume_applet_toggle_mute (applet); return TRUE; case 3: /* menu */ if (applet->pop) { mate_volume_applet_popdown_dock (applet); return TRUE; } break; default: break; } } if (GTK_WIDGET_CLASS (parent_class)->button_press_event) return GTK_WIDGET_CLASS (parent_class)->button_press_event (widget, event); return FALSE; } /* This is not static so we can inject external events * into the applet. Its to work around a GTK+ misfeature. See dock.c * for details. */ gboolean mate_volume_applet_key (GtkWidget *widget, GdkEventKey *event) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (widget); if (!applet->mixer) { show_no_mixer_dialog (applet); } else switch (event->keyval) { case GDK_KP_Enter: case GDK_ISO_Enter: case GDK_3270_Enter: case GDK_Return: case GDK_space: case GDK_KP_Space: mate_volume_applet_pop_dock (applet); return TRUE; case GDK_m: if (event->state == GDK_CONTROL_MASK) { mate_volume_applet_toggle_mute (applet); return TRUE; } break; case GDK_o: if (event->state == GDK_CONTROL_MASK) { mate_volume_applet_run_mixer (applet); return TRUE; } break; case GDK_Escape: mate_volume_applet_popdown_dock (applet); return TRUE; case GDK_Page_Up: case GDK_Page_Down: case GDK_Left: case GDK_Right: case GDK_Up: case GDK_Down: { gdouble volume = gtk_adjustment_get_value (applet->adjustment); gdouble increment; if (event->state != 0) break; if (event->keyval == GDK_Up || event->keyval == GDK_Down ||event->keyval == GDK_Left) increment = gtk_adjustment_get_step_increment (applet->adjustment); else increment = gtk_adjustment_get_page_increment (applet->adjustment); if (event->keyval == GDK_Page_Up || event->keyval == GDK_Up ||event->keyval == GDK_Right) { volume += increment; if (volume > gtk_adjustment_get_upper (applet->adjustment)) volume = gtk_adjustment_get_upper (applet->adjustment); } else { volume -= increment; if (volume < gtk_adjustment_get_lower (applet->adjustment)) volume = gtk_adjustment_get_lower (applet->adjustment); } gtk_adjustment_set_value (applet->adjustment, volume); return TRUE; } default: break; } return GTK_WIDGET_CLASS (parent_class)->key_press_event (widget, event); } static gboolean mate_volume_applet_dock_focus_out (GtkWidget *dock, GdkEventFocus *event, MateVolumeApplet *applet) { mate_volume_applet_popdown_dock (applet); return FALSE; } /* * Change orientation or size of panel. */ static void mate_volume_applet_orientation (MatePanelApplet *_applet, MatePanelAppletOrient orientation) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (_applet); GtkWidget *dock; if (applet->dock) { gtk_widget_destroy (GTK_WIDGET (applet->dock)); } dock = mate_volume_applet_dock_new (GTK_ORIENTATION_VERTICAL, applet); g_object_ref_sink (dock); /* It isn't a child, but we do own it. */ gtk_widget_add_events (dock, GDK_FOCUS_CHANGE_MASK); g_signal_connect (G_OBJECT (dock), "focus-out-event", G_CALLBACK (mate_volume_applet_dock_focus_out), applet); applet->dock = MATE_VOLUME_APPLET_DOCK (dock); mate_volume_applet_dock_change (applet->dock, applet->adjustment); if (MATE_PANEL_APPLET_CLASS (parent_class)->change_orient) MATE_PANEL_APPLET_CLASS (parent_class)->change_orient (_applet, orientation); } void mate_volume_applet_size_allocate (GtkWidget *widget, GtkAllocation *allocation) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (widget); MatePanelAppletOrient orient; if (GTK_WIDGET_CLASS (parent_class)->size_allocate) GTK_WIDGET_CLASS (parent_class)->size_allocate (widget, allocation); orient = mate_panel_applet_get_orient (MATE_PANEL_APPLET (applet)); if (orient == MATE_PANEL_APPLET_ORIENT_UP || orient == MATE_PANEL_APPLET_ORIENT_DOWN) { if (applet->panel_size == allocation->height) return; applet->panel_size = allocation->height; } else { if (applet->panel_size == allocation->width) return; applet->panel_size = allocation->width; } init_pixbufs (applet); mate_volume_applet_refresh (applet, TRUE, -1, -1); } static void mate_volume_applet_background (MatePanelApplet *_applet, MatePanelAppletBackgroundType type, GdkColor *colour, GdkPixmap *pixmap) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (_applet); GtkRcStyle *rc_style; GtkStyle *style; /* reset style */ gtk_widget_set_style (GTK_WIDGET (applet), NULL); rc_style = gtk_rc_style_new (); gtk_widget_modify_style (GTK_WIDGET (applet), rc_style); g_object_unref (rc_style); switch (type) { case PANEL_NO_BACKGROUND: break; case PANEL_COLOR_BACKGROUND: gtk_widget_modify_bg (GTK_WIDGET (applet), GTK_STATE_NORMAL, colour); break; case PANEL_PIXMAP_BACKGROUND: style = gtk_style_copy (gtk_widget_get_style (GTK_WIDGET (applet))); if (style->bg_pixmap[GTK_STATE_NORMAL]) g_object_unref (style->bg_pixmap[GTK_STATE_NORMAL]); style->bg_pixmap[GTK_STATE_NORMAL] = g_object_ref (pixmap); gtk_widget_set_style (GTK_WIDGET (applet), style); g_object_unref (style); break; } } /* * This needs to be here because not all tracks have the same volume range, * so you can send this function the track and a new volume and it will be * scaled according to the volume range of the track in question. */ void mate_volume_applet_adjust_volume (GstMixer *mixer, GstMixerTrack *track, gdouble volume) { int range = track->max_volume - track->min_volume; gdouble scale = ((gdouble) range) / 100; int *volumes, n, volint; if (volume == 1.0) { volint = track->max_volume; } else if (volume == 0.0) { volint = track->min_volume; } else { volume *= scale; volume += track->min_volume; volint = lrint (volume); } volumes = g_new (gint, track->num_channels); for (n = 0; n < track->num_channels; n++) volumes[n] = volint; gst_mixer_set_volume (mixer, track, volumes); g_free (volumes); } static gdouble mate_volume_applet_get_volume (GstMixer *mixer, GstMixerTrack *track) { int *volumes, n; gdouble j; if (!track || !mixer) return -1; volumes = g_new (gint, track->num_channels); gst_mixer_get_volume (mixer, track, volumes); j = 0; for (n = 0; n < track->num_channels; n++) j += volumes[n]; g_free (volumes); j /= track->num_channels; return 100 * (j - track->min_volume) / (track->max_volume - track->min_volume); } /* * Volume changed. */ static void cb_volume (GtkAdjustment *adj, gpointer data) { MateVolumeApplet *applet = data; gdouble v; GList *iter; if (applet->lock) return; applet->lock = TRUE; v = gtk_adjustment_get_value (adj); for (iter = g_list_first (applet->tracks); iter; iter = iter->next) { GstMixerTrack *track = iter->data; mate_volume_applet_adjust_volume (applet->mixer, track, v); } applet->lock = FALSE; applet->force_next_update = TRUE; mate_volume_applet_refresh (MATE_VOLUME_APPLET (data), FALSE, v, -1); } /* * Automatic timer. Check for changes. */ #define STATE(vol,m) (((gint) vol << 1) | (m ? 1 : 0)) static gboolean mate_volume_applet_refresh (MateVolumeApplet *applet, gboolean force_refresh, gdouble volume, gint mute) { GdkPixbuf *pixbuf; gint n; gboolean show_mute, did_change; gchar *tooltip_str; GstMixerTrack *first_track; GString *track_names; GList *iter; show_mute = 0; if (!applet->mixer) { n = 0; show_mute = 0; mute = 0; } else if (!applet->tracks) { return FALSE; } else { first_track = g_list_first (applet->tracks)->data; if (volume == -1) { /* only first track */ volume = mate_volume_applet_get_volume (applet->mixer, first_track); } if (mute == -1) { mute = GST_MIXER_TRACK_HAS_FLAG (first_track, GST_MIXER_TRACK_MUTE) ? 1 : 0; } if (volume <= 0 || mute) { show_mute = 1; n = 0; } else { /* select image */ n = 3 * volume / 100 + 1; if (n < 1) n = 1; if (n > 3) n = 3; } } did_change = (force_refresh || (STATE (volume, mute) != applet->state) || applet->force_next_update); applet->force_next_update = FALSE; if (did_change) { if (show_mute) { pixbuf = applet->pix[0]; } else { pixbuf = applet->pix[n]; } gtk_image_set_from_pixbuf (applet->image, pixbuf); applet->state = STATE (volume, mute); if (applet->dock) { gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (applet->dock->mute), mute); } } if (!did_change || !applet->mixer) return did_change; /* build names of selecter tracks */ track_names = g_string_new (""); for (iter = g_list_first (applet->tracks); iter; iter = iter->next) { GstMixerTrack *track = iter->data; if (iter->next != NULL) g_string_append_printf (track_names, "%s / ", track->label); else track_names = g_string_append (track_names, track->label); } if (show_mute) { tooltip_str = g_strdup_printf (_("%s: muted"), track_names->str); } else { /* Translator comment: I'm not all too sure if this makes sense * to mark as a translation, but anyway. The string is a list of * selected tracks, the number is the volume in percent. You * most likely want to keep this as-is. */ tooltip_str = g_strdup_printf (_("%s: %d%%"), track_names->str, (int) volume); } g_string_free (track_names, TRUE); gtk_widget_set_tooltip_text (GTK_WIDGET (applet), tooltip_str); g_free (tooltip_str); applet->lock = TRUE; if (volume != 0) { gtk_adjustment_set_value (applet->adjustment, volume); } applet->lock = FALSE; /* update mute status */ mate_volume_applet_update_mute_action (applet, mute); return did_change; } static void cb_notify_message (GstBus *bus, GstMessage *message, gpointer data) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (data); GstMixerMessageType type; GstMixerTrack *first_track; GstMixerTrack *track = NULL; gint mute; gdouble volume; if (applet->tracks == NULL || GST_MESSAGE_SRC (message) != GST_OBJECT (applet->mixer)) { /* No tracks, or not from our mixer - can't update anything anyway */ return; } volume = mute = -1; first_track = g_list_first (applet->tracks)->data; /* This code only calls refresh if the first_track changes, because the * refresh code only retrieves the current value from that track anyway */ type = gst_mixer_message_get_type (message); if (type == GST_MIXER_MESSAGE_MUTE_TOGGLED) { gboolean muted; gst_mixer_message_parse_mute_toggled (message, &track, &muted); mute = muted ? 1 : 0; } else if (type == GST_MIXER_MESSAGE_VOLUME_CHANGED) { gint n, num_channels, *vols; volume = 0.0; gst_mixer_message_parse_volume_changed (message, &track, &vols, &num_channels); for (n = 0; n < num_channels; n++) volume += vols[n]; volume /= track->num_channels; volume = 100 * volume / (track->max_volume - track->min_volume); } else { return; } if (first_track == track) mate_volume_applet_refresh (MATE_VOLUME_APPLET (data), FALSE, volume, mute); } static gboolean cb_check (gpointer data) { static int time_counter = -1; static int timeout = 15; static gboolean recent_change = FALSE; gboolean did_change; time_counter++; /* * This timeout is called 10 times per second. Only do the update every * 15 times this function is called (every 1.5 seconds), unless the value * actually changed. */ if (time_counter % timeout == 0 || recent_change) { did_change = mate_volume_applet_refresh (MATE_VOLUME_APPLET (data), FALSE, -1, -1); /* * If a change was done, set recent_change so that the update is * done 10 times a second for 10 seconds and reset the counter to 0. * This way we update frequently for 10 seconds after the last time * the value is actually changed. */ if (did_change) { recent_change = TRUE; time_counter = 0; timeout = 100; } else if (time_counter % timeout == 0) { /* * When the counter gets to the timeout, reset recent_change and * time_counter so we go back to the behavior where we only check * every 15 times the function is called. */ recent_change = FALSE; time_counter = 0; timeout = 15; } } return TRUE; } /* * GSettings callback. */ static void cb_gsettings (GSettings *settings, gchar *key, gpointer data) { MateVolumeApplet *applet = data; const gchar *str; const GList *item; gboolean newdevice = FALSE; GList *active_tracks; active_tracks = NULL; g_list_free(applet->elements); applet->elements = mate_volume_applet_create_mixer_collection (); if (!strcmp (key, MATE_VOLUME_APPLET_KEY_ACTIVE_ELEMENT)) { for (item = applet->elements; item != NULL; item = item->next) { gchar *cur_el_str = g_object_get_data (item->data, "mate-volume-applet-name"); if (!strcmp (cur_el_str, str)) { GstElement *old_element = GST_ELEMENT (applet->mixer), *new_element = item->data; if (new_element != old_element) { /* change element */ gst_element_set_state (item->data, GST_STATE_READY); if (gst_element_get_state (item->data, NULL, NULL, -1) != GST_STATE_CHANGE_SUCCESS) continue; /* save */ gst_object_replace ((GstObject **) &applet->mixer, item->data); gst_element_set_state (old_element, GST_STATE_NULL); mate_volume_applet_setup_timeout (applet); newdevice = TRUE; } break; } } } if (!strcmp (key, MATE_VOLUME_APPLET_KEY_ACTIVE_TRACK) || newdevice) { if (!active_tracks) { active_tracks = select_tracks (GST_ELEMENT (applet->mixer), str, FALSE); } if (active_tracks) { GstMixerTrack *first_track; /* copy the newly created track list over to the main list */ g_list_free (applet->tracks); applet->tracks = g_list_copy (active_tracks); first_track = g_list_first (active_tracks)->data; /* dock */ gtk_adjustment_set_value (applet->adjustment, mate_volume_applet_get_volume (applet->mixer, first_track)); /* if preferences window is open, update */ if (applet->prefs) { mate_volume_applet_preferences_change ( MATE_VOLUME_APPLET_PREFERENCES (applet->prefs), applet->mixer, applet->tracks); } applet->force_next_update = TRUE; } } } /* * verb callback. */ static void cb_prefs_destroy (GtkWidget *widget, gpointer data) { MATE_VOLUME_APPLET (data)->prefs = NULL; } static void cb_verb (GtkAction *action, gpointer data) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (data); const gchar *verbname = gtk_action_get_name (action); if (!strcmp (verbname, "RunMixer")) { mate_volume_applet_run_mixer (applet); } else if (!strcmp (verbname, "Help")) { GError *error = NULL; gtk_show_uri (gtk_widget_get_screen (GTK_WIDGET (applet)), "help:mixer_applet2", gtk_get_current_event_time (), &error); if (error) { GtkWidget *dialog; dialog = gtk_message_dialog_new (NULL, 0, GTK_MESSAGE_ERROR, GTK_BUTTONS_CLOSE, _("Failed to display help: %s"), error->message); g_signal_connect (dialog, "response", G_CALLBACK (gtk_widget_destroy), NULL); gtk_widget_show (dialog); g_error_free (error); } } else if (!strcmp (verbname, "About")) { const gchar *authors[] = { "Ronald Bultje <rbultje@ronald.bitfreak.net>", NULL }; char *comments = g_strdup_printf ("%s\n\n%s", _("Volume control for your MATE Panel."), _("Using GStreamer 0.10.") ); gtk_show_about_dialog (NULL, "version", VERSION, "copyright", "\xC2\xA9 2004 Ronald Bultje", "comments", comments, "authors", authors, "translator-credits", _("translator-credits"), "logo-icon-name", "multimedia-volume-control", NULL); g_free (comments); } else if (!strcmp (verbname, "Pref")) { if (!applet->mixer) { show_no_mixer_dialog (applet); } else { if (applet->prefs) return; g_list_free(applet->elements); applet->elements = mate_volume_applet_create_mixer_collection (); applet->prefs = mate_volume_applet_preferences_new (applet, applet->elements, applet->mixer, applet->tracks); g_signal_connect (applet->prefs, "destroy", G_CALLBACK (cb_prefs_destroy), applet); gtk_widget_show (applet->prefs); } } else if (!strcmp (verbname, "Mute")) { if (!applet->mixer) { show_no_mixer_dialog (applet); } else { gboolean mute = applet->state & 1, want_mute = gtk_toggle_action_get_active (GTK_TOGGLE_ACTION (action)); if (mute != want_mute) mate_volume_applet_toggle_mute (applet); } } else { g_warning ("Unknown menu action '%s'", verbname); } } static void cb_theme_change (GtkIconTheme *icon_theme, gpointer data) { MateVolumeApplet *applet = MATE_VOLUME_APPLET (data); init_pixbufs (applet); mate_volume_applet_refresh (applet, TRUE, -1, -1); } /* * Block the tooltips event-after handler on scroll events. */ static void cb_stop_scroll_events (GtkWidget *widget, GdkEvent *event) { if (event->type == GDK_SCROLL) g_signal_stop_emission_by_name (widget, "event-after"); }