something-x-dev 1.4.0.dev7__py3-none-any.whl → 1.5.0.dev9__py3-none-any.whl
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.
- nothing_app/data/style.css +7 -0
- nothing_app/pages/home.py +29 -5
- something_x_dev-1.5.0.dev9.dist-info/METADATA +252 -0
- {something_x_dev-1.4.0.dev7.dist-info → something_x_dev-1.5.0.dev9.dist-info}/RECORD +8 -8
- something_x_dev-1.4.0.dev7.dist-info/METADATA +0 -201
- {something_x_dev-1.4.0.dev7.dist-info → something_x_dev-1.5.0.dev9.dist-info}/WHEEL +0 -0
- {something_x_dev-1.4.0.dev7.dist-info → something_x_dev-1.5.0.dev9.dist-info}/entry_points.txt +0 -0
- {something_x_dev-1.4.0.dev7.dist-info → something_x_dev-1.5.0.dev9.dist-info}/licenses/LICENSE +0 -0
- {something_x_dev-1.4.0.dev7.dist-info → something_x_dev-1.5.0.dev9.dist-info}/top_level.txt +0 -0
nothing_app/data/style.css
CHANGED
|
@@ -499,6 +499,13 @@ switch:checked slider {
|
|
|
499
499
|
}
|
|
500
500
|
.connecting-button label { color: rgba(255, 255, 255, 0.22); }
|
|
501
501
|
|
|
502
|
+
/* compact variant for inline row buttons (home page quick connect) */
|
|
503
|
+
.row-action {
|
|
504
|
+
padding: 7px 16px;
|
|
505
|
+
font-size: 10px;
|
|
506
|
+
letter-spacing: 1.2px;
|
|
507
|
+
}
|
|
508
|
+
|
|
502
509
|
/* ── Empty state ─────────────────────────────────────────────────────────── */
|
|
503
510
|
|
|
504
511
|
.empty-title {
|
nothing_app/pages/home.py
CHANGED
|
@@ -7,10 +7,11 @@ from ..bluetooth import BluetoothDevice, BluetoothManager, device_icon_name
|
|
|
7
7
|
|
|
8
8
|
|
|
9
9
|
class DeviceRow(Gtk.Box):
|
|
10
|
-
def __init__(self, device: BluetoothDevice):
|
|
10
|
+
def __init__(self, device: BluetoothDevice, on_action=None):
|
|
11
11
|
super().__init__(orientation=Gtk.Orientation.HORIZONTAL, spacing=14)
|
|
12
12
|
self.add_css_class("device-card")
|
|
13
13
|
self.device = device
|
|
14
|
+
self._on_action = on_action
|
|
14
15
|
self._build()
|
|
15
16
|
|
|
16
17
|
def _build(self):
|
|
@@ -55,9 +56,22 @@ class DeviceRow(Gtk.Box):
|
|
|
55
56
|
text_box.append(bottom_row)
|
|
56
57
|
self.append(text_box)
|
|
57
58
|
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
59
|
+
if self._on_action is not None:
|
|
60
|
+
action_btn = Gtk.Button(label="DISCONNECT" if self.device.connected else "CONNECT")
|
|
61
|
+
action_btn.add_css_class("disconnect-button" if self.device.connected else "connect-button")
|
|
62
|
+
action_btn.add_css_class("row-action")
|
|
63
|
+
action_btn.set_valign(Gtk.Align.CENTER)
|
|
64
|
+
action_btn.connect("clicked", self._on_action_clicked)
|
|
65
|
+
self.append(action_btn)
|
|
66
|
+
else:
|
|
67
|
+
arrow = Gtk.Image.new_from_icon_name("go-next-symbolic")
|
|
68
|
+
arrow.set_opacity(0.3)
|
|
69
|
+
self.append(arrow)
|
|
70
|
+
|
|
71
|
+
def _on_action_clicked(self, btn: Gtk.Button):
|
|
72
|
+
btn.set_sensitive(False)
|
|
73
|
+
btn.set_label("…")
|
|
74
|
+
self._on_action(self.device)
|
|
61
75
|
|
|
62
76
|
|
|
63
77
|
class HomePage(Gtk.Box):
|
|
@@ -176,7 +190,11 @@ class HomePage(Gtk.Box):
|
|
|
176
190
|
self._other_label.set_visible(bool(others))
|
|
177
191
|
self._empty_box.set_visible(not devices)
|
|
178
192
|
|
|
179
|
-
def _make_row(self, device: BluetoothDevice) -> Gtk.
|
|
193
|
+
def _make_row(self, device: BluetoothDevice) -> Gtk.Widget:
|
|
194
|
+
# Only Nothing devices have a detail page; other devices get an inline
|
|
195
|
+
# connect/disconnect button instead of navigation.
|
|
196
|
+
if not device.is_nothing:
|
|
197
|
+
return DeviceRow(device, on_action=self._on_quick_action)
|
|
180
198
|
btn = Gtk.Button()
|
|
181
199
|
btn.set_has_frame(False)
|
|
182
200
|
btn.add_css_class("device-row-btn")
|
|
@@ -185,6 +203,12 @@ class HomePage(Gtk.Box):
|
|
|
185
203
|
btn.connect("clicked", lambda _b, d=device: self.emit("device-selected", d))
|
|
186
204
|
return btn
|
|
187
205
|
|
|
206
|
+
def _on_quick_action(self, device: BluetoothDevice):
|
|
207
|
+
if device.connected:
|
|
208
|
+
self._bt.disconnect_device(device.path)
|
|
209
|
+
else:
|
|
210
|
+
self._bt.connect_device(device.path, on_error=self._refresh_list)
|
|
211
|
+
|
|
188
212
|
def _on_devices_changed(self, _manager):
|
|
189
213
|
self._refresh_list()
|
|
190
214
|
|
|
@@ -0,0 +1,252 @@
|
|
|
1
|
+
Metadata-Version: 2.4
|
|
2
|
+
Name: something-x-dev
|
|
3
|
+
Version: 1.5.0.dev9
|
|
4
|
+
Summary: Something X device manager for Omarchy / Linux
|
|
5
|
+
Author: Raphael
|
|
6
|
+
License: MIT
|
|
7
|
+
Keywords: nothing,bluetooth,gtk4,linux,omarchy,ear
|
|
8
|
+
Classifier: Development Status :: 4 - Beta
|
|
9
|
+
Classifier: Environment :: X11 Applications :: GTK
|
|
10
|
+
Classifier: Intended Audience :: End Users/Desktop
|
|
11
|
+
Classifier: License :: OSI Approved :: MIT License
|
|
12
|
+
Classifier: Operating System :: POSIX :: Linux
|
|
13
|
+
Classifier: Programming Language :: Python :: 3
|
|
14
|
+
Classifier: Topic :: Utilities
|
|
15
|
+
Requires-Python: >=3.11
|
|
16
|
+
Description-Content-Type: text/markdown
|
|
17
|
+
License-File: LICENSE
|
|
18
|
+
Requires-Dist: PyGObject>=3.42
|
|
19
|
+
Requires-Dist: dbus-python>=1.3
|
|
20
|
+
Provides-Extra: dev
|
|
21
|
+
Requires-Dist: ruff; extra == "dev"
|
|
22
|
+
Dynamic: license-file
|
|
23
|
+
|
|
24
|
+
<div align="center">
|
|
25
|
+
|
|
26
|
+
# Something X
|
|
27
|
+
|
|
28
|
+
**A Linux-native companion app for Nothing and CMF Bluetooth devices.**
|
|
29
|
+
Built for [Omarchy](https://omarchy.org) · GTK4 · Pure black · JetBrains Mono · Nothing Red
|
|
30
|
+
|
|
31
|
+
[](https://pypi.org/project/something-x/)
|
|
32
|
+
[](https://aur.archlinux.org/packages/something-x)
|
|
33
|
+
[](LICENSE)
|
|
34
|
+
[](https://github.com/SoaOaoS/something-x)
|
|
35
|
+
|
|
36
|
+
</div>
|
|
37
|
+
|
|
38
|
+
---
|
|
39
|
+
|
|
40
|
+
## Features
|
|
41
|
+
|
|
42
|
+
| | Feature | Details |
|
|
43
|
+
|---|---|---|
|
|
44
|
+
| 🎧 | **Earbud visual** | Cairo-rendered glowing battery rings for L / R / Case, live updates |
|
|
45
|
+
| 🔇 | **ANC control** | Off · Noise Cancellation · Transparency over real RFCOMM protocol |
|
|
46
|
+
| 🎵 | **EQ presets** | Balanced · More Bass · More Treble · Voice |
|
|
47
|
+
| 🔊 | **Volume slider** | Direct PulseAudio / PipeWire A2DP sink control via `pactl` |
|
|
48
|
+
| 💾 | **Per-device profiles** | ANC + EQ saved per device address, restored automatically on reconnect |
|
|
49
|
+
| 🔋 | **Battery notifications** | Desktop alerts at 20 %, 15 %, 10 %, and 5 % per earbud and case |
|
|
50
|
+
| 🔗 | **Auto-connect RFCOMM** | Connects to the device protocol as soon as BlueZ reports it paired |
|
|
51
|
+
| 🏃 | **Background mode** | Closing the window keeps the app running; relaunch to reopen |
|
|
52
|
+
| 💻 | **CLI** | Control your earbuds from the terminal without opening the GUI |
|
|
53
|
+
| 📱 | **Device discovery** | BlueZ D-Bus scan with Nothing / CMF devices highlighted |
|
|
54
|
+
| ℹ️ | **Device info** | Firmware version and serial number read over RFCOMM |
|
|
55
|
+
|
|
56
|
+
---
|
|
57
|
+
|
|
58
|
+
## Device support
|
|
59
|
+
|
|
60
|
+
| Device | Battery | ANC | EQ | Volume | Firmware |
|
|
61
|
+
|---|:---:|:---:|:---:|:---:|:---:|
|
|
62
|
+
| Nothing Ear (1) | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
63
|
+
| Nothing Ear (2) | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
64
|
+
| Nothing Ear (a) | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
65
|
+
| Nothing Ear (stick) | ✅ | — | ✅ | ✅ | ✅ |
|
|
66
|
+
| CMF Buds / Buds Pro | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
67
|
+
| Nothing Phone (1/2) | ✅ | — | — | — | — |
|
|
68
|
+
| Other Bluetooth devices | ✅* | — | — | ✅ | — |
|
|
69
|
+
|
|
70
|
+
<sub>* via BlueZ `Battery1` interface · RFCOMM features require an active connection</sub>
|
|
71
|
+
|
|
72
|
+
---
|
|
73
|
+
|
|
74
|
+
## Installation
|
|
75
|
+
|
|
76
|
+
### Arch / Omarchy (recommended)
|
|
77
|
+
|
|
78
|
+
Install system dependencies first:
|
|
79
|
+
|
|
80
|
+
```bash
|
|
81
|
+
sudo pacman -S python-gobject python-dbus python-cairo gtk4 libadwaita
|
|
82
|
+
```
|
|
83
|
+
|
|
84
|
+
Then install from **AUR**:
|
|
85
|
+
|
|
86
|
+
```bash
|
|
87
|
+
yay -S something-x
|
|
88
|
+
# or: paru -S something-x
|
|
89
|
+
```
|
|
90
|
+
|
|
91
|
+
Or via **pip**:
|
|
92
|
+
|
|
93
|
+
```bash
|
|
94
|
+
pip install something-x
|
|
95
|
+
```
|
|
96
|
+
|
|
97
|
+
### Other distros
|
|
98
|
+
|
|
99
|
+
<details>
|
|
100
|
+
<summary>Ubuntu 24.04+</summary>
|
|
101
|
+
|
|
102
|
+
```bash
|
|
103
|
+
sudo apt install python3-gi python3-dbus python3-cairo gir1.2-gtk-4.0 gir1.2-adw-1
|
|
104
|
+
pip install something-x
|
|
105
|
+
```
|
|
106
|
+
|
|
107
|
+
</details>
|
|
108
|
+
|
|
109
|
+
<details>
|
|
110
|
+
<summary>Fedora 39+</summary>
|
|
111
|
+
|
|
112
|
+
```bash
|
|
113
|
+
sudo dnf install python3-gobject python3-dbus python3-cairo gtk4 libadwaita
|
|
114
|
+
pip install something-x
|
|
115
|
+
```
|
|
116
|
+
|
|
117
|
+
</details>
|
|
118
|
+
|
|
119
|
+
<details>
|
|
120
|
+
<summary>NixOS</summary>
|
|
121
|
+
|
|
122
|
+
```bash
|
|
123
|
+
nix run github:SoaOaoS/something-x
|
|
124
|
+
```
|
|
125
|
+
|
|
126
|
+
A `flake.nix` is included for reproducible builds.
|
|
127
|
+
|
|
128
|
+
</details>
|
|
129
|
+
|
|
130
|
+
### Run from source
|
|
131
|
+
|
|
132
|
+
```bash
|
|
133
|
+
git clone https://github.com/SoaOaoS/something-x
|
|
134
|
+
cd something-x
|
|
135
|
+
pip install -e .
|
|
136
|
+
something-x
|
|
137
|
+
```
|
|
138
|
+
|
|
139
|
+
---
|
|
140
|
+
|
|
141
|
+
## Usage
|
|
142
|
+
|
|
143
|
+
### GUI
|
|
144
|
+
|
|
145
|
+
```bash
|
|
146
|
+
something-x
|
|
147
|
+
```
|
|
148
|
+
|
|
149
|
+
1. **Splash** — animated intro, main window appears after ~2 s
|
|
150
|
+
2. **Home** — lists all paired Bluetooth devices; Nothing / CMF devices get a `NOTHING` badge
|
|
151
|
+
3. **Scan** — tap `SCAN FOR DEVICES` to run a 30 s BlueZ discovery
|
|
152
|
+
4. **Device page** — tap a card to open controls:
|
|
153
|
+
- Battery rings (L / R / Case) update in real time
|
|
154
|
+
- ANC and EQ apply immediately over RFCOMM and are saved to your profile
|
|
155
|
+
- Volume slider drives the A2DP sink via `pactl`
|
|
156
|
+
- Firmware version and serial number appear after RFCOMM connects
|
|
157
|
+
5. **Close** — hides to background; run `something-x` again to reopen
|
|
158
|
+
|
|
159
|
+
### CLI
|
|
160
|
+
|
|
161
|
+
Control your earbuds without opening the GUI:
|
|
162
|
+
|
|
163
|
+
```bash
|
|
164
|
+
# Battery levels
|
|
165
|
+
something-x --battery
|
|
166
|
+
|
|
167
|
+
# ANC mode
|
|
168
|
+
something-x --anc off
|
|
169
|
+
something-x --anc on
|
|
170
|
+
something-x --anc transparency
|
|
171
|
+
|
|
172
|
+
# EQ preset
|
|
173
|
+
something-x --eq balanced
|
|
174
|
+
something-x --eq bass
|
|
175
|
+
something-x --eq treble
|
|
176
|
+
something-x --eq voice
|
|
177
|
+
|
|
178
|
+
# Combine
|
|
179
|
+
something-x --anc on --eq bass
|
|
180
|
+
|
|
181
|
+
# Target a specific device by address
|
|
182
|
+
something-x --device AA:BB:CC:DD:EE:FF --battery
|
|
183
|
+
```
|
|
184
|
+
|
|
185
|
+
---
|
|
186
|
+
|
|
187
|
+
## Development releases
|
|
188
|
+
|
|
189
|
+
The `develop` branch publishes pre-release builds to PyPI automatically as `something-x-dev`:
|
|
190
|
+
|
|
191
|
+
```bash
|
|
192
|
+
pip install something-x-dev
|
|
193
|
+
something-x-dev
|
|
194
|
+
```
|
|
195
|
+
|
|
196
|
+
Dev builds use version numbers like `1.3.0.dev42`. Not for production use.
|
|
197
|
+
|
|
198
|
+
---
|
|
199
|
+
|
|
200
|
+
## Releases & versioning
|
|
201
|
+
|
|
202
|
+
Pushing to `main` triggers automatic versioning, a GitHub Release, a PyPI publish, and an AUR update — all from Conventional Commits:
|
|
203
|
+
|
|
204
|
+
| Commit prefix | Version bump |
|
|
205
|
+
|---|---|
|
|
206
|
+
| `feat!:` / `BREAKING CHANGE:` | Major `x.0.0` |
|
|
207
|
+
| `feat:` | Minor `1.x.0` |
|
|
208
|
+
| `fix:` / `perf:` / `refactor:` | Patch `1.0.x` |
|
|
209
|
+
| `docs:` / `chore:` / `style:` / `ci:` / `test:` | No release |
|
|
210
|
+
|
|
211
|
+
---
|
|
212
|
+
|
|
213
|
+
## Architecture
|
|
214
|
+
|
|
215
|
+
```
|
|
216
|
+
nothing_app/
|
|
217
|
+
├── application.py Adw.Application — lifecycle, CSS, CLI arg handling, background mode
|
|
218
|
+
├── window.py AdwNavigationView — home ↔ device routing, RFCOMM auto-connect manager
|
|
219
|
+
├── bluetooth.py BlueZ D-Bus manager — device discovery, connect/disconnect signals
|
|
220
|
+
├── protocol.py Nothing Ear 0x55 RFCOMM protocol (reverse-engineered from APK)
|
|
221
|
+
├── profiles.py Per-device ANC/EQ persistence (~/.config/something-x/profiles.json)
|
|
222
|
+
├── splash.py Animated splash screen (Cairo, typewriter, ripple rings)
|
|
223
|
+
├── data/
|
|
224
|
+
│ └── style.css Glass-morphism CSS theme
|
|
225
|
+
└── pages/
|
|
226
|
+
├── home.py Device list + scan button
|
|
227
|
+
└── device.py ANC / EQ / volume / settings + Cairo earbud visual
|
|
228
|
+
```
|
|
229
|
+
|
|
230
|
+
### Protocol
|
|
231
|
+
|
|
232
|
+
Frame format: `[SOF=0x55][ctrl:2 LE][cmd:2 LE][len:2 LE][FSN:1][payload][CRC16:2 LE]`
|
|
233
|
+
|
|
234
|
+
All outgoing frames use `ctrl=0x0160` with CRC16-ARC. The device silently drops SET commands if any frame in the session was sent without CRC.
|
|
235
|
+
|
|
236
|
+
Enable raw frame logging:
|
|
237
|
+
|
|
238
|
+
```bash
|
|
239
|
+
SOMETHING_X_DEBUG=1 something-x
|
|
240
|
+
```
|
|
241
|
+
|
|
242
|
+
---
|
|
243
|
+
|
|
244
|
+
## Contributing
|
|
245
|
+
|
|
246
|
+
The RFCOMM protocol in [`nothing_app/protocol.py`](nothing_app/protocol.py) is reverse-engineered from the official Android APK. If your device uses different command IDs, channels, or ANC values, patches are very welcome — please include the raw RFCOMM dump (`SOMETHING_X_DEBUG=1`) in your issue or PR.
|
|
247
|
+
|
|
248
|
+
---
|
|
249
|
+
|
|
250
|
+
## License
|
|
251
|
+
|
|
252
|
+
[MIT](LICENSE)
|
|
@@ -8,13 +8,13 @@ nothing_app/tray.py,sha256=ofT4nIoAIoAz2JWK36_AkxmHdW2fqtcrzrVE-c1kXXo,5826
|
|
|
8
8
|
nothing_app/window.py,sha256=pk6LTcqR6mcPMVYx2zagEngXnK8vlcw7zpKWj7ue7pA,4114
|
|
9
9
|
nothing_app/data/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
10
10
|
nothing_app/data/com.something.x.omarchy.desktop,sha256=Uet1FGwgMDvHlejxNuhR8cTWNrYEqnJlTvipl2ztFp8,362
|
|
11
|
-
nothing_app/data/style.css,sha256=
|
|
11
|
+
nothing_app/data/style.css,sha256=jUGlObQoSBqe9uBltxPRtq0gjXfwfW7kqPc2KznGi1U,16373
|
|
12
12
|
nothing_app/pages/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
|
13
13
|
nothing_app/pages/device.py,sha256=MvFarW7zHF45jHobjR2pzK9eLJ-Q-9-2k7c5y2aGiBE,22845
|
|
14
|
-
nothing_app/pages/home.py,sha256=
|
|
15
|
-
something_x_dev-1.
|
|
16
|
-
something_x_dev-1.
|
|
17
|
-
something_x_dev-1.
|
|
18
|
-
something_x_dev-1.
|
|
19
|
-
something_x_dev-1.
|
|
20
|
-
something_x_dev-1.
|
|
14
|
+
nothing_app/pages/home.py,sha256=n4gj-aaKObFNwrASki2Sway6trIvZvSUCn9iHbhyjqY,8499
|
|
15
|
+
something_x_dev-1.5.0.dev9.dist-info/licenses/LICENSE,sha256=f82aGY-Qd4Huw5T9EsynF6CJhVPdcnsTSuuaskxc1ak,1064
|
|
16
|
+
something_x_dev-1.5.0.dev9.dist-info/METADATA,sha256=rG8HF32cDEl2-flWy5kKeyZtVLVjxGaMLgW2snuaVkE,7237
|
|
17
|
+
something_x_dev-1.5.0.dev9.dist-info/WHEEL,sha256=aeYiig01lYGDzBgS8HxWXOg3uV61G9ijOsup-k9o1sk,91
|
|
18
|
+
something_x_dev-1.5.0.dev9.dist-info/entry_points.txt,sha256=HpMQVOiNSmzNKoa2Fb65XtbK7NWTfxMDBrBNlk9WHi8,65
|
|
19
|
+
something_x_dev-1.5.0.dev9.dist-info/top_level.txt,sha256=yERYCJXvIBXR40iWxixVVZI3z-B4gNYXEdXFiiWa7KI,12
|
|
20
|
+
something_x_dev-1.5.0.dev9.dist-info/RECORD,,
|
|
@@ -1,201 +0,0 @@
|
|
|
1
|
-
Metadata-Version: 2.4
|
|
2
|
-
Name: something-x-dev
|
|
3
|
-
Version: 1.4.0.dev7
|
|
4
|
-
Summary: Something X device manager for Omarchy / Linux
|
|
5
|
-
Author: Raphael
|
|
6
|
-
License: MIT
|
|
7
|
-
Keywords: nothing,bluetooth,gtk4,linux,omarchy,ear
|
|
8
|
-
Classifier: Development Status :: 4 - Beta
|
|
9
|
-
Classifier: Environment :: X11 Applications :: GTK
|
|
10
|
-
Classifier: Intended Audience :: End Users/Desktop
|
|
11
|
-
Classifier: License :: OSI Approved :: MIT License
|
|
12
|
-
Classifier: Operating System :: POSIX :: Linux
|
|
13
|
-
Classifier: Programming Language :: Python :: 3
|
|
14
|
-
Classifier: Topic :: Utilities
|
|
15
|
-
Requires-Python: >=3.11
|
|
16
|
-
Description-Content-Type: text/markdown
|
|
17
|
-
License-File: LICENSE
|
|
18
|
-
Requires-Dist: PyGObject>=3.42
|
|
19
|
-
Requires-Dist: dbus-python>=1.3
|
|
20
|
-
Provides-Extra: dev
|
|
21
|
-
Requires-Dist: ruff; extra == "dev"
|
|
22
|
-
Dynamic: license-file
|
|
23
|
-
|
|
24
|
-
# Something X — for Linux
|
|
25
|
-
|
|
26
|
-
> A Linux-native companion app for **Nothing** and **CMF** Bluetooth devices.
|
|
27
|
-
> Built for [Omarchy](https://omarchy.org) (Hyprland / Wayland) — pure black, JetBrains Mono, Nothing Red.
|
|
28
|
-
|
|
29
|
-
```
|
|
30
|
-
● SOMETHING X
|
|
31
|
-
FOR LINUX
|
|
32
|
-
```
|
|
33
|
-
|
|
34
|
-
[](https://pypi.org/project/something-x/)
|
|
35
|
-
[](LICENSE)
|
|
36
|
-
[](https://github.com/SoaOaoS/something-x)
|
|
37
|
-
|
|
38
|
-
---
|
|
39
|
-
|
|
40
|
-
## Features
|
|
41
|
-
|
|
42
|
-
- **Animated splash screen** — Nothing-branded intro with typewriter effect and ripple rings
|
|
43
|
-
- **Earbud visual** — Cairo-rendered glowing battery rings with radial gradients for L / R / Case
|
|
44
|
-
- **ANC control** — Off · Noise Cancellation · Transparency (real RFCOMM protocol)
|
|
45
|
-
- **EQ presets** — Balanced · More Bass · More Treble · Voice
|
|
46
|
-
- **Volume slider** — controls the PulseAudio/PipeWire A2DP sink directly
|
|
47
|
-
- **Per-device profiles** — ANC and EQ saved per device, restored automatically on reconnect
|
|
48
|
-
- **Background mode** — closing the window keeps the app running; relaunch to reopen
|
|
49
|
-
- **CLI quick-toggles** — control your earbuds without opening the GUI (see [CLI usage](#cli-usage))
|
|
50
|
-
- **Low battery notifications** — `notify-send` alert when any bud drops below 20 %
|
|
51
|
-
- **Firmware version & serial number** — read from the device over RFCOMM
|
|
52
|
-
- **In-ear detection toggle**
|
|
53
|
-
- **Device discovery** — BlueZ D-Bus; Nothing/CMF devices highlighted with a badge
|
|
54
|
-
- **Scan for new devices** — 30 s BlueZ discovery window
|
|
55
|
-
- **Glass morphism UI** — pure black base, frosted glass cards, red gradient accents
|
|
56
|
-
|
|
57
|
-
---
|
|
58
|
-
|
|
59
|
-
## Device support
|
|
60
|
-
|
|
61
|
-
| Device | Discovery | Battery | ANC | EQ | Volume | Firmware |
|
|
62
|
-
|---|---|---|---|---|---|---|
|
|
63
|
-
| Nothing Ear (1) | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
64
|
-
| Nothing Ear (2) | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
65
|
-
| Nothing Ear (a) | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
66
|
-
| Nothing Ear (stick) | ✅ | ✅ | — | ✅ | ✅ | ✅ |
|
|
67
|
-
| CMF Buds / Buds Pro | ✅ | ✅ | ✅ | ✅ | ✅ | ✅ |
|
|
68
|
-
| Nothing Phone (1/2) | ✅ | — | — | — | — | — |
|
|
69
|
-
| Other BT devices | ✅ | ✅* | — | — | ✅ | — |
|
|
70
|
-
|
|
71
|
-
\* via BlueZ `Battery1` interface · RFCOMM features require the device to be connected
|
|
72
|
-
|
|
73
|
-
---
|
|
74
|
-
|
|
75
|
-
## Requirements
|
|
76
|
-
|
|
77
|
-
### System packages (Arch / Omarchy)
|
|
78
|
-
|
|
79
|
-
```bash
|
|
80
|
-
sudo pacman -S python-gobject python-dbus python-cairo gtk4 libadwaita
|
|
81
|
-
```
|
|
82
|
-
|
|
83
|
-
| Package | Purpose |
|
|
84
|
-
|---|---|
|
|
85
|
-
| `python-gobject` | GTK4, libadwaita, GLib bindings |
|
|
86
|
-
| `python-dbus` | BlueZ D-Bus access |
|
|
87
|
-
| `python-cairo` | Cairo drawing (earbud visual, splash) |
|
|
88
|
-
| `gtk4` | UI toolkit |
|
|
89
|
-
| `libadwaita` | Navigation, dark theme |
|
|
90
|
-
|
|
91
|
-
> `pactl` (from `libpulse` / `pipewire-pulse`) is used for volume control — already present on any PulseAudio/PipeWire system.
|
|
92
|
-
|
|
93
|
-
---
|
|
94
|
-
|
|
95
|
-
## Installation
|
|
96
|
-
|
|
97
|
-
### Recommended — pip (after system packages above)
|
|
98
|
-
|
|
99
|
-
```bash
|
|
100
|
-
pip install something-x
|
|
101
|
-
something-x
|
|
102
|
-
```
|
|
103
|
-
|
|
104
|
-
### Run from source
|
|
105
|
-
|
|
106
|
-
```bash
|
|
107
|
-
git clone https://github.com/SoaOaoS/something-x
|
|
108
|
-
cd something-x
|
|
109
|
-
./somethingx
|
|
110
|
-
```
|
|
111
|
-
|
|
112
|
-
### Desktop launcher (Walker / Rofi / app menu)
|
|
113
|
-
|
|
114
|
-
```bash
|
|
115
|
-
cp nothing_app/data/com.something.x.omarchy.desktop ~/.local/share/applications/
|
|
116
|
-
update-desktop-database ~/.local/share/applications/
|
|
117
|
-
```
|
|
118
|
-
|
|
119
|
-
---
|
|
120
|
-
|
|
121
|
-
## Usage
|
|
122
|
-
|
|
123
|
-
```
|
|
124
|
-
./somethingx # from source
|
|
125
|
-
something-x # if installed via pip
|
|
126
|
-
```
|
|
127
|
-
|
|
128
|
-
1. **Splash** — animated intro, main window opens after ~2.3 s
|
|
129
|
-
2. **Home** — all paired BT devices; Nothing/CMF get a `NOTHING` badge
|
|
130
|
-
3. **Scan** — "SCAN FOR DEVICES" runs 30 s BlueZ discovery
|
|
131
|
-
4. **Device page** — tap a card to open controls:
|
|
132
|
-
- Battery rings (L / R / Case) update in real time
|
|
133
|
-
- ANC and EQ apply immediately over RFCOMM; settings saved automatically
|
|
134
|
-
- Volume slider controls the A2DP sink via `pactl`
|
|
135
|
-
- Firmware and serial number shown after connection
|
|
136
|
-
5. **Disconnect** — red button sends a clean BlueZ disconnect
|
|
137
|
-
6. **Close** — hides to background; run `something-x` again to reopen
|
|
138
|
-
|
|
139
|
-
---
|
|
140
|
-
|
|
141
|
-
## CLI usage
|
|
142
|
-
|
|
143
|
-
After connecting to a device at least once via the GUI, you can control it from the terminal:
|
|
144
|
-
|
|
145
|
-
```bash
|
|
146
|
-
something-x --battery # print battery levels
|
|
147
|
-
something-x --anc off|on|transparency # set ANC mode
|
|
148
|
-
something-x --eq balanced|bass|treble|voice # set EQ preset
|
|
149
|
-
something-x --anc on --eq bass # combine actions
|
|
150
|
-
something-x --device AA:BB:CC:DD:EE:FF --battery # target a specific device
|
|
151
|
-
```
|
|
152
|
-
|
|
153
|
-
---
|
|
154
|
-
|
|
155
|
-
## Releases & versioning
|
|
156
|
-
|
|
157
|
-
This project uses **Conventional Commits**. Pushing to `main` triggers automatic versioning and a PyPI release:
|
|
158
|
-
|
|
159
|
-
| Commit prefix | Version bump | Example |
|
|
160
|
-
|---|---|---|
|
|
161
|
-
| `feat!:` / `BREAKING CHANGE` | Major (`x.0.0`) | `feat!: new protocol engine` |
|
|
162
|
-
| `feat:` | Minor (`1.x.0`) | `feat: add Ear (open) support` |
|
|
163
|
-
| `fix:` / `perf:` / `refactor:` | Patch (`1.0.x`) | `fix: ANC off not applying` |
|
|
164
|
-
| `docs:` / `chore:` / `style:` / `ci:` | — (no release) | `chore: update readme` |
|
|
165
|
-
|
|
166
|
-
---
|
|
167
|
-
|
|
168
|
-
## Architecture
|
|
169
|
-
|
|
170
|
-
```
|
|
171
|
-
nothing_app/
|
|
172
|
-
├── application.py Adw.Application — CSS, dark theme, splash, background mode, CLI
|
|
173
|
-
├── splash.py Animated splash screen (Cairo, typewriter, ripples)
|
|
174
|
-
├── window.py AdwNavigationView — home ↔ device routing
|
|
175
|
-
├── bluetooth.py BlueZ D-Bus manager (discovery, connect/disconnect signals)
|
|
176
|
-
├── protocol.py Nothing Ear RFCOMM 0x55 binary protocol (reverse-engineered)
|
|
177
|
-
├── profiles.py Per-device ANC/EQ profile persistence (~/.config/something-x/)
|
|
178
|
-
├── data/
|
|
179
|
-
│ └── style.css Nothing X glass-morphism CSS theme
|
|
180
|
-
└── pages/
|
|
181
|
-
├── home.py Device list + scan button
|
|
182
|
-
└── device.py ANC / EQ / volume / settings + Cairo earbud visual
|
|
183
|
-
```
|
|
184
|
-
|
|
185
|
-
### Protocol notes
|
|
186
|
-
|
|
187
|
-
Frame format: `[SOF=0x55][ctrl:2 LE][cmd:2 LE][len:2 LE][FSN:1][payload][crc16:2 LE]`
|
|
188
|
-
|
|
189
|
-
All outgoing frames use `ctrl=0x0160` with CRC16-ARC — the device silently drops SET commands if any frame in the session was sent without CRC.
|
|
190
|
-
|
|
191
|
-
---
|
|
192
|
-
|
|
193
|
-
## Contributing
|
|
194
|
-
|
|
195
|
-
The RFCOMM protocol in [nothing_app/protocol.py](nothing_app/protocol.py) is reverse-engineered from the official Android APK. If your device uses different command IDs or channel numbers, patches are very welcome.
|
|
196
|
-
|
|
197
|
-
---
|
|
198
|
-
|
|
199
|
-
## License
|
|
200
|
-
|
|
201
|
-
MIT
|
|
File without changes
|
{something_x_dev-1.4.0.dev7.dist-info → something_x_dev-1.5.0.dev9.dist-info}/entry_points.txt
RENAMED
|
File without changes
|
{something_x_dev-1.4.0.dev7.dist-info → something_x_dev-1.5.0.dev9.dist-info}/licenses/LICENSE
RENAMED
|
File without changes
|
|
File without changes
|