something-x-dev 1.2.3.dev4__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.2.3.dev4/something_x_dev.egg-info → something_x_dev-1.3.0.dev6}/PKG-INFO +1 -1
  2. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/application.py +3 -0
  3. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/bluetooth.py +34 -0
  4. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/pages/device.py +6 -1
  5. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/pages/home.py +2 -7
  6. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/protocol.py +27 -21
  7. something_x_dev-1.3.0.dev6/nothing_app/tray.py +149 -0
  8. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/pyproject.toml +1 -1
  9. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6/something_x_dev.egg-info}/PKG-INFO +1 -1
  10. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/SOURCES.txt +1 -0
  11. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/LICENSE +0 -0
  12. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/README.md +0 -0
  13. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/__init__.py +0 -0
  14. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/data/__init__.py +0 -0
  15. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/data/com.something.x.omarchy.desktop +0 -0
  16. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/data/style.css +0 -0
  17. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/pages/__init__.py +0 -0
  18. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/profiles.py +0 -0
  19. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/splash.py +0 -0
  20. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/window.py +0 -0
  21. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/setup.cfg +0 -0
  22. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/dependency_links.txt +0 -0
  23. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/entry_points.txt +0 -0
  24. {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/requires.txt +0 -0
  25. {something_x_dev-1.2.3.dev4 → 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.2.3.dev4
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, ()),
@@ -265,7 +265,12 @@ def _settings_row(title: str, subtitle: str = "", right_widget: Gtk.Widget | Non
265
265
 
266
266
 
267
267
  class DevicePage(Gtk.Box):
268
- def __init__(self, bt_device: BluetoothDevice, bt_manager: BluetoothManager, nothing_dev: NothingDevice | None = None):
268
+ def __init__(
269
+ self,
270
+ bt_device: BluetoothDevice,
271
+ bt_manager: BluetoothManager,
272
+ nothing_dev: NothingDevice | None = None,
273
+ ):
269
274
  super().__init__(orientation=Gtk.Orientation.VERTICAL)
270
275
  self._bt_device = bt_device
271
276
  self._bt = bt_manager
@@ -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)
@@ -147,7 +147,7 @@ class NothingDevice(GObject.Object):
147
147
  self._anc_pending_mode: int = ANCMode.OFF
148
148
  self._last_anc_level: int = _ANC_STRONG
149
149
  self._thread: threading.Thread | None = None
150
- self._low_bat_notified: set[str] = set()
150
+ self._low_bat_notified: dict[str, set[int]] = {}
151
151
 
152
152
  # ── Public API ────────────────────────────────────────────────────────────
153
153
 
@@ -602,29 +602,35 @@ class NothingDevice(GObject.Object):
602
602
  eq_val = EQ_PRESETS.get(p["eq"], 0)
603
603
  self._x55_send(_CMD_SET_EQ, bytes([eq_val]), label=f"restore EQ={p['eq']}")
604
604
 
605
+ _LOW_BAT_THRESHOLDS = (20, 15, 10, 5)
606
+
605
607
  def _check_low_battery(self, slot: str, pct: int, label: str):
606
608
  if pct < 0:
607
609
  return
608
- if pct <= 20 and slot not in self._low_bat_notified:
609
- self._low_bat_notified.add(slot)
610
- threading.Thread(
611
- target=subprocess.run,
612
- args=(
613
- [
614
- "notify-send",
615
- "-u",
616
- "critical",
617
- "-i",
618
- "battery-caution",
619
- "Something X",
620
- f"{label}: {pct}% battery remaining",
621
- ],
622
- ),
623
- kwargs={"capture_output": True},
624
- daemon=True,
625
- ).start()
626
- elif pct > 25:
627
- self._low_bat_notified.discard(slot)
610
+ if pct > 25:
611
+ self._low_bat_notified.pop(slot, None)
612
+ return
613
+ notified = self._low_bat_notified.setdefault(slot, set())
614
+ for threshold in self._LOW_BAT_THRESHOLDS:
615
+ if pct <= threshold and threshold not in notified:
616
+ notified.add(threshold)
617
+ threading.Thread(
618
+ target=subprocess.run,
619
+ args=(
620
+ [
621
+ "notify-send",
622
+ "-u",
623
+ "critical",
624
+ "-i",
625
+ "battery-caution",
626
+ "Something X",
627
+ f"{label}: {pct}% battery remaining",
628
+ ],
629
+ ),
630
+ kwargs={"capture_output": True},
631
+ daemon=True,
632
+ ).start()
633
+ break
628
634
 
629
635
  def _activation_fallback(self):
630
636
  if not self._activated and self._rfcomm_connected:
@@ -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.2.3.dev4"
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.2.3.dev4
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