From 3724d7a47cf594089f0b1e513c7c4af167f59270 Mon Sep 17 00:00:00 2001 From: JosephMcc Date: Mon, 4 Nov 2024 08:53:24 -0800 Subject: [PATCH] Add a clutter audio device selection dialog (#12461) Requires: https://github.com/linuxmint/cinnamon-settings-daemon/pull/401 --- .../theme/cinnamon-sass/widgets/_dialogs.scss | 22 ++ js/ui/audioDeviceSelection.js | 216 ++++++++++++++++++ js/ui/main.js | 3 + 3 files changed, 241 insertions(+) create mode 100644 js/ui/audioDeviceSelection.js diff --git a/data/theme/cinnamon-sass/widgets/_dialogs.scss b/data/theme/cinnamon-sass/widgets/_dialogs.scss index fbb58f26ec..659e405e36 100644 --- a/data/theme/cinnamon-sass/widgets/_dialogs.scss +++ b/data/theme/cinnamon-sass/widgets/_dialogs.scss @@ -115,3 +115,25 @@ &-user-root-label { color: $error_color; } } + +// Audio selection dialog + +.audio-device-selection-dialog { + min-width: 24em; + + .audio-selection-box { + spacing: $base_padding *2; + + .audio-selection-device { + @extend %flat_button; + border-radius: $base_border_radius; + + .audio-selection-device-box { + padding: $base_padding * 2; + spacing: $base_padding * 2; + } + + .audio-selection-device-icon { icon-size: 64px;} + } + } +} diff --git a/js/ui/audioDeviceSelection.js b/js/ui/audioDeviceSelection.js new file mode 100644 index 0000000000..d2757b6db9 --- /dev/null +++ b/js/ui/audioDeviceSelection.js @@ -0,0 +1,216 @@ +const Cinnamon = imports.gi.Cinnamon; +const Clutter = imports.gi.Clutter; +const Gio = imports.gi.Gio; +const GLib = imports.gi.GLib; +const GObject = imports.gi.GObject; +const Meta = imports.gi.Meta; +const St = imports.gi.St; + +const Dialog = imports.ui.dialog; +const Main = imports.ui.main; +const ModalDialog = imports.ui.modalDialog; + +const Util = imports.misc.util; + +const AudioDeviceSelectionIface = ` + + + + + + + + + + + +`; + +const AudioDevice = { + HEADPHONES: 1 << 0, + HEADSET: 1 << 1, + MICROPHONE: 1 << 2, +}; + +const AudioDeviceSelectionDialog = GObject.registerClass({ + Signals: { 'device-selected': { param_types: [GObject.TYPE_UINT] }}, +}, class AudioDeviceSelectionDialog extends ModalDialog.ModalDialog { + _init(devices) { + super._init({ styleClass: 'audio-device-selection-dialog' }); + + this._deviceItems = {}; + + this._buildLayout(); + + if (devices & AudioDevice.HEADPHONES) + this._addDevice(AudioDevice.HEADPHONES); + if (devices & AudioDevice.HEADSET) + this._addDevice(AudioDevice.HEADSET); + if (devices & AudioDevice.MICROPHONE) + this._addDevice(AudioDevice.MICROPHONE); + + if (this._selectionBox.get_n_children() < 2) + throw new Error('Too few devices for a selection'); + } + + _buildLayout() { + let content = new Dialog.MessageDialogContent({ + title: _('Select Audio Device'), + }); + + this._selectionBox = new St.BoxLayout({ + style_class: 'audio-selection-box', + x_align: Clutter.ActorAlign.CENTER, + x_expand: true, + }); + content.add_child(this._selectionBox); + + this.contentLayout.add_child(content); + + this.addButton({ + action: () => this.close(), + label: _('Cancel'), + key: Clutter.KEY_Escape, + destructive_action: true, + }); + + this.addButton({ + action: this._openSettings.bind(this), + label: _('Sound Settings'), + }); + } + + _getDeviceLabel(device) { + switch (device) { + case AudioDevice.HEADPHONES: + return _('Headphones'); + case AudioDevice.HEADSET: + return _('Headset'); + case AudioDevice.MICROPHONE: + return _('Microphone'); + default: + return null; + } + } + + _getDeviceIcon(device) { + switch (device) { + case AudioDevice.HEADPHONES: + return 'audio-headphones-symbolic'; + case AudioDevice.HEADSET: + return 'audio-headset-symbolic'; + case AudioDevice.MICROPHONE: + return 'audio-input-microphone-symbolic'; + default: + return null; + } + } + + _addDevice(device) { + const box = new St.BoxLayout({ + style_class: 'audio-selection-device-box', + vertical: true, + }); + box.connect('notify::height', () => { + Meta.later_add(Meta.LaterType.BEFORE_REDRAW, () => { + box.width = box.height; + return GLib.SOURCE_REMOVE; + }); + }); + + const icon = new St.Icon({ + style_class: 'audio-selection-device-icon', + icon_name: this._getDeviceIcon(device), + }); + box.add_child(icon); + + const label = new St.Label({ + style_class: 'audio-selection-device-label', + text: this._getDeviceLabel(device), + x_align: Clutter.ActorAlign.CENTER, + }); + box.add_child(label); + + const button = new St.Button({ + style_class: 'audio-selection-device', + can_focus: true, + child: box, + }); + this._selectionBox.add_child(button); + + button.connect('clicked', () => { + this.emit('device-selected', device); + this.close(); + Main.overview.hide(); + }); + } + + _openSettings() { + Util.spawnCommandLine('cinnamon-settings sound'); + this.close(); + } +}); + +var AudioDeviceSelectionDBus = class AudioDeviceSelectionDBus { + constructor() { + this._audioSelectionDialog = null; + + this._dbusImpl = Gio.DBusExportedObject.wrapJSObject(AudioDeviceSelectionIface, this); + this._dbusImpl.export(Gio.DBus.session, '/org/Cinnamon/AudioDeviceSelection'); + + Gio.DBus.session.own_name('org.Cinnamon.AudioDeviceSelection', Gio.BusNameOwnerFlags.REPLACE, null, null); + } + + _onDialogClosed() { + this._audioSelectionDialog = null; + } + + _onDeviceSelected(dialog, device) { + let connection = this._dbusImpl.get_connection(); + let info = this._dbusImpl.get_info(); + const deviceName = Object.keys(AudioDevice) + .filter(dev => AudioDevice[dev] === device)[0].toLowerCase(); + connection.emit_signal( + this._audioSelectionDialog._sender, + this._dbusImpl.get_object_path(), + info ? info.name : null, + 'DeviceSelected', + GLib.Variant.new('(s)', [deviceName])); + } + + OpenAsync(params, invocation) { + if (this._audioSelectionDialog) { + invocation.return_value(null); + return; + } + + let [deviceNames] = params; + let devices = 0; + deviceNames.forEach(n => (devices |= AudioDevice[n.toUpperCase()])); + + let dialog; + try { + dialog = new AudioDeviceSelectionDialog(devices); + } catch (e) { + invocation.return_value(null); + return; + } + dialog._sender = invocation.get_sender(); + + dialog.connect('closed', this._onDialogClosed.bind(this)); + dialog.connect('device-selected', + this._onDeviceSelected.bind(this)); + dialog.open(); + + this._audioSelectionDialog = dialog; + invocation.return_value(null); + } + + CloseAsync(params, invocation) { + if (this._audioSelectionDialog && + this._audioSelectionDialog._sender === invocation.get_sender()) + this._audioSelectionDialog.close(); + + invocation.return_value(null); + } +} diff --git a/js/ui/main.js b/js/ui/main.js index e0ff96f439..b0d23d33ce 100644 --- a/js/ui/main.js +++ b/js/ui/main.js @@ -88,6 +88,7 @@ const GObject = imports.gi.GObject; const XApp = imports.gi.XApp; const PointerTracker = imports.misc.pointerTracker; +const AudioDeviceSelection = imports.ui.audioDeviceSelection; const SoundManager = imports.ui.soundManager; const BackgroundManager = imports.ui.backgroundManager; const Config = imports.misc.config; @@ -154,6 +155,7 @@ var messageTray = null; var notificationDaemon = null; var windowAttentionHandler = null; var screenRecorder = null; +var cinnamonAudioSelectionDBusService = null; var cinnamonDBusService = null; var screenshotService = null; var modalCount = 0; @@ -312,6 +314,7 @@ function start() { Clutter.get_default_backend().set_input_method(new InputMethod.InputMethod()); new CinnamonPortalHandler(); + cinnamonAudioSelectionDBusService = new AudioDeviceSelection.AudioDeviceSelectionDBus(); cinnamonDBusService = new CinnamonDBus.CinnamonDBus(); setRunState(RunState.STARTUP);