// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
const GLib = imports.gi.GLib;
const Gio = imports.gi.Gio;
const Lang = imports.lang;
const Shell = imports.gi.Shell;
const PanelMenu = imports.ui.panelMenu;
const PopupMenu = imports.ui.popupMenu;
const LOCATION_SCHEMA = 'org.gnome.shell.location';
const MAX_ACCURACY_LEVEL = 'max-accuracy-level';
var GeoclueIface = ' \
\
\
\
\
\
\
\
';
const GeoclueManager = Gio.DBusProxy.makeProxyWrapper(GeoclueIface);
var AgentIface = ' \
\
\
\
\
\
\
\
\
\
';
const AccuracyLevel = {
NONE: 0,
COUNTRY: 1,
CITY: 4,
STREET: 6,
EXACT: 8,
};
const Indicator = new Lang.Class({
Name: 'LocationIndicator',
Extends: PanelMenu.SystemIndicator,
_init: function() {
this.parent();
this._settings = new Gio.Settings({ schema: LOCATION_SCHEMA });
this._settings.connect('changed::' + MAX_ACCURACY_LEVEL,
Lang.bind(this, this._onMaxAccuracyLevelChanged));
this._indicator = this._addIndicator();
this._indicator.icon_name = 'find-location-symbolic';
this._item = new PopupMenu.PopupSubMenuMenuItem(_("Location"), true);
this._item.icon.icon_name = 'find-location-symbolic';
var credentials = new Gio.Credentials();
var uid = credentials.get_unix_user();
this._agent = Gio.DBusExportedObject.wrapJSObject(AgentIface, this);
this._agent.export(Gio.DBus.system,
'/org/freedesktop/GeoClue2/Agent/' + uid);
this._item.status.text = _("On");
this._onoffAction = this._item.menu.addAction(_("Turn Off"), Lang.bind(this, this._onOnOffAction));
this._accurateItem = new PopupMenu.PopupMenuItem(_("Accurate (GPS + Network)"), false);
this._accurateItem.connect('activate', Lang.bind(this, this._onAccurateItemActivated));
this._item.menu.addMenuItem(this._accurateItem);
this._powerSavingItem = new PopupMenu.PopupMenuItem(_("Power Saving (Network Only)"), false);
this._powerSavingItem.connect('activate', Lang.bind(this, this._onPowerSavingItemActivated));
this._item.menu.addMenuItem(this._powerSavingItem);
this._offItem = new PopupMenu.PopupMenuItem(_("Off"), false);
this._offItem.connect('activate', Lang.bind(this, this._onOffItemActivated));
this._item.menu.addMenuItem(this._offItem);
this.menu.addMenuItem(this._item);
this._watchId = Gio.bus_watch_name(Gio.BusType.SYSTEM,
'org.freedesktop.GeoClue2',
0,
Lang.bind(this, this._connectToGeoclue),
Lang.bind(this, this._onGeoclueVanished));
this._onMaxAccuracyLevelChanged();
this._connectToGeoclue();
},
get MaxAccuracyLevel() {
return this._getMaxAccuracyLevel();
},
// We (and geoclue) have currently no way to reliably identifying apps so
// for now, lets just authorize all apps as long as they provide a valid
// desktop ID. We also ensure they don't get more accuracy than global max.
AuthorizeApp: function(desktop_id, reqAccuracyLevel) {
var appSystem = Shell.AppSystem.get_default();
var app = appSystem.lookup_app(desktop_id + ".desktop");
if (app == null) {
return [false, 0];
}
let allowedAccuracyLevel = clamp(reqAccuracyLevel, 0, this._getMaxAccuracyLevel());
return [true, allowedAccuracyLevel];
},
_syncIndicator: function() {
if (this._proxy == null) {
this._indicator.visible = false;
return;
}
this._indicator.visible = this._proxy.InUse;
},
_connectToGeoclue: function() {
if (this._proxy != null || this._connecting)
return false;
this._connecting = true;
new GeoclueManager(Gio.DBus.system,
'org.freedesktop.GeoClue2',
'/org/freedesktop/GeoClue2/Manager',
Lang.bind(this, this._onProxyReady));
return true;
},
_onProxyReady: function(proxy, error) {
if (error != null) {
log(error.message);
this._connecting = false;
return;
}
this._proxy = proxy;
this._propertiesChangedId = this._proxy.connect('g-properties-changed',
Lang.bind(this, this._onGeocluePropsChanged));
this._updateMenu();
this._syncIndicator();
this._proxy.AddAgentRemote('gnome-shell', Lang.bind(this, this._onAgentRegistered));
},
_onAgentRegistered: function(result, error) {
this._connecting = false;
this._notifyMaxAccuracyLevel();
if (error != null)
log(error.message);
},
_onGeoclueVanished: function() {
if (this._propertiesChangedId) {
this._proxy.disconnect(this._propertiesChangedId);
this._propertiesChangedId = 0;
}
this._proxy = null;
this._syncIndicator();
},
_onOnOffAction: function() {
if (this._getMaxAccuracyLevel() == 0)
this._settings.set_enum(MAX_ACCURACY_LEVEL, this._availableAccuracyLevel);
else
this._settings.set_enum(MAX_ACCURACY_LEVEL, AccuracyLevel.NONE);
},
_onAccurateItemActivated: function() {
this._settings.set_enum(MAX_ACCURACY_LEVEL, AccuracyLevel.EXACT);
},
_onPowerSavingItemActivated: function() {
this._settings.set_enum(MAX_ACCURACY_LEVEL, AccuracyLevel.STREET);
},
_onOffItemActivated: function() {
this._settings.set_enum(MAX_ACCURACY_LEVEL, AccuracyLevel.NONE);
},
_onMaxAccuracyLevelChanged: function() {
let maxAccuracyLevel = this._getMaxAccuracyLevel();
if (this._availableAccuracyLevel < AccuracyLevel.EXACT) {
if (maxAccuracyLevel == 0) {
this._item.status.text = _("Off");
this._onoffAction.label.text = "Turn On";
} else {
this._item.status.text = _("On");
this._onoffAction.label.text = "Turn Off";
}
} else {
if (maxAccuracyLevel == 0) {
this._item.status.text = _("Off");
this._offItem.setOrnament(PopupMenu.Ornament.DOT);
this._accurateItem.setOrnament(PopupMenu.Ornament.NONE);
this._powerSavingItem.setOrnament(PopupMenu.Ornament.NONE);
} else if (maxAccuracyLevel < AccuracyLevel.EXACT) {
this._item.status.text = _("Power Saving");
this._powerSavingItem.setOrnament(PopupMenu.Ornament.DOT);
this._accurateItem.setOrnament(PopupMenu.Ornament.NONE);
this._offItem.setOrnament(PopupMenu.Ornament.NONE);
} else {
this._item.status.text = _("Accurate");
this._accurateItem.setOrnament(PopupMenu.Ornament.DOT);
this._offItem.setOrnament(PopupMenu.Ornament.NONE);
this._powerSavingItem.setOrnament(PopupMenu.Ornament.NONE);
}
}
// Gotta ensure geoclue is up and we are registered as agent to it
// before we emit the notify for this property change.
if (!this._connectToGeoclue())
this._notifyMaxAccuracyLevel();
},
_getMaxAccuracyLevel: function() {
return this._settings.get_enum(MAX_ACCURACY_LEVEL);
},
_notifyMaxAccuracyLevel: function() {
let variant = new GLib.Variant('u', this._getMaxAccuracyLevel());
this._agent.emit_property_changed('MaxAccuracyLevel', variant);
},
_updateMenu: function() {
this._availableAccuracyLevel = this._proxy.AvailableAccuracyLevel;
this.menu.actor.visible = (this._availableAccuracyLevel != 0);
if (this._availableAccuracyLevel == 0)
return;
if (this._availableAccuracyLevel < AccuracyLevel.EXACT) {
this._onoffAction.actor.show();
this._accurateItem.actor.hide();
this._powerSavingItem.actor.hide();
this._offItem.actor.hide();
} else {
this._onoffAction.actor.hide();
this._accurateItem.actor.show();
this._powerSavingItem.actor.show();
this._offItem.actor.show();
}
this._onMaxAccuracyLevelChanged();
},
_onGeocluePropsChanged: function(proxy, properties) {
let unpacked = properties.deep_unpack();
if ("InUse" in unpacked)
this._syncIndicator();
if ("AvailableAccuracyLevel" in unpacked)
this._updateMenu();
}
});
function clamp(value, min, max) {
return Math.max(min, Math.min(max, value));
}