neodotfiles/config/fuzzel/scripts/rs_blue

315 lines
7.6 KiB
Text
Raw Normal View History

2022-12-20 00:15:21 +00:00
#!/usr/bin/env bash
2023-10-21 13:44:39 +00:00
# __ _ _ _ _ _ _
# _ __ ___ / _(_) | |__ | |_ _ ___| |_ ___ ___ | |_| |__
# | '__/ _ \| |_| |_____| '_ \| | | | |/ _ \ __/ _ \ / _ \| __| '_ \
# | | | (_) | _| |_____| |_) | | |_| | __/ || (_) | (_) | |_| | | |
# |_| \___/|_| |_| |_.__/|_|\__,_|\___|\__\___/ \___/ \__|_| |_|
2022-12-20 00:15:21 +00:00
#
# Author: Nick Clyde (clydedroid)
#
2023-10-21 13:44:39 +00:00
# A script that generates a rofi menu that uses bluetoothctl to
2022-12-20 00:15:21 +00:00
# connect to bluetooth devices and display status info.
#
# Inspired by networkmanager-dmenu (https://github.com/firecat53/networkmanager-dmenu)
# Thanks to x70b1 (https://github.com/polybar/polybar-scripts/tree/master/polybar-scripts/system-bluetooth-bluetoothctl)
#
# Depends on:
2023-10-21 13:44:39 +00:00
# Arch repositories: rofi, bluez-utils (contains bluetoothctl)
2022-12-20 00:15:21 +00:00
# Constants
divider="---------"
goback="Back"
# Checks if bluetooth controller is powered on
power_on() {
2023-10-21 13:44:39 +00:00
if bluetoothctl show | grep -q "Powered: yes"; then
2022-12-20 00:15:21 +00:00
return 0
else
return 1
fi
}
# Toggles power state
toggle_power() {
if power_on; then
bluetoothctl power off
show_menu
else
2023-10-21 13:44:39 +00:00
if rfkill list bluetooth | grep -q 'blocked: yes'; then
2022-12-20 00:15:21 +00:00
rfkill unblock bluetooth && sleep 3
fi
bluetoothctl power on
show_menu
fi
}
# Checks if controller is scanning for new devices
scan_on() {
2023-10-21 13:44:39 +00:00
if bluetoothctl show | grep -q "Discovering: yes"; then
2022-12-20 00:15:21 +00:00
echo "Scan: on"
return 0
else
echo "Scan: off"
return 1
fi
}
# Toggles scanning state
toggle_scan() {
if scan_on; then
2023-10-21 13:44:39 +00:00
kill $(pgrep -f "bluetoothctl scan on")
2022-12-20 00:15:21 +00:00
bluetoothctl scan off
show_menu
else
bluetoothctl scan on &
echo "Scanning..."
sleep 5
show_menu
fi
}
# Checks if controller is able to pair to devices
pairable_on() {
2023-10-21 13:44:39 +00:00
if bluetoothctl show | grep -q "Pairable: yes"; then
2022-12-20 00:15:21 +00:00
echo "Pairable: on"
return 0
else
echo "Pairable: off"
return 1
fi
}
# Toggles pairable state
toggle_pairable() {
if pairable_on; then
bluetoothctl pairable off
show_menu
else
bluetoothctl pairable on
show_menu
fi
}
# Checks if controller is discoverable by other devices
discoverable_on() {
2023-10-21 13:44:39 +00:00
if bluetoothctl show | grep -q "Discoverable: yes"; then
2022-12-20 00:15:21 +00:00
echo "Discoverable: on"
return 0
else
echo "Discoverable: off"
return 1
fi
}
# Toggles discoverable state
toggle_discoverable() {
if discoverable_on; then
bluetoothctl discoverable off
show_menu
else
bluetoothctl discoverable on
show_menu
fi
}
# Checks if a device is connected
device_connected() {
device_info=$(bluetoothctl info "$1")
2023-10-21 13:44:39 +00:00
if echo "$device_info" | grep -q "Connected: yes"; then
2022-12-20 00:15:21 +00:00
return 0
else
return 1
fi
}
# Toggles device connection
toggle_connection() {
if device_connected "$1"; then
bluetoothctl disconnect "$1"
2023-10-21 13:44:39 +00:00
device_menu "$device"
2022-12-20 00:15:21 +00:00
else
bluetoothctl connect "$1"
2023-10-21 13:44:39 +00:00
device_menu "$device"
2022-12-20 00:15:21 +00:00
fi
}
# Checks if a device is paired
device_paired() {
device_info=$(bluetoothctl info "$1")
2023-10-21 13:44:39 +00:00
if echo "$device_info" | grep -q "Paired: yes"; then
2022-12-20 00:15:21 +00:00
echo "Paired: yes"
return 0
else
echo "Paired: no"
return 1
fi
}
# Toggles device paired state
toggle_paired() {
if device_paired "$1"; then
bluetoothctl remove "$1"
device_menu "$device"
else
bluetoothctl pair "$1"
device_menu "$device"
fi
}
# Checks if a device is trusted
device_trusted() {
device_info=$(bluetoothctl info "$1")
2023-10-21 13:44:39 +00:00
if echo "$device_info" | grep -q "Trusted: yes"; then
2022-12-20 00:15:21 +00:00
echo "Trusted: yes"
return 0
else
echo "Trusted: no"
return 1
fi
}
# Toggles device connection
toggle_trust() {
if device_trusted "$1"; then
bluetoothctl untrust "$1"
device_menu "$device"
else
bluetoothctl trust "$1"
device_menu "$device"
fi
}
# Prints a short string with the current bluetooth status
# Useful for status bars like polybar, etc.
print_status() {
if power_on; then
printf ''
2023-10-21 13:44:39 +00:00
paired_devices_cmd="devices Paired"
# Check if an outdated version of bluetoothctl is used to preserve backwards compatibility
if (( $(echo "$(bluetoothctl version | cut -d ' ' -f 2) < 5.65" | bc -l) )); then
paired_devices_cmd="paired-devices"
fi
mapfile -t paired_devices < <(bluetoothctl "$paired_devices_cmd" | grep Device | cut -d ' ' -f 2)
2022-12-20 00:15:21 +00:00
counter=0
for device in "${paired_devices[@]}"; do
if device_connected "$device"; then
2023-10-21 13:44:39 +00:00
device_alias=$(bluetoothctl info "$device" | grep "Alias" | cut -d ' ' -f 2-)
2022-12-20 00:15:21 +00:00
if [ $counter -gt 0 ]; then
printf ", %s" "$device_alias"
else
printf " %s" "$device_alias"
fi
((counter++))
fi
done
printf "\n"
else
echo ""
fi
}
# A submenu for a specific device that allows connecting, pairing, and trusting
device_menu() {
device=$1
# Get device name and mac address
2023-10-21 13:44:39 +00:00
device_name=$(echo "$device" | cut -d ' ' -f 3-)
mac=$(echo "$device" | cut -d ' ' -f 2)
2022-12-20 00:15:21 +00:00
# Build options
if device_connected "$mac"; then
connected="Connected: yes"
else
connected="Connected: no"
fi
paired=$(device_paired "$mac")
trusted=$(device_trusted "$mac")
options="$connected\n$paired\n$trusted\n$divider\n$goback\nExit"
2023-10-21 13:44:39 +00:00
# Open rofi menu, read chosen option
2023-10-26 19:56:17 +00:00
chosen="$(echo -e "$options" | $RUNNER -i -l 8 -p "[$device_name]  ")"
2022-12-20 00:15:21 +00:00
# Match chosen option to command
2023-10-21 13:44:39 +00:00
case "$chosen" in
2022-12-20 00:15:21 +00:00
"" | "$divider")
echo "No option chosen."
;;
"$connected")
toggle_connection "$mac"
;;
"$paired")
toggle_paired "$mac"
;;
"$trusted")
toggle_trust "$mac"
;;
"$goback")
show_menu
;;
esac
}
2023-10-21 13:44:39 +00:00
# Opens a rofi menu with current bluetooth status and options to connect
2022-12-20 00:15:21 +00:00
show_menu() {
# Get menu options
if power_on; then
power="Power: on"
# Human-readable names of devices, one per line
# If scan is off, will only list paired devices
2023-10-21 13:44:39 +00:00
devices=$(bluetoothctl devices | grep Device | cut -d ' ' -f 3-)
2022-12-20 00:15:21 +00:00
# Get controller flags
scan=$(scan_on)
pairable=$(pairable_on)
discoverable=$(discoverable_on)
2023-10-21 13:44:39 +00:00
# Options passed to rofi
2022-12-20 00:15:21 +00:00
options="$devices\n$divider\n$power\n$scan\n$pairable\n$discoverable\nExit"
else
power="Power: off"
options="$power\nExit"
fi
2023-10-21 13:44:39 +00:00
# Open rofi menu, read chosen option
2023-10-26 19:56:17 +00:00
chosen="$(echo -e "$options" | $RUNNER -i -l 7 -p "[ Bluetooth]  ")"
2022-12-20 00:15:21 +00:00
# Match chosen option to command
2023-10-21 13:44:39 +00:00
case "$chosen" in
2022-12-20 00:15:21 +00:00
"" | "$divider")
echo "No option chosen."
;;
"$power")
toggle_power
;;
"$scan")
toggle_scan
;;
"$discoverable")
toggle_discoverable
;;
"$pairable")
toggle_pairable
;;
*)
2023-10-21 13:44:39 +00:00
device=$(bluetoothctl devices | grep "$chosen")
2022-12-20 00:15:21 +00:00
# Open a submenu if a device is selected
if [[ $device ]]; then device_menu "$device"; fi
;;
esac
}
case "$1" in
--status)
print_status
;;
*)
show_menu
;;
esac