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.
- {something_x_dev-1.2.3.dev4/something_x_dev.egg-info → something_x_dev-1.3.0.dev6}/PKG-INFO +1 -1
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/application.py +3 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/bluetooth.py +34 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/pages/device.py +6 -1
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/pages/home.py +2 -7
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/protocol.py +27 -21
- something_x_dev-1.3.0.dev6/nothing_app/tray.py +149 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/pyproject.toml +1 -1
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6/something_x_dev.egg-info}/PKG-INFO +1 -1
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/SOURCES.txt +1 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/LICENSE +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/README.md +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/__init__.py +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/data/__init__.py +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/data/com.something.x.omarchy.desktop +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/data/style.css +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/pages/__init__.py +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/profiles.py +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/splash.py +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/nothing_app/window.py +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/setup.cfg +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/dependency_links.txt +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/entry_points.txt +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/requires.txt +0 -0
- {something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/top_level.txt +0 -0
|
@@ -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__(
|
|
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
|
-
|
|
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[
|
|
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
|
|
609
|
-
self._low_bat_notified.
|
|
610
|
-
|
|
611
|
-
|
|
612
|
-
|
|
613
|
-
|
|
614
|
-
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
|
|
619
|
-
|
|
620
|
-
|
|
621
|
-
|
|
622
|
-
|
|
623
|
-
|
|
624
|
-
|
|
625
|
-
|
|
626
|
-
|
|
627
|
-
|
|
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)
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
{something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/entry_points.txt
RENAMED
|
File without changes
|
{something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/requires.txt
RENAMED
|
File without changes
|
{something_x_dev-1.2.3.dev4 → something_x_dev-1.3.0.dev6}/something_x_dev.egg-info/top_level.txt
RENAMED
|
File without changes
|