nixos-configs/modules/ags/config/widgets/bluetooth/main.tsx

153 lines
4.4 KiB
TypeScript
Raw Normal View History

2024-12-04 00:18:11 -05:00
import { bind } from 'astal';
import { Gtk } from 'astal/gtk3';
import AstalBluetooth from 'gi://AstalBluetooth';
2024-12-04 23:29:00 -05:00
import { ListBox, ToggleButton } from '../misc/subclasses';
2024-12-04 00:18:11 -05:00
import Separator from '../misc/separator';
import DeviceWidget from './device';
2024-12-04 23:29:00 -05:00
const calculateDevSort = (dev: AstalBluetooth.Device) => {
let value = 0;
if (dev.get_connected()) {
2024-12-04 23:29:00 -05:00
value += 1000;
}
if (dev.get_paired()) {
2024-12-04 23:29:00 -05:00
value += 100;
}
if (dev.get_blocked()) {
2024-12-04 23:29:00 -05:00
value += 10;
}
if (dev.get_icon()) {
if (dev.get_icon() === 'audio-headset') {
2024-12-04 23:29:00 -05:00
value += 9;
}
if (dev.get_icon() === 'audio-headphones') {
2024-12-04 23:29:00 -05:00
value += 8;
}
if (dev.get_icon() === 'audio-card') {
2024-12-04 23:29:00 -05:00
value += 7;
}
if (dev.get_icon() === 'computer') {
2024-12-04 23:29:00 -05:00
value += 6;
}
if (dev.get_icon() === 'phone') {
2024-12-04 23:29:00 -05:00
value += 5;
}
}
return value;
};
2024-12-04 00:18:11 -05:00
export default () => {
const bluetooth = AstalBluetooth.get_default();
const deviceList = (
<scrollable
className="list"
2024-12-04 23:29:00 -05:00
css="min-height: 300px;"
hscroll={Gtk.PolicyType.NEVER}
vscroll={Gtk.PolicyType.AUTOMATIC}
>
2024-12-04 23:29:00 -05:00
<ListBox
selectionMode={Gtk.SelectionMode.SINGLE}
setup={(self) => {
2024-12-04 23:29:00 -05:00
bluetooth.devices
.filter((dev) => dev.get_name())
2024-12-04 23:29:00 -05:00
.forEach((dev) => {
self.add(<DeviceWidget dev={dev} />);
});
self.hook(bluetooth, 'device-added', (_, dev) => {
if (dev.get_name()) {
self.add(<DeviceWidget dev={dev} />);
2024-12-04 23:29:00 -05:00
self.invalidate_sort();
}
});
self.hook(bluetooth, 'device-removed', (_, dev) => {
2024-12-04 23:29:00 -05:00
const children = self
.get_children()
.map((ch) => ch.get_child()) as DeviceWidget[];
const devWidget = children.find((ch) => ch.dev === dev);
if (devWidget) {
devWidget.set_reveal_child(false);
setTimeout(() => {
2024-12-04 23:29:00 -05:00
devWidget.get_parent()?.destroy();
}, devWidget.get_transition_duration() + 100);
}
});
2024-12-04 23:29:00 -05:00
self.set_sort_func((a, b) => {
const devA = (a.get_child() as DeviceWidget).dev;
const devB = (b.get_child() as DeviceWidget).dev;
const sort = calculateDevSort(devB) - calculateDevSort(devA);
return sort !== 0 ? sort : devA.get_name().localeCompare(devB.get_name());
2024-12-04 23:29:00 -05:00
});
}}
2024-12-04 23:29:00 -05:00
/>
</scrollable>
);
2024-12-04 00:18:11 -05:00
return (
<box
className="bluetooth widget"
vertical
>
<centerbox homogeneous>
<switch
cursor="pointer"
valign={Gtk.Align.CENTER}
halign={Gtk.Align.START}
active={bind(bluetooth, 'isPowered')}
setup={(self) => {
self.connect('notify::active', () => {
bluetooth.get_adapter()?.set_powered(self.active);
2024-12-04 00:18:11 -05:00
});
}}
/>
<box />
<ToggleButton
cursor="pointer"
halign={Gtk.Align.END}
className="toggle-button"
sensitive={bind(bluetooth, 'isPowered')}
onToggled={(self) => {
self.toggleClassName('active', self.active);
if (self.active) {
bluetooth.get_adapter()?.start_discovery();
2024-12-04 00:18:11 -05:00
}
else {
bluetooth.get_adapter()?.stop_discovery();
2024-12-04 00:18:11 -05:00
}
}}
>
<icon icon="emblem-synchronizing-symbolic" css="font-size: 30px;" />
2024-12-04 00:18:11 -05:00
</ToggleButton>
</centerbox>
<Separator size={8} vertical />
{deviceList}
2024-12-04 00:18:11 -05:00
</box>
);
};