Source code for libqtile.widget.bluetooth

# Copyright (c) 2021 Graeme Holliday
#
# Permission is hereby granted, free of charge, to any person obtaining a copy
# of this software and associated documentation files (the "Software"), to deal
# in the Software without restriction, including without limitation the rights
# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
# copies of the Software, and to permit persons to whom the Software is
# furnished to do so, subject to the following conditions:
#
# The above copyright notice and this permission notice shall be included in
# all copies or substantial portions of the Software.
#
# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
# SOFTWARE.

from dbus_next.aio import MessageBus  # type: ignore
from dbus_next.constants import BusType  # type: ignore

from libqtile.widget import base

BLUEZ = 'org.bluez'
BLUEZ_PATH = '/org/bluez/hci0'
BLUEZ_ADAPTER = 'org.bluez.Adapter1'
BLUEZ_DEVICE = 'org.bluez.Device1'
BLUEZ_PROPERTIES = 'org.freedesktop.DBus.Properties'


[docs]class Bluetooth(base._TextBox): """ Displays bluetooth status or connected device. Uses dbus to communicate with the system bus. Widget requirements: dbus-next_. .. _dbus-next: https://pypi.org/project/dbus-next/ """ orientations = base.ORIENTATION_HORIZONTAL defaults = [ ('hci', '/dev_XX_XX_XX_XX_XX_XX', 'hci0 device path, can be found with d-feet or similar dbus explorer.') ] def __init__(self, **config): base._TextBox.__init__(self, '', **config) self.add_defaults(Bluetooth.defaults) async def _config_async(self): # set initial values self.powered = await self._init_adapter() self.connected, self.device = await self._init_device() self.update_text() async def _init_adapter(self): # set up interface to adapter properties using high-level api bus = await MessageBus(bus_type=BusType.SYSTEM).connect() introspect = await bus.introspect(BLUEZ, BLUEZ_PATH) obj = bus.get_proxy_object(BLUEZ, BLUEZ_PATH, introspect) iface = obj.get_interface(BLUEZ_ADAPTER) props = obj.get_interface(BLUEZ_PROPERTIES) powered = await iface.get_powered() # subscribe receiver to property changed props.on_properties_changed(self._signal_received) return powered async def _init_device(self): # set up interface to device properties using high-level api bus = await MessageBus(bus_type=BusType.SYSTEM).connect() introspect = await bus.introspect(BLUEZ, BLUEZ_PATH + self.hci) obj = bus.get_proxy_object(BLUEZ, BLUEZ_PATH + self.hci, introspect) iface = obj.get_interface(BLUEZ_DEVICE) props = obj.get_interface(BLUEZ_PROPERTIES) connected = await iface.get_connected() name = await iface.get_name() # subscribe receiver to property changed props.on_properties_changed(self._signal_received) return connected, name def _signal_received(self, interface_name, changed_properties, _invalidated_properties): powered = changed_properties.get('Powered', None) if powered is not None: self.powered = powered.value self.update_text() connected = changed_properties.get('Connected', None) if connected is not None: self.connected = connected.value self.update_text() device = changed_properties.get('Name', None) if device is not None: self.device = device.value self.update_text() def update_text(self): text = "" if not self.powered: text = 'off' else: if not self.connected: text = 'on' else: text = self.device self.update(text)