something-x-dev 1.3.0.dev5__tar.gz → 1.3.0.dev6__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (25) hide show
  1. {something_x_dev-1.3.0.dev5/something_x_dev.egg-info → something_x_dev-1.3.0.dev6}/PKG-INFO +1 -1
  2. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/application.py +3 -0
  3. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/bluetooth.py +34 -0
  4. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/pages/home.py +2 -7
  5. something_x_dev-1.3.0.dev6/nothing_app/tray.py +149 -0
  6. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/pyproject.toml +1 -1
  7. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6/something_x_dev.egg-info}/PKG-INFO +1 -1
  8. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/SOURCES.txt +1 -0
  9. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/LICENSE +0 -0
  10. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/README.md +0 -0
  11. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/__init__.py +0 -0
  12. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/data/__init__.py +0 -0
  13. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/data/com.something.x.omarchy.desktop +0 -0
  14. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/data/style.css +0 -0
  15. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/pages/__init__.py +0 -0
  16. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/pages/device.py +0 -0
  17. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/profiles.py +0 -0
  18. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/protocol.py +0 -0
  19. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/splash.py +0 -0
  20. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/nothing_app/window.py +0 -0
  21. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/setup.cfg +0 -0
  22. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/dependency_links.txt +0 -0
  23. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/entry_points.txt +0 -0
  24. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/requires.txt +0 -0
  25. {something_x_dev-1.3.0.dev5 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/top_level.txt +0 -0
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: something-x-dev
3
- Version: 1.3.0.dev5
3
+ Version: 1.3.0.dev6
4
4
  Summary: Something X device manager for Omarchy / Linux
5
5
  Author: Raphael
6
6
  License: MIT
@@ -13,6 +13,7 @@ from gi.repository import Gtk, Adw, Gdk, Gio, GLib
13
13
  from .bluetooth import BluetoothManager
14
14
  from .window import SomethingXWindow
15
15
  from .splash import SplashScreen
16
+ from .tray import SomethingXTray
16
17
 
17
18
 
18
19
  def _install_desktop_file():
@@ -50,6 +51,7 @@ class SomethingXApplication(Adw.Application):
50
51
  self._bt: BluetoothManager | None = None
51
52
  self._splash: SplashScreen | None = None
52
53
  self._window: SomethingXWindow | None = None
54
+ self._tray: SomethingXTray | None = None
53
55
  self.connect("activate", self._on_activate)
54
56
 
55
57
  def _on_activate(self, _app):
@@ -72,6 +74,7 @@ class SomethingXApplication(Adw.Application):
72
74
  win = SomethingXWindow(bt_manager=self._bt, application=self)
73
75
  win.connect("close-request", self._on_window_close)
74
76
  self._window = win
77
+ self._tray = SomethingXTray(self._bt, on_show_window=win.present)
75
78
  win.present()
76
79
  if self._splash:
77
80
  self._splash.destroy()
@@ -44,6 +44,40 @@ class BluetoothDevice:
44
44
  return f"<BTDevice {self.name!r} {'●' if self.connected else '○'}>"
45
45
 
46
46
 
47
+ # BlueZ Device1.Icon → GTK symbolic icon name
48
+ _BLUEZ_ICON_MAP: dict[str, str] = {
49
+ "audio-headphones": "audio-headphones-symbolic",
50
+ "audio-headset": "audio-headset-symbolic",
51
+ "audio-card": "audio-card-symbolic",
52
+ "input-mouse": "input-mouse-symbolic",
53
+ "input-keyboard": "input-keyboard-symbolic",
54
+ "input-gaming": "input-gaming-symbolic",
55
+ "input-tablet": "input-tablet-symbolic",
56
+ "phone": "phone-symbolic",
57
+ "computer": "computer-symbolic",
58
+ "printer": "printer-symbolic",
59
+ }
60
+
61
+ _WATCH_NAME_PATTERNS = ("watch", "band", "gear", "amazfit", "fenix", "vivoactive", "galaxy fit")
62
+
63
+
64
+ def device_icon_name(device: "BluetoothDevice | None") -> str:
65
+ """Return a GTK symbolic icon name for a Bluetooth device."""
66
+ if device is None:
67
+ return "audio-headphones-symbolic"
68
+ name_lower = device.name.lower()
69
+ if any(p in name_lower for p in _WATCH_NAME_PATTERNS):
70
+ return "alarm-symbolic"
71
+ if "ear (stick)" in name_lower:
72
+ return "audio-input-microphone-symbolic"
73
+ mapped = _BLUEZ_ICON_MAP.get(device.icon)
74
+ if mapped:
75
+ return mapped
76
+ if "phone" in name_lower:
77
+ return "phone-symbolic"
78
+ return "audio-headphones-symbolic"
79
+
80
+
47
81
  class BluetoothManager(GObject.Object):
48
82
  __gsignals__ = {
49
83
  "devices-changed": (GObject.SignalFlags.RUN_FIRST, None, ()),
@@ -3,7 +3,7 @@ import gi
3
3
  gi.require_version("Gtk", "4.0")
4
4
  from gi.repository import Gtk, GLib, GObject
5
5
 
6
- from ..bluetooth import BluetoothDevice, BluetoothManager
6
+ from ..bluetooth import BluetoothDevice, BluetoothManager, device_icon_name
7
7
 
8
8
 
9
9
  class DeviceRow(Gtk.Box):
@@ -14,12 +14,7 @@ class DeviceRow(Gtk.Box):
14
14
  self._build()
15
15
 
16
16
  def _build(self):
17
- icon_name = "audio-headphones-symbolic"
18
- if "phone" in self.device.name.lower():
19
- icon_name = "phone-symbolic"
20
- elif "ear (stick)" in self.device.name.lower():
21
- icon_name = "audio-input-microphone-symbolic"
22
- icon = Gtk.Image.new_from_icon_name(icon_name)
17
+ icon = Gtk.Image.new_from_icon_name(device_icon_name(self.device))
23
18
  icon.set_pixel_size(28)
24
19
  icon.set_opacity(0.6)
25
20
  self.append(icon)
@@ -0,0 +1,149 @@
1
+ import os
2
+ import dbus
3
+ import dbus.service
4
+ from gi.repository import GLib, GObject
5
+
6
+ from .bluetooth import BluetoothManager, BluetoothDevice, device_icon_name
7
+
8
+ _ITEM_IFACE = "org.kde.StatusNotifierItem"
9
+ _WATCHER_IFACE = "org.kde.StatusNotifierWatcher"
10
+ _WATCHER_SERVICE = "org.kde.StatusNotifierWatcher"
11
+ _ITEM_PATH = "/StatusNotifierItem"
12
+
13
+ _EMPTY_PIXMAPS = dbus.Array([], signature="(iiay)")
14
+
15
+
16
+ class _SNIItem(dbus.service.Object):
17
+ def __init__(self, bus, service_name, on_activate):
18
+ bus_name = dbus.service.BusName(service_name, bus)
19
+ super().__init__(bus_name, _ITEM_PATH)
20
+ self._on_activate = on_activate
21
+ self._icon_name = "audio-headphones"
22
+ self._tooltip_title = "Something X"
23
+ self._tooltip_body = ""
24
+
25
+ # ── SNI methods ────────────────────────────────────────────────────────────
26
+
27
+ @dbus.service.method(_ITEM_IFACE, in_signature="ii")
28
+ def Activate(self, x, y):
29
+ GLib.idle_add(self._on_activate)
30
+
31
+ @dbus.service.method(_ITEM_IFACE, in_signature="ii")
32
+ def SecondaryActivate(self, x, y):
33
+ pass
34
+
35
+ @dbus.service.method(_ITEM_IFACE, in_signature="ii")
36
+ def ContextMenu(self, x, y):
37
+ pass
38
+
39
+ @dbus.service.method(_ITEM_IFACE, in_signature="is")
40
+ def Scroll(self, delta, orientation):
41
+ pass
42
+
43
+ # ── SNI signals ───────────────────────────────────────────────────────────
44
+
45
+ @dbus.service.signal(_ITEM_IFACE)
46
+ def NewIcon(self):
47
+ pass
48
+
49
+ @dbus.service.signal(_ITEM_IFACE)
50
+ def NewToolTip(self):
51
+ pass
52
+
53
+ @dbus.service.signal(_ITEM_IFACE, signature="s")
54
+ def NewStatus(self, status):
55
+ pass
56
+
57
+ # ── D-Bus Properties ──────────────────────────────────────────────────────
58
+
59
+ @dbus.service.method(dbus.PROPERTIES_IFACE, in_signature="ss", out_signature="v")
60
+ def Get(self, interface, prop):
61
+ return self._props()[prop]
62
+
63
+ @dbus.service.method(dbus.PROPERTIES_IFACE, in_signature="s", out_signature="a{sv}")
64
+ def GetAll(self, interface):
65
+ return self._props()
66
+
67
+ def _props(self):
68
+ tooltip = dbus.Struct(
69
+ (
70
+ dbus.String(""),
71
+ _EMPTY_PIXMAPS,
72
+ dbus.String(self._tooltip_title),
73
+ dbus.String(self._tooltip_body),
74
+ ),
75
+ signature="sa(iiay)ss",
76
+ )
77
+ return {
78
+ "Id": dbus.String("something-x"),
79
+ "Category": dbus.String("Hardware"),
80
+ "Title": dbus.String("Something X"),
81
+ "Status": dbus.String("Active"),
82
+ "WindowId": dbus.UInt32(0),
83
+ "IconName": dbus.String(self._icon_name),
84
+ "IconPixmap": _EMPTY_PIXMAPS,
85
+ "OverlayIconName": dbus.String(""),
86
+ "OverlayIconPixmap": _EMPTY_PIXMAPS,
87
+ "AttentionIconName": dbus.String(""),
88
+ "AttentionIconPixmap": _EMPTY_PIXMAPS,
89
+ "AttentionMovieName": dbus.String(""),
90
+ "ToolTip": tooltip,
91
+ "ItemIsMenu": dbus.Boolean(False),
92
+ "Menu": dbus.ObjectPath(_ITEM_PATH),
93
+ }
94
+
95
+ # ── update helpers ────────────────────────────────────────────────────────
96
+
97
+ def set_icon(self, icon_name: str):
98
+ if icon_name != self._icon_name:
99
+ self._icon_name = icon_name
100
+ self.NewIcon()
101
+
102
+ def set_tooltip(self, title: str, body: str):
103
+ self._tooltip_title = title
104
+ self._tooltip_body = body
105
+ self.NewToolTip()
106
+
107
+
108
+ class SomethingXTray(GObject.Object):
109
+ """StatusNotifierItem tray icon. Shows battery on hover; icon adapts to device type."""
110
+
111
+ def __init__(self, bt_manager: BluetoothManager, on_show_window):
112
+ super().__init__()
113
+ self._bt = bt_manager
114
+ self._on_show = on_show_window
115
+ self._item: _SNIItem | None = None
116
+ self._setup()
117
+ bt_manager.connect("devices-changed", self._on_devices_changed)
118
+
119
+ def _setup(self):
120
+ try:
121
+ bus = dbus.SessionBus()
122
+ service_name = f"org.kde.StatusNotifierItem-{os.getpid()}-1"
123
+ self._item = _SNIItem(bus, service_name, self._on_show)
124
+ try:
125
+ watcher = bus.get_object(_WATCHER_SERVICE, "/StatusNotifierWatcher")
126
+ dbus.Interface(watcher, _WATCHER_IFACE).RegisterStatusNotifierItem(service_name)
127
+ except dbus.exceptions.DBusException:
128
+ pass # watcher not running; item is still exported on the bus
129
+ except Exception as exc:
130
+ print(f"[tray] SNI setup failed: {exc}")
131
+
132
+ def _on_devices_changed(self, _manager):
133
+ if not self._item:
134
+ return
135
+ nothing_devs = self._bt.get_nothing_devices()
136
+ connected = [d for d in nothing_devs if d.connected]
137
+ if connected:
138
+ dev = connected[0]
139
+ parts = []
140
+ if dev.battery is not None:
141
+ parts.append(f"{dev.name}: {dev.battery}%")
142
+ self._item.set_tooltip("Something X", "\n".join(parts) if parts else "Connected")
143
+ self._item.set_icon(device_icon_name(dev))
144
+ else:
145
+ # fall back to first paired Nothing device, or generic icon
146
+ paired = nothing_devs[0] if nothing_devs else None
147
+ icon = device_icon_name(paired) if paired else "audio-headphones"
148
+ self._item.set_tooltip("Something X", "No devices connected")
149
+ self._item.set_icon(icon)
@@ -4,7 +4,7 @@ build-backend = "setuptools.build_meta"
4
4
 
5
5
  [project]
6
6
  name = "something-x-dev"
7
- version = "1.3.0.dev5"
7
+ version = "1.3.0.dev6"
8
8
  description = "Something X device manager for Omarchy / Linux"
9
9
  readme = "README.md"
10
10
  license = { text = "MIT" }
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: something-x-dev
3
- Version: 1.3.0.dev5
3
+ Version: 1.3.0.dev6
4
4
  Summary: Something X device manager for Omarchy / Linux
5
5
  Author: Raphael
6
6
  License: MIT
@@ -7,6 +7,7 @@ nothing_app/bluetooth.py
7
7
  nothing_app/profiles.py
8
8
  nothing_app/protocol.py
9
9
  nothing_app/splash.py
10
+ nothing_app/tray.py
10
11
  nothing_app/window.py
11
12
  nothing_app/data/__init__.py
12
13
  nothing_app/data/com.something.x.omarchy.desktop