115 lines
4.1 KiB
Bash
Executable File
115 lines
4.1 KiB
Bash
Executable File
#!/usr/bin/env bash
|
|
# Script taken from and modified little bit from:
|
|
# https://github.com/dusklinux/dusky/blob/f8a28f425743e478cddefc327e6e554dfcf9521d/user_scripts/hypr/screen_rotate.sh
|
|
|
|
# 1. Strict Mode & Safety (Bash 5+ Standards)
|
|
# ------------------------------------------------------------------------------
|
|
set -euo pipefail
|
|
IFS=$'\n\t'
|
|
|
|
# 2. Global Constants (ANSI-C Quoting for "Elite" Color Handling)
|
|
# ------------------------------------------------------------------------------
|
|
readonly C_RED=$'\e[31m'
|
|
readonly C_GREEN=$'\e[32m'
|
|
readonly C_YELLOW=$'\e[33m'
|
|
readonly C_BLUE=$'\e[34m'
|
|
readonly C_BOLD=$'\e[1m'
|
|
readonly C_RESET=$'\e[0m'
|
|
|
|
# cleanup_trap: Ensures clean exit codes are respected.
|
|
cleanup_trap() {
|
|
local exit_code=$?
|
|
if [[ $exit_code -ne 0 ]]; then
|
|
printf "%s[ERROR]%s Script aborted unexpectedly (Exit Code: %d).\n" \
|
|
"$C_RED" "$C_RESET" "$exit_code" >&2
|
|
fi
|
|
}
|
|
trap cleanup_trap EXIT
|
|
|
|
# 3. Environment & Privilege Checks
|
|
# ------------------------------------------------------------------------------
|
|
# Dependency Check: We need 'jq' for JSON parsing.
|
|
if ! command -v jq &> /dev/null; then
|
|
printf "%s[ERROR]%s 'jq' is missing. Install it with: sudo pacman -S jq\n" \
|
|
"$C_RED" "$C_RESET" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# Root Check: Hyprland IPC fails if executed as root/sudo due to socket ownership.
|
|
if [[ $EUID -eq 0 ]]; then
|
|
printf "%s[ERROR]%s Root detected. Please run this as your normal user to access the Hyprland socket.\n" \
|
|
"$C_RED" "$C_RESET" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# 4. Argument Parsing (+90 or -90)
|
|
# ------------------------------------------------------------------------------
|
|
DIRECTION=0
|
|
|
|
if [[ $# -ne 1 ]]; then
|
|
printf "%s[INFO]%s Usage: %s [+90|-90]\n" \
|
|
"$C_YELLOW" "$C_RESET" "${0##*/}"
|
|
exit 1
|
|
fi
|
|
|
|
case "$1" in
|
|
"+90") DIRECTION=1 ;; # Clockwise
|
|
"-90") DIRECTION=-1 ;; # Counter-Clockwise
|
|
*)
|
|
printf "%s[ERROR]%s Invalid flag '%s'. Use +90 or -90.\n" \
|
|
"$C_RED" "$C_RESET" "$1" >&2
|
|
exit 1
|
|
;;
|
|
esac
|
|
|
|
# 5. Hardware Detection (Smart Query)
|
|
# ------------------------------------------------------------------------------
|
|
# We fetch the entire JSON blob once to minimize IPC calls (Performance).
|
|
# We select the current focused monitor
|
|
MON_STATE=$(hyprctl monitors -j | jq -r '.[] | select(.focused == true)')
|
|
|
|
# Extract precise values using jq
|
|
NAME=$(printf "%s" "$MON_STATE" | jq -r '.name')
|
|
SCALE=$(printf "%s" "$MON_STATE" | jq -r '.scale')
|
|
CURRENT_TRANSFORM=$(printf "%s" "$MON_STATE" | jq -r '.transform')
|
|
|
|
# Validation: Ensure we actually found a monitor
|
|
if [[ -z "$NAME" || "$NAME" == "null" ]]; then
|
|
printf "%s[ERROR]%s No active monitors detected via Hyprland IPC.\n" \
|
|
"$C_RED" "$C_RESET" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# 6. Transformation Logic (Modulo Arithmetic)
|
|
# ------------------------------------------------------------------------------
|
|
# Hyprland Transforms: 0=Normal, 1=90, 2=180, 3=270
|
|
# The '+ 4' ensures we handle negative wraparounds correctly in Bash logic.
|
|
NEW_TRANSFORM=$(( (CURRENT_TRANSFORM + DIRECTION + 4) % 4 ))
|
|
|
|
# 7. Execution (State overwrite)
|
|
# ------------------------------------------------------------------------------
|
|
# We use 'preferred' and 'auto' to remain robust against resolution changes,
|
|
# but we STRICTLY inject the detected $SCALE to prevent UI scaling issues.
|
|
|
|
printf "%s[INFO]%s Rotating %s%s%s (Scale: %s): %d -> %d\n" \
|
|
"$C_BLUE" "$C_RESET" "$C_BOLD" "$NAME" "$C_RESET" "$SCALE" "$CURRENT_TRANSFORM" "$NEW_TRANSFORM"
|
|
|
|
# Apply the new configuration immediately via IPC
|
|
if hyprctl keyword monitor "${NAME}, preferred, auto, ${SCALE}, transform, ${NEW_TRANSFORM}" > /dev/null; then
|
|
printf "%s[SUCCESS]%s Rotation applied successfully.\n" \
|
|
"$C_GREEN" "$C_RESET"
|
|
|
|
# Notify user visually if notify-send is available (optional UX improvement)
|
|
if command -v notify-send &> /dev/null; then
|
|
notify-send -a "System" "Display Rotated" "Monitor: $NAME\nTransform: $NEW_TRANSFORM" -h string:x-canonical-private-synchronous:display-rotate
|
|
fi
|
|
else
|
|
printf "%s[ERROR]%s Failed to apply Hyprland keyword.\n" \
|
|
"$C_RED" "$C_RESET" >&2
|
|
exit 1
|
|
fi
|
|
|
|
# Clean exit
|
|
trap - EXIT
|
|
exit 0
|