dotfiles/.config/hypr/scripts/Toggle-Active-Window-Audio.sh
2026-05-13 21:22:17 +02:00

153 lines
5.7 KiB
Bash
Executable file

#!/usr/bin/env bash
# ==================================================
# KoolDots (2026)
# Project URL: https://github.com/LinuxBeginnings
# License: GNU GPLv3
# SPDX-License-Identifier: GPL-3.0-or-later
# ==================================================
set -euo pipefail
XDG_CONFIG_HOME="${XDG_CONFIG_HOME:-$HOME/.config}"
swayIconDir="${XDG_CONFIG_HOME}/swaync/icons"
#// Credits to sl1ng for the orginal script. Rewritten by Vyle.
ctlcheck=("pactl" "jq" "notify-send" "awk" "pgrep" "hyprctl" "iconv")
missing=()
for ctl in "${ctlcheck[@]}"; do
command -v "${ctl}" >/dev/null || missing+=("${ctl}")
done
if (( ${#missing[@]} )) 2>/dev/null; then
if printf '%s\n' "${missing[@]}" | grep -qx "pactl"; then
notify-send -a "t1" -r 91190 -t 2000 -i "${swayIconDir}/volume-low.png" "ERROR: pactl not installed" "Install 'pactl' (pulseaudio-utils or pipewire-pulse)."
fi
echo "Missing required dependencies: \"${missing[*]}\""
exit 1
fi
#// Parse .pid, .class, .title to __pid, __class, __title.
active_json="$(hyprctl -j activewindow 2>/dev/null || { echo -e "Did hyprctl fail to run? [EXIT-CODE:-1]"; exit 1; } )"
PID="$(jq -r '"\(.pid)\t\(.class)\t\(.title)"' <<< "${active_json}" || { echo -e "Did jq fail to run? [EXIT-CODE:-1]"; exit 1; } )"
IFS=$'\t' read -r __pid __class __title <<< "${PID}"
[[ -z "${__pid}" ]] && { echo -e "Could not resolve PID for focused window."; exit 1; }
sink_json="$(pactl -f json list sink-inputs 2>/dev/null | iconv -f utf-8 -t utf-8 -c || { echo -e "Did pactl or iconv fail to run? Required manual intervention."; exit 1; } )"
#// Collect all descendant PIDs for the active window (Chrome/Wayland audio often runs in child processes).
declare -A seen_pids=()
queue=("${__pid}")
all_pids=()
while ((${#queue[@]})); do
pid="${queue[0]}"
queue=("${queue[@]:1}")
[[ -n "${seen_pids[$pid]:-}" ]] && continue
seen_pids["$pid"]=1
all_pids+=("$pid")
mapfile -t children < <(pgrep -P "$pid" || true)
for child in "${children[@]}"; do
[[ -n "${seen_pids[$child]:-}" ]] || queue+=("$child")
done
done
pidsJson="$(printf '%s\n' "${all_pids[@]}" | jq -s 'map(tonumber)')"
#// Check if any descendant PID matches application.process.id or else verify other statements.
mapfile -t sink_ids < <(jq -r --argjson pids "${pidsJson}" --arg class "${__class}" --arg title "${__title}" '
.[] |
def lc(x): (x // "" | ascii_downcase);
def normalize(x): x | gsub("[-_~.]+";" ") ;
select(
(.properties["application.process.id"] | tostring | tonumber? as $p | $p != null and ($pids | index($p)))
or
(lc(.properties["application.name"]) | contains(lc($class)))
or
(lc(.properties["application.id"]) | contains(lc($class)))
or
(lc(.properties["application.process.binary"]) | contains(lc($class)))
or
(normalize(lc(.properties["media.name"])) | contains(normalize(lc($title))))
) | .index' <<< "${sink_json}"
)
if [[ "${#sink_ids[@]}" -eq 0 ]]; then
mapfile -t fallback_pids < <(pgrep -x "${__class}" || true)
if [[ "${#fallback_pids[@]}" -gt 0 ]]; then
declare -A seen_fallback=()
queue=("${fallback_pids[@]}")
all_fallback=()
while ((${#queue[@]})); do
pid="${queue[0]}"
queue=("${queue[@]:1}")
[[ -n "${seen_fallback[$pid]:-}" ]] && continue
seen_fallback["$pid"]=1
all_fallback+=("$pid")
mapfile -t children < <(pgrep -P "$pid" || true)
for child in "${children[@]}"; do
[[ -n "${seen_fallback[$child]:-}" ]] || queue+=("$child")
done
done
fallbackJson="$(printf '%s\n' "${all_fallback[@]}" | jq -s 'map(tonumber)')"
mapfile -t sink_ids < <( jq -r --argjson pids "${fallbackJson}" '.[] |
select((.properties["application.process.id"] | tostring | tonumber? as $p | $p != null and ($pids | index($p)))) | .index' <<< "${sink_json}" )
fi
fi
#// Auto-Detect if the environment is on Hyprland or $HYPRLAND_INSTANCE_SIGNATURE.
if [[ ${#sink_ids[@]} -eq 0 ]]; then
if [[ -n "${HYPRLAND_INSTANCE_SIGNATURE}" ]]; then
# Even if the fallback_pid remains empty, we will dispatch exit code based on $HYPRLAND_INSTANCE_SIGNATURE.
notify-send -a "t1" -r 91190 -t 1200 -i "${swayIconDir}/volume-low.png" "No sink input for the active_window: ${__class}"
echo "No sink input for focused window: ${__class}"
exit 1
else
echo "No sink input for focused active_window ${__class}"
exit 1
fi
fi
idsJson=$(printf '%s\n' "${sink_ids[@]}" | jq -s 'map(tonumber)')
#// Get the available option from pactl.
want_mute=$(jq -r --argjson ids "$idsJson" '
[ .[] | select(.index as $i | $ids | index($i)) | .mute ] as $m |
if all($m[]; . == true) then "no"
else "yes"
end' <<< "${sink_json}"
)
if [[ "${want_mute}" == "no" ]]; then
state_msg="Unmuted"
swayIcon="${swayIconDir}/volume-high.png"
else
state_msg="Muted"
swayIcon="${swayIconDir}/volume-mute.png"
fi
[[ -f "${swayIcon}" ]] || echo -e "Missing swaync icons."
changed=0
failed_ids=()
for id in "${sink_ids[@]}"; do
if pactl set-sink-input-mute "$id" "$want_mute"; then
changed=1
else
failed_ids+=("$id")
fi
done
if [[ "$changed" -eq 0 ]]; then
notify-send -a "t2" -r 91190 -t 1200 -i "${swayIconDir}/volume-low.png" "Failed to change sink input(s)" "${failed_ids[*]:-unknown}"
exit 1
fi
#// Append pamixer to get a nice result. Pamixer is complete optional here.
if command -v pamixer >/dev/null; then
sink_name="$(pamixer --get-default-sink 2>/dev/null | awk -F '"' 'END{print $(NF - 1)}' 2>/dev/null || true)"
if [[ -n "${sink_name}" ]]; then
notify-send -a "t2" -r 91190 -t 800 -i "${swayIcon}" "${state_msg} ${__class}" "${sink_name}"
else
notify-send -a "t2" -r 91190 -t 800 -i "${swayIcon}" "${state_msg} ${__class}"
fi
else
notify-send -a "t2" -r 91190 -t 800 -i "${swayIcon}" "${state_msg} ${__class}"
fi