status/bluetooth: Add device menu
The new quick toggle gives us a good place for exposing connected and connectable devices. This was part of the original mockups, but didn't make the cut for GNOME 43 due to time constraints. https://gitlab.gnome.org/Teams/Design/os-mockups/-/issues/178 Part-of: <https://gitlab.gnome.org/GNOME/gnome-shell/-/merge_requests/2501>
This commit is contained in:
parent
b7c3a7f6ed
commit
600b921246
@ -137,4 +137,15 @@
|
|||||||
.wireless-secure-icon { icon-size: 0.5 * $base_icon_size; }
|
.wireless-secure-icon { icon-size: 0.5 * $base_icon_size; }
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.bt-device-item {
|
||||||
|
.popup-menu-icon { -st-icon-style: symbolic; }
|
||||||
|
}
|
||||||
|
|
||||||
|
.bt-menu-placeholder {
|
||||||
|
@extend %title_4;
|
||||||
|
text-align: center;
|
||||||
|
|
||||||
|
padding: 2em 4em;
|
||||||
|
}
|
||||||
|
|
||||||
.device-subtitle { color: transparentize($fg_color, 0.5); }
|
.device-subtitle { color: transparentize($fg_color, 0.5); }
|
||||||
|
@ -1,9 +1,11 @@
|
|||||||
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
// -*- mode: js; js-indent-level: 4; indent-tabs-mode: nil -*-
|
||||||
/* exported Indicator */
|
/* exported Indicator */
|
||||||
|
|
||||||
const {Gio, GLib, GnomeBluetooth, GObject} = imports.gi;
|
const {Gio, GLib, GnomeBluetooth, GObject, Pango, St} = imports.gi;
|
||||||
|
|
||||||
const {QuickToggle, SystemIndicator} = imports.ui.quickSettings;
|
const {Spinner} = imports.ui.animation;
|
||||||
|
const PopupMenu = imports.ui.popupMenu;
|
||||||
|
const {QuickMenuToggle, SystemIndicator} = imports.ui.quickSettings;
|
||||||
|
|
||||||
const {loadInterfaceXML} = imports.misc.fileUtils;
|
const {loadInterfaceXML} = imports.misc.fileUtils;
|
||||||
|
|
||||||
@ -15,6 +17,8 @@ const OBJECT_PATH = '/org/gnome/SettingsDaemon/Rfkill';
|
|||||||
const RfkillManagerInterface = loadInterfaceXML('org.gnome.SettingsDaemon.Rfkill');
|
const RfkillManagerInterface = loadInterfaceXML('org.gnome.SettingsDaemon.Rfkill');
|
||||||
const rfkillManagerInfo = Gio.DBusInterfaceInfo.new_for_xml(RfkillManagerInterface);
|
const rfkillManagerInfo = Gio.DBusInterfaceInfo.new_for_xml(RfkillManagerInterface);
|
||||||
|
|
||||||
|
Gio._promisify(GnomeBluetooth.Client.prototype, 'connect_service');
|
||||||
|
|
||||||
const BtClient = GObject.registerClass({
|
const BtClient = GObject.registerClass({
|
||||||
Properties: {
|
Properties: {
|
||||||
'available': GObject.ParamSpec.boolean('available', '', '',
|
'available': GObject.ParamSpec.boolean('available', '', '',
|
||||||
@ -29,6 +33,7 @@ const BtClient = GObject.registerClass({
|
|||||||
},
|
},
|
||||||
Signals: {
|
Signals: {
|
||||||
'devices-changed': {},
|
'devices-changed': {},
|
||||||
|
'device-removed': {param_types: [GObject.TYPE_STRING]},
|
||||||
},
|
},
|
||||||
}, class BtClient extends GObject.Object {
|
}, class BtClient extends GObject.Object {
|
||||||
_init() {
|
_init() {
|
||||||
@ -79,6 +84,7 @@ const BtClient = GObject.registerClass({
|
|||||||
|
|
||||||
this._client.connect('device-removed', (c, path) => {
|
this._client.connect('device-removed', (c, path) => {
|
||||||
this._deviceNotifyConnected.delete(path);
|
this._deviceNotifyConnected.delete(path);
|
||||||
|
this.emit('device-removed', path);
|
||||||
this.emit('devices-changed');
|
this.emit('devices-changed');
|
||||||
});
|
});
|
||||||
this._client.connect('device-added', (c, device) => {
|
this._client.connect('device-added', (c, device) => {
|
||||||
@ -109,6 +115,24 @@ const BtClient = GObject.registerClass({
|
|||||||
this._client.default_adapter_powered = true;
|
this._client.default_adapter_powered = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
async toggleDevice(device) {
|
||||||
|
const connect = !device.connected;
|
||||||
|
console.debug(`${connect
|
||||||
|
? 'Connect' : 'Disconnect'} device "${device.name}"`);
|
||||||
|
|
||||||
|
try {
|
||||||
|
await this._client.connect_service(
|
||||||
|
device.get_object_path(),
|
||||||
|
connect,
|
||||||
|
null);
|
||||||
|
console.debug(`Device "${device.name}" ${
|
||||||
|
connect ? 'connected' : 'disconnected'}`);
|
||||||
|
} catch (e) {
|
||||||
|
console.error(`Failed to ${connect
|
||||||
|
? 'connect' : 'disconnect'} device "${device.name}": ${e.message}`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
*getDevices() {
|
*getDevices() {
|
||||||
const deviceStore = this._client.get_devices();
|
const deviceStore = this._client.get_devices();
|
||||||
|
|
||||||
@ -145,11 +169,96 @@ const BtClient = GObject.registerClass({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
const BluetoothDeviceItem = GObject.registerClass(
|
||||||
|
class BluetoothDeviceItem extends PopupMenu.PopupBaseMenuItem {
|
||||||
|
constructor(device, client) {
|
||||||
|
super({
|
||||||
|
style_class: 'bt-device-item',
|
||||||
|
});
|
||||||
|
|
||||||
|
this._device = device;
|
||||||
|
this._client = client;
|
||||||
|
|
||||||
|
this._icon = new St.Icon({
|
||||||
|
style_class: 'popup-menu-icon',
|
||||||
|
});
|
||||||
|
this.add_child(this._icon);
|
||||||
|
|
||||||
|
this._label = new St.Label({
|
||||||
|
x_expand: true,
|
||||||
|
});
|
||||||
|
this.add_child(this._label);
|
||||||
|
|
||||||
|
this._subtitle = new St.Label({
|
||||||
|
style_class: 'device-subtitle',
|
||||||
|
});
|
||||||
|
this.add_child(this._subtitle);
|
||||||
|
|
||||||
|
this._spinner = new Spinner(16, {hideOnStop: true});
|
||||||
|
this.add_child(this._spinner);
|
||||||
|
|
||||||
|
this._spinner.bind_property('visible',
|
||||||
|
this._subtitle, 'visible',
|
||||||
|
GObject.BindingFlags.SYNC_CREATE |
|
||||||
|
GObject.BindingFlags.INVERT_BOOLEAN);
|
||||||
|
|
||||||
|
this._device.bind_property('connectable',
|
||||||
|
this, 'visible',
|
||||||
|
GObject.BindingFlags.SYNC_CREATE);
|
||||||
|
this._device.bind_property('icon',
|
||||||
|
this._icon, 'icon-name',
|
||||||
|
GObject.BindingFlags.SYNC_CREATE);
|
||||||
|
this._device.bind_property('name',
|
||||||
|
this._label, 'text',
|
||||||
|
GObject.BindingFlags.SYNC_CREATE);
|
||||||
|
this._device.bind_property_full('connected',
|
||||||
|
this._subtitle, 'text',
|
||||||
|
GObject.BindingFlags.SYNC_CREATE,
|
||||||
|
(bind, source) => [true, source ? _('Disconnect') : _('Connect')],
|
||||||
|
null);
|
||||||
|
|
||||||
|
this.connect('destroy', () => (this._spinner = null));
|
||||||
|
this.connect('activate', () => this._toggleConnected().catch(logError));
|
||||||
|
}
|
||||||
|
|
||||||
|
async _toggleConnected() {
|
||||||
|
this._spinner.play();
|
||||||
|
await this._client.toggleDevice(this._device);
|
||||||
|
this._spinner?.stop();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
const BluetoothToggle = GObject.registerClass(
|
const BluetoothToggle = GObject.registerClass(
|
||||||
class BluetoothToggle extends QuickToggle {
|
class BluetoothToggle extends QuickMenuToggle {
|
||||||
_init(client) {
|
_init(client) {
|
||||||
super._init({title: _('Bluetooth')});
|
super._init({title: _('Bluetooth')});
|
||||||
|
|
||||||
|
this.menu.setHeader('bluetooth-active-symbolic', _('Bluetooth'));
|
||||||
|
|
||||||
|
this._deviceItems = new Map();
|
||||||
|
this._deviceSection = new PopupMenu.PopupMenuSection();
|
||||||
|
this.menu.addMenuItem(this._deviceSection);
|
||||||
|
|
||||||
|
this._placeholderItem = new PopupMenu.PopupMenuItem('', {
|
||||||
|
style_class: 'bt-menu-placeholder',
|
||||||
|
reactive: false,
|
||||||
|
can_focus: false,
|
||||||
|
});
|
||||||
|
this._placeholderItem.label.clutter_text.set({
|
||||||
|
ellipsize: Pango.EllipsizeMode.NONE,
|
||||||
|
line_wrap: true,
|
||||||
|
});
|
||||||
|
this.menu.addMenuItem(this._placeholderItem);
|
||||||
|
|
||||||
|
this._deviceSection.actor.bind_property('visible',
|
||||||
|
this._placeholderItem, 'visible',
|
||||||
|
GObject.BindingFlags.SYNC_CREATE |
|
||||||
|
GObject.BindingFlags.INVERT_BOOLEAN);
|
||||||
|
|
||||||
|
this.menu.addMenuItem(new PopupMenu.PopupSeparatorMenuItem());
|
||||||
|
this.menu.addSettingsAction(_('Bluetooth Settings'),
|
||||||
|
'gnome-bluetooth-panel.desktop');
|
||||||
|
|
||||||
this._client = client;
|
this._client = client;
|
||||||
|
|
||||||
this._client.bind_property('available',
|
this._client.bind_property('available',
|
||||||
@ -165,17 +274,86 @@ class BluetoothToggle extends QuickToggle {
|
|||||||
null);
|
null);
|
||||||
|
|
||||||
this._client.connectObject(
|
this._client.connectObject(
|
||||||
|
'notify::active', () => this._onActiveChanged(),
|
||||||
'devices-changed', () => this._sync(),
|
'devices-changed', () => this._sync(),
|
||||||
|
'device-removed', (c, path) => this._removeDevice(path),
|
||||||
this);
|
this);
|
||||||
|
|
||||||
|
this.menu.connect('open-state-changed', isOpen => {
|
||||||
|
// We don't reorder the list while the menu is open,
|
||||||
|
// so do it now to start with the proper order
|
||||||
|
if (isOpen)
|
||||||
|
this._reorderDeviceItems();
|
||||||
|
});
|
||||||
|
|
||||||
this.connect('clicked', () => this._client.toggleActive());
|
this.connect('clicked', () => this._client.toggleActive());
|
||||||
|
|
||||||
|
this._updatePlaceholder();
|
||||||
|
this._sync();
|
||||||
|
}
|
||||||
|
|
||||||
|
_onActiveChanged() {
|
||||||
|
this._updatePlaceholder();
|
||||||
|
|
||||||
|
this._deviceItems.forEach(item => item.destroy());
|
||||||
|
this._deviceItems.clear();
|
||||||
|
|
||||||
this._sync();
|
this._sync();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_updatePlaceholder() {
|
||||||
|
this._placeholderItem.label.text = this._client.active
|
||||||
|
? _('No available or connected devices')
|
||||||
|
: _('Turn on Bluetooth to connect to devices');
|
||||||
|
}
|
||||||
|
|
||||||
|
_updateDeviceVisibility() {
|
||||||
|
this._deviceSection.actor.visible =
|
||||||
|
[...this._deviceItems.values()].some(item => item.visible);
|
||||||
|
}
|
||||||
|
|
||||||
|
_getSortedDevices() {
|
||||||
|
return [...this._client.getDevices()].sort((dev1, dev2) => {
|
||||||
|
if (dev1.connected !== dev2.connected)
|
||||||
|
return dev2.connected - dev1.connected;
|
||||||
|
return dev1.alias.localeCompare(dev2.alias);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
_removeDevice(path) {
|
||||||
|
this._deviceItems.get(path)?.destroy();
|
||||||
|
this._deviceItems.delete(path);
|
||||||
|
|
||||||
|
this._updateDeviceVisibility();
|
||||||
|
}
|
||||||
|
|
||||||
|
_reorderDeviceItems() {
|
||||||
|
const devices = this._getSortedDevices();
|
||||||
|
for (const [i, dev] of devices.entries()) {
|
||||||
|
const item = this._deviceItems.get(dev.get_object_path());
|
||||||
|
if (!item)
|
||||||
|
continue;
|
||||||
|
|
||||||
|
this._deviceSection.moveMenuItem(item, i);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
_sync() {
|
_sync() {
|
||||||
const connectedDevices = [...this._client.getDevices()]
|
const devices = this._getSortedDevices();
|
||||||
.filter(dev => dev.connected);
|
|
||||||
|
for (const dev of devices) {
|
||||||
|
const path = dev.get_object_path();
|
||||||
|
if (this._deviceItems.has(path))
|
||||||
|
continue;
|
||||||
|
|
||||||
|
const item = new BluetoothDeviceItem(dev, this._client);
|
||||||
|
item.connect('notify::visible', () => this._updateDeviceVisibility());
|
||||||
|
|
||||||
|
this._deviceSection.addMenuItem(item);
|
||||||
|
this._deviceItems.set(path, item);
|
||||||
|
}
|
||||||
|
|
||||||
|
const connectedDevices = devices.filter(dev => dev.connected);
|
||||||
const nConnected = connectedDevices.length;
|
const nConnected = connectedDevices.length;
|
||||||
|
|
||||||
if (nConnected > 1)
|
if (nConnected > 1)
|
||||||
@ -185,6 +363,8 @@ class BluetoothToggle extends QuickToggle {
|
|||||||
this.subtitle = connectedDevices[0].alias;
|
this.subtitle = connectedDevices[0].alias;
|
||||||
else
|
else
|
||||||
this.subtitle = null;
|
this.subtitle = null;
|
||||||
|
|
||||||
|
this._updateDeviceVisibility();
|
||||||
}
|
}
|
||||||
|
|
||||||
_getIconNameFromState(state) {
|
_getIconNameFromState(state) {
|
||||||
|
Loading…
Reference in New Issue
Block a user