aboutsummaryrefslogtreecommitdiff
path: root/src/volume-control.vala
diff options
context:
space:
mode:
Diffstat (limited to 'src/volume-control.vala')
-rw-r--r--src/volume-control.vala106
1 files changed, 105 insertions, 1 deletions
diff --git a/src/volume-control.vala b/src/volume-control.vala
index e994922..a735057 100644
--- a/src/volume-control.vala
+++ b/src/volume-control.vala
@@ -34,6 +34,10 @@ public class VolumeControl : Object
private double _volume = 0.0;
private double _mic_volume = 0.0;
+ private DBusProxy _user_proxy;
+ private DBusProxy _greeter_proxy;
+ private bool _greeter_mode = false;
+
public signal void volume_changed (double v);
public signal void mic_volume_changed (double v);
@@ -48,6 +52,10 @@ public class VolumeControl : Object
if (loop == null)
loop = new PulseAudio.GLibMainLoop ();
+ _greeter_mode = Environment.get_variable ("XDG_SESSION_CLASS") == "greeter";
+ if (_greeter_mode)
+ watch_greeter_user.begin ();
+
this.reconnect_to_pulse ();
}
@@ -101,6 +109,7 @@ public class VolumeControl : Object
{
_volume = volume_to_double (i.volume.values[0]);
volume_changed (_volume);
+ sync_volume_to_accountsservice.begin ();
}
}
@@ -250,8 +259,10 @@ public class VolumeControl : Object
private void set_volume_success_cb (Context c, int success)
{
- if ((bool)success)
+ if ((bool)success) {
volume_changed (_volume);
+ sync_volume_to_accountsservice.begin ();
+ }
}
private void sink_info_set_volume_cb (Context c, SinkInfo? i, int eol)
@@ -315,4 +326,97 @@ public class VolumeControl : Object
{
return _mic_volume;
}
+
+ private async DBusProxy? get_user_proxy (string? username = null)
+ {
+ if (username == null)
+ username = Environment.get_variable ("USER");
+ if (username == "" || username == null)
+ return null;
+
+ DBusProxy accounts_proxy;
+ try {
+ accounts_proxy = yield DBusProxy.create_for_bus (BusType.SYSTEM, DBusProxyFlags.DO_NOT_LOAD_PROPERTIES | DBusProxyFlags.DO_NOT_CONNECT_SIGNALS, null, "org.freedesktop.Accounts", "/org/freedesktop/Accounts", "org.freedesktop.Accounts");
+ } catch (GLib.Error e) {
+ warning ("unable to get greeter proxy: %s", e.message);
+ return null;
+ }
+
+ try {
+ var user_path_variant = yield accounts_proxy.call ("FindUserByName", new Variant ("(s)", username), DBusCallFlags.NONE, -1);
+ string user_path;
+ user_path_variant.get ("(&s)", out user_path);
+ return yield DBusProxy.create_for_bus (BusType.SYSTEM, DBusProxyFlags.NONE, null, "org.freedesktop.Accounts", user_path, "org.freedesktop.DBus.Properties");
+ } catch (GLib.Error e) {
+ warning ("unable to find Accounts path for user %s: %s", username, e.message);
+ return null;
+ }
+ }
+
+ private async void sync_volume_from_accountsservice (string? username = null)
+ {
+ if (!_greeter_mode)
+ return; // skip sync if we are not in greeter
+
+ if (username == null) {
+ try {
+ var username_variant = yield _greeter_proxy.call ("GetActiveEntry", null, DBusCallFlags.NONE, -1);
+ username = username_variant.get_string ();
+ if (username == "" || username == null)
+ return;
+ } catch (GLib.Error e) {
+ warning ("unable to find Accounts path for user %s: %s", username, e.message);
+ return;
+ }
+ }
+
+ var user_proxy = yield get_user_proxy (username);
+ if (user_proxy == null)
+ return;
+
+ try {
+ var volume_outer_variant = yield user_proxy.call ("Get", new Variant ("(ss)", "com.ubuntu.touch.AccountsService.Sound", "Volume"), DBusCallFlags.NONE, -1);
+ Variant volume_variant;
+ volume_outer_variant.get ("(v)", out volume_variant);
+ set_volume (volume_variant.get_double ());
+ } catch (GLib.Error e) {
+ warning ("unable to sync volume from AccountsService: %s", e.message);
+ }
+ }
+
+ private void greeter_user_changed (string username)
+ {
+ sync_volume_from_accountsservice.begin (username);
+ }
+
+ private async void watch_greeter_user ()
+ {
+ try {
+ _greeter_proxy = yield DBusProxy.create_for_bus (BusType.SESSION, DBusProxyFlags.NONE, null, "com.canonical.UnityGreeter", "/list", "com.canonical.UnityGreeter.List");
+ } catch (GLib.Error e) {
+ warning ("unable to get greeter proxy: %s", e.message);
+ return;
+ }
+
+ _greeter_proxy.connect ("EntrySelected", greeter_user_changed);
+ yield sync_volume_from_accountsservice ();
+ }
+
+ private async void sync_volume_to_accountsservice ()
+ {
+ if (_greeter_mode)
+ return; // skip sync if we are in greeter
+
+ if (_user_proxy == null) {
+ _user_proxy = yield get_user_proxy ();
+ if (_user_proxy == null)
+ return;
+ }
+
+ try {
+ yield _user_proxy.call ("Set", new Variant ("(ssv)", "com.ubuntu.touch.AccountsService.Sound", "Volume", new Variant ("d", _volume)), DBusCallFlags.NONE, -1);
+ } catch (GLib.Error e) {
+ warning ("unable to sync volume to AccountsService: %s", e.message);
+ }
+ }
}