This commit is contained in:
2026-02-26 19:29:31 +01:00
parent 5391d45a08
commit 77eda5a2a4
3 changed files with 705 additions and 795 deletions
+437 -467
View File
File diff suppressed because it is too large Load Diff
+134 -164
View File
@@ -1709,179 +1709,149 @@ in
** Workspace_Wallpaper
workspace_wallpaper installs wpaperd and deploys your wallpaper files from the repo (./assets/conf/desktop/wallpaper/pictures/) into ~/conf/desktop/wallpaper/pictures. It also deploys the default wallpaper configuration from assets/conf/desktop/wallpaper/wallpaper.conf into ~/conf/desktop/wallpaper/wallpaper.conf, which is the file you can edit as a user override.
#+begin_src nix :tangle home/desktop/workspace_wallpaper.nix :noweb tangle :mkdirp yes
{ config, pkgs, lib, flakeRoot, ... }:
let
repoWallpaperDir = flakeRoot + "/assets/conf/desktop/wallpaper";
#!/usr/bin/env bash
set -euo pipefail
userRelRoot = "nixos_conf/wallpaperstuff";
userAbsRoot = "${config.home.homeDirectory}/${userRelRoot}";
# Global (not per-monitor) wallpaper per workspace 1..9 for Hyprland + wpaperd (wpaperctl v1.2.1)
#
# Expected images:
# ~/nixos_conf/wallpaperstuff/pictures/wallpaper{1..9}.jpg
#
# What it does:
# - Maintains a single symlink:
# ~/nixos_conf/wallpaperstuff/current/workspace.jpg
# - Writes/ensures wpaperd config:
# ~/.config/wpaperd/config.toml
# with:
# [default] mode = "stretch"
# [any] path = "<symlink>"
# - On workspace change, updates symlink and runs:
# wpaperctl reload-wallpaper
#
# Notes:
# - This is intentionally NOT per-monitor. If you have 2 monitors on different workspaces at once,
# the wallpaper will follow the *currently active/focused workspace*.
picturesDir = "${userAbsRoot}/pictures";
currentDir = "${userAbsRoot}/current";
log() { echo "[wpaperd-ws] $*" >&2; }
scriptRel = "hypr/scripts/wpaperd-workspace-1to9.sh";
wpaperdConfRel = "wpaperd/config.toml";
wpaperdConfAbs = "${config.xdg.configHome}/${wpaperdConfRel}";
in
{
home.packages = [
pkgs.wpaperd
pkgs.socat
pkgs.jq
pkgs.coreutils
];
PICTURES_DIR="${HOME}/nixos_conf/wallpaperstuff/pictures"
STATE_DIR="${HOME}/nixos_conf/wallpaperstuff/current"
SYMLINK="${STATE_DIR}/workspace.jpg"
# Copy wallpapers from repo → ~/nixos_conf/wallpaperstuff
home.file."${userRelRoot}" = {
source = repoWallpaperDir;
recursive = true;
};
XDG_CONFIG_HOME="${XDG_CONFIG_HOME:-$HOME/.config}"
WPAPERD_DIR="${XDG_CONFIG_HOME}/wpaperd"
CONF="${WPAPERD_DIR}/config.toml"
# Create a correct wpaperd config.
# - [default] sets stretch globally
# - [any] acts as fallback for outputs not explicitly listed (wpaperd-output(5))
# - Per-output sections point at CURRENT/<MON>.jpg (symlinks the script updates)
#
# NOTE: This is safe even if you add/remove monitors: [any] keeps things working.
xdg.configFile."${wpaperdConfRel}".text = ''
[default]
mode = "stretch"
: "${XDG_RUNTIME_DIR:?XDG_RUNTIME_DIR not set}"
: "${HYPRLAND_INSTANCE_SIGNATURE:?HYPRLAND_INSTANCE_SIGNATURE not set}"
SOCK="${XDG_RUNTIME_DIR}/hypr/${HYPRLAND_INSTANCE_SIGNATURE}/.socket2.sock"
[any]
path = "${picturesDir}/wallpaper1.jpg"
mkdir -p "$STATE_DIR" "$WPAPERD_DIR"
[eDP-1]
path = "${currentDir}/eDP-1.jpg"
# Wait for Hyprland socket so we don't exit too early
for i in $(seq 1 120); do
[[ -S "$SOCK" ]] && break
sleep 0.1
done
[[ -S "$SOCK" ]] || { log "Hyprland socket not found: $SOCK"; exit 1; }
[DP-1]
path = "${currentDir}/DP-1.jpg"
'';
# Listener: on focused monitor change, update CURRENT/<MON>.jpg -> pictures/wallpaperN.jpg
xdg.configFile."${scriptRel}" = {
executable = true;
text = ''
#!/usr/bin/env bash
set -euo pipefail
log() { echo "[wpaperd-ws] $*" >&2; }
PICTURES="${picturesDir}"
CURRENT="${currentDir}"
CONF="${wpaperdConfAbs}"
: "''${XDG_RUNTIME_DIR:?XDG_RUNTIME_DIR not set}"
: "''${HYPRLAND_INSTANCE_SIGNATURE:?HYPRLAND_INSTANCE_SIGNATURE not set}"
SOCK="''${XDG_RUNTIME_DIR}/hypr/''${HYPRLAND_INSTANCE_SIGNATURE}/.socket2.sock"
mkdir -p "$CURRENT" "$(dirname "$CONF")"
# Wait for Hyprland socket so the service doesn't exit too early
for i in $(seq 1 120); do
[[ -S "$SOCK" ]] && break
sleep 0.1
done
[[ -S "$SOCK" ]] || { log "Hyprland socket not found: $SOCK"; exit 1; }
ws_num() {
local w="''${1%%:*}"
[[ "$w" =~ ^[0-9]+$ ]] || return 1
printf '%s\n' "$w"
}
img_for_ws() {
local n="$1"
case "$n" in
1|2|3|4|5|6|7|8|9) printf '%s/wallpaper%s.jpg\n' "$PICTURES" "$n" ;;
*) return 1 ;;
esac
}
ensure_link_for() {
local mon="$1"
local ws="$2"
local n img
n="$(ws_num "$ws")" || return 0
img="$(img_for_ws "$n")" || return 0
if [[ ! -f "$img" ]]; then
log "Missing image for ws=$n: $img"
return 0
fi
ln -sf "$img" "$CURRENT/$mon.jpg"
log "Link: $CURRENT/$mon.jpg -> $img"
}
reload() {
# wpaperctl reload-wallpaper exists on v1.2.1
${pkgs.wpaperd}/bin/wpaperctl reload-wallpaper >/dev/null 2>&1 || true
}
apply_initial() {
${pkgs.hyprland}/bin/hyprctl -j monitors \
| ${pkgs.jq}/bin/jq -r '.[] | "\(.name)\t\(.activeWorkspace.name)"' \
| while IFS=$'\t' read -r mon ws; do
[[ -n "$mon" && -n "$ws" ]] || continue
ensure_link_for "$mon" "$ws" || true
done
reload
}
handle_line() {
case "$1" in
focusedmon\>\>*)
# focusedmon>>MONNAME,WORKSPACENAME
local payload="''${1#focusedmon>>}"
local mon="''${payload%%,*}"
local ws="''${payload#*,}"
[[ -n "$mon" && -n "$ws" ]] || return 0
ensure_link_for "$mon" "$ws" || true
reload
;;
esac
}
apply_initial
log "Listening on $SOCK"
exec ${pkgs.socat}/bin/socat -U - "UNIX-CONNECT:$SOCK" \
| while IFS= read -r line; do
handle_line "$line" || true
done
'';
};
# Start wpaperd (reads XDG_CONFIG_HOME/wpaperd/config.toml by default)
systemd.user.services.wpaperd = {
Unit = {
Description = "wpaperd wallpaper daemon";
After = [ "graphical-session.target" ];
PartOf = [ "graphical-session.target" ];
};
Service = {
ExecStart = "${pkgs.wpaperd}/bin/wpaperd";
Restart = "on-failure";
RestartSec = 1;
};
Install.WantedBy = [ "graphical-session.target" ];
};
# Listener service
systemd.user.services.wpaperd-workspace-wallpapers = {
Unit = {
Description = "Set wallpaper per monitor based on workspace number (1..9)";
After = [ "graphical-session.target" "wpaperd.service" ];
PartOf = [ "graphical-session.target" ];
};
Service = {
ExecStart = "${pkgs.bash}/bin/bash ${config.xdg.configHome}/${scriptRel}";
Restart = "on-failure";
RestartSec = 1;
};
Install.WantedBy = [ "graphical-session.target" ];
};
ws_num() {
# workspace name might be "8" or "8:foo"
local w="${1%%:*}"
[[ "$w" =~ ^[0-9]+$ ]] || return 1
printf '%s\n' "$w"
}
img_for_ws() {
local n="$1"
case "$n" in
1|2|3|4|5|6|7|8|9) printf '%s/wallpaper%s.jpg\n' "$PICTURES_DIR" "$n" ;;
*) return 1 ;;
esac
}
ensure_config() {
# Keep it simple and stable: always ensure config points [any] to the symlink.
# wpaperd reads this path by default.
cat >"$CONF" <<EOF
[default]
mode = "stretch"
[any]
path = "$SYMLINK"
EOF
}
reload_wallpaper() {
# wpaperctl v1.2.1 supports reload-wallpaper
wpaperctl reload-wallpaper >/dev/null 2>&1 || true
}
apply_workspace() {
local ws_raw="$1"
local n img
n="$(ws_num "$ws_raw")" || { log "Ignoring non-numeric workspace: $ws_raw"; return 0; }
img="$(img_for_ws "$n")" || return 0
if [[ ! -f "$img" ]]; then
log "Missing image for ws=$n: $img"
return 0
fi
# If already set, skip work
if [[ -L "$SYMLINK" ]] && [[ "$(readlink -f "$SYMLINK")" == "$(readlink -f "$img")" ]]; then
return 0
fi
ln -sf "$img" "$SYMLINK"
log "Workspace $n -> $img"
ensure_config
reload_wallpaper
}
apply_initial() {
# Use focused monitor's active workspace as "current workspace" for the global wallpaper.
# This keeps behavior consistent with "per workspace (focused)".
local ws
ws="$(hyprctl -j monitors | jq -r '.[] | select(.focused==true) | .activeWorkspace.name' | head -n1)"
if [[ -n "$ws" && "$ws" != "null" ]]; then
apply_workspace "$ws"
else
log "Could not determine initial focused workspace"
fi
}
handle_line() {
case "$1" in
# workspace>>NAME
workspace\>\>*)
apply_workspace "${1#workspace>>}"
;;
# workspacev2>>ID,NAME
workspacev2\>\>*)
local payload="${1#workspacev2>>}"
apply_workspace "${payload#*,}"
;;
# focusedmon>>MON,WORKSPACE (also catches some focus-driven changes)
focusedmon\>\>*)
local payload="${1#focusedmon>>}"
local ws="${payload#*,}"
apply_workspace "$ws"
;;
esac
}
ensure_config
apply_initial
log "Listening on $SOCK"
exec socat -U - "UNIX-CONNECT:$SOCK" \
| while IFS= read -r line; do
handle_line "$line" || true
done
#+end_src
** Animated Wallpaper
+134 -164
View File
@@ -1,173 +1,143 @@
{ config, pkgs, lib, flakeRoot, ... }:
let
repoWallpaperDir = flakeRoot + "/assets/conf/desktop/wallpaper";
#!/usr/bin/env bash
set -euo pipefail
userRelRoot = "nixos_conf/wallpaperstuff";
userAbsRoot = "${config.home.homeDirectory}/${userRelRoot}";
# Global (not per-monitor) wallpaper per workspace 1..9 for Hyprland + wpaperd (wpaperctl v1.2.1)
#
# Expected images:
# ~/nixos_conf/wallpaperstuff/pictures/wallpaper{1..9}.jpg
#
# What it does:
# - Maintains a single symlink:
# ~/nixos_conf/wallpaperstuff/current/workspace.jpg
# - Writes/ensures wpaperd config:
# ~/.config/wpaperd/config.toml
# with:
# [default] mode = "stretch"
# [any] path = "<symlink>"
# - On workspace change, updates symlink and runs:
# wpaperctl reload-wallpaper
#
# Notes:
# - This is intentionally NOT per-monitor. If you have 2 monitors on different workspaces at once,
# the wallpaper will follow the *currently active/focused workspace*.
picturesDir = "${userAbsRoot}/pictures";
currentDir = "${userAbsRoot}/current";
log() { echo "[wpaperd-ws] $*" >&2; }
scriptRel = "hypr/scripts/wpaperd-workspace-1to9.sh";
wpaperdConfRel = "wpaperd/config.toml";
wpaperdConfAbs = "${config.xdg.configHome}/${wpaperdConfRel}";
in
{
home.packages = [
pkgs.wpaperd
pkgs.socat
pkgs.jq
pkgs.coreutils
];
PICTURES_DIR="${HOME}/nixos_conf/wallpaperstuff/pictures"
STATE_DIR="${HOME}/nixos_conf/wallpaperstuff/current"
SYMLINK="${STATE_DIR}/workspace.jpg"
# Copy wallpapers from repo → ~/nixos_conf/wallpaperstuff
home.file."${userRelRoot}" = {
source = repoWallpaperDir;
recursive = true;
};
XDG_CONFIG_HOME="${XDG_CONFIG_HOME:-$HOME/.config}"
WPAPERD_DIR="${XDG_CONFIG_HOME}/wpaperd"
CONF="${WPAPERD_DIR}/config.toml"
# Create a correct wpaperd config.
# - [default] sets stretch globally
# - [any] acts as fallback for outputs not explicitly listed (wpaperd-output(5))
# - Per-output sections point at CURRENT/<MON>.jpg (symlinks the script updates)
#
# NOTE: This is safe even if you add/remove monitors: [any] keeps things working.
xdg.configFile."${wpaperdConfRel}".text = ''
[default]
mode = "stretch"
: "${XDG_RUNTIME_DIR:?XDG_RUNTIME_DIR not set}"
: "${HYPRLAND_INSTANCE_SIGNATURE:?HYPRLAND_INSTANCE_SIGNATURE not set}"
SOCK="${XDG_RUNTIME_DIR}/hypr/${HYPRLAND_INSTANCE_SIGNATURE}/.socket2.sock"
[any]
path = "${picturesDir}/wallpaper1.jpg"
mkdir -p "$STATE_DIR" "$WPAPERD_DIR"
[eDP-1]
path = "${currentDir}/eDP-1.jpg"
# Wait for Hyprland socket so we don't exit too early
for i in $(seq 1 120); do
[[ -S "$SOCK" ]] && break
sleep 0.1
done
[[ -S "$SOCK" ]] || { log "Hyprland socket not found: $SOCK"; exit 1; }
[DP-1]
path = "${currentDir}/DP-1.jpg"
'';
# Listener: on focused monitor change, update CURRENT/<MON>.jpg -> pictures/wallpaperN.jpg
xdg.configFile."${scriptRel}" = {
executable = true;
text = ''
#!/usr/bin/env bash
set -euo pipefail
log() { echo "[wpaperd-ws] $*" >&2; }
PICTURES="${picturesDir}"
CURRENT="${currentDir}"
CONF="${wpaperdConfAbs}"
: "''${XDG_RUNTIME_DIR:?XDG_RUNTIME_DIR not set}"
: "''${HYPRLAND_INSTANCE_SIGNATURE:?HYPRLAND_INSTANCE_SIGNATURE not set}"
SOCK="''${XDG_RUNTIME_DIR}/hypr/''${HYPRLAND_INSTANCE_SIGNATURE}/.socket2.sock"
mkdir -p "$CURRENT" "$(dirname "$CONF")"
# Wait for Hyprland socket so the service doesn't exit too early
for i in $(seq 1 120); do
[[ -S "$SOCK" ]] && break
sleep 0.1
done
[[ -S "$SOCK" ]] || { log "Hyprland socket not found: $SOCK"; exit 1; }
ws_num() {
local w="''${1%%:*}"
[[ "$w" =~ ^[0-9]+$ ]] || return 1
printf '%s\n' "$w"
}
img_for_ws() {
local n="$1"
case "$n" in
1|2|3|4|5|6|7|8|9) printf '%s/wallpaper%s.jpg\n' "$PICTURES" "$n" ;;
*) return 1 ;;
esac
}
ensure_link_for() {
local mon="$1"
local ws="$2"
local n img
n="$(ws_num "$ws")" || return 0
img="$(img_for_ws "$n")" || return 0
if [[ ! -f "$img" ]]; then
log "Missing image for ws=$n: $img"
return 0
fi
ln -sf "$img" "$CURRENT/$mon.jpg"
log "Link: $CURRENT/$mon.jpg -> $img"
}
reload() {
# wpaperctl reload-wallpaper exists on v1.2.1
${pkgs.wpaperd}/bin/wpaperctl reload-wallpaper >/dev/null 2>&1 || true
}
apply_initial() {
${pkgs.hyprland}/bin/hyprctl -j monitors \
| ${pkgs.jq}/bin/jq -r '.[] | "\(.name)\t\(.activeWorkspace.name)"' \
| while IFS=$'\t' read -r mon ws; do
[[ -n "$mon" && -n "$ws" ]] || continue
ensure_link_for "$mon" "$ws" || true
done
reload
}
handle_line() {
case "$1" in
focusedmon\>\>*)
# focusedmon>>MONNAME,WORKSPACENAME
local payload="''${1#focusedmon>>}"
local mon="''${payload%%,*}"
local ws="''${payload#*,}"
[[ -n "$mon" && -n "$ws" ]] || return 0
ensure_link_for "$mon" "$ws" || true
reload
;;
esac
}
apply_initial
log "Listening on $SOCK"
exec ${pkgs.socat}/bin/socat -U - "UNIX-CONNECT:$SOCK" \
| while IFS= read -r line; do
handle_line "$line" || true
done
'';
};
# Start wpaperd (reads XDG_CONFIG_HOME/wpaperd/config.toml by default)
systemd.user.services.wpaperd = {
Unit = {
Description = "wpaperd wallpaper daemon";
After = [ "graphical-session.target" ];
PartOf = [ "graphical-session.target" ];
};
Service = {
ExecStart = "${pkgs.wpaperd}/bin/wpaperd";
Restart = "on-failure";
RestartSec = 1;
};
Install.WantedBy = [ "graphical-session.target" ];
};
# Listener service
systemd.user.services.wpaperd-workspace-wallpapers = {
Unit = {
Description = "Set wallpaper per monitor based on workspace number (1..9)";
After = [ "graphical-session.target" "wpaperd.service" ];
PartOf = [ "graphical-session.target" ];
};
Service = {
ExecStart = "${pkgs.bash}/bin/bash ${config.xdg.configHome}/${scriptRel}";
Restart = "on-failure";
RestartSec = 1;
};
Install.WantedBy = [ "graphical-session.target" ];
};
ws_num() {
# workspace name might be "8" or "8:foo"
local w="${1%%:*}"
[[ "$w" =~ ^[0-9]+$ ]] || return 1
printf '%s\n' "$w"
}
img_for_ws() {
local n="$1"
case "$n" in
1|2|3|4|5|6|7|8|9) printf '%s/wallpaper%s.jpg\n' "$PICTURES_DIR" "$n" ;;
*) return 1 ;;
esac
}
ensure_config() {
# Keep it simple and stable: always ensure config points [any] to the symlink.
# wpaperd reads this path by default.
cat >"$CONF" <<EOF
[default]
mode = "stretch"
[any]
path = "$SYMLINK"
EOF
}
reload_wallpaper() {
# wpaperctl v1.2.1 supports reload-wallpaper
wpaperctl reload-wallpaper >/dev/null 2>&1 || true
}
apply_workspace() {
local ws_raw="$1"
local n img
n="$(ws_num "$ws_raw")" || { log "Ignoring non-numeric workspace: $ws_raw"; return 0; }
img="$(img_for_ws "$n")" || return 0
if [[ ! -f "$img" ]]; then
log "Missing image for ws=$n: $img"
return 0
fi
# If already set, skip work
if [[ -L "$SYMLINK" ]] && [[ "$(readlink -f "$SYMLINK")" == "$(readlink -f "$img")" ]]; then
return 0
fi
ln -sf "$img" "$SYMLINK"
log "Workspace $n -> $img"
ensure_config
reload_wallpaper
}
apply_initial() {
# Use focused monitor's active workspace as "current workspace" for the global wallpaper.
# This keeps behavior consistent with "per workspace (focused)".
local ws
ws="$(hyprctl -j monitors | jq -r '.[] | select(.focused==true) | .activeWorkspace.name' | head -n1)"
if [[ -n "$ws" && "$ws" != "null" ]]; then
apply_workspace "$ws"
else
log "Could not determine initial focused workspace"
fi
}
handle_line() {
case "$1" in
# workspace>>NAME
workspace\>\>*)
apply_workspace "${1#workspace>>}"
;;
# workspacev2>>ID,NAME
workspacev2\>\>*)
local payload="${1#workspacev2>>}"
apply_workspace "${payload#*,}"
;;
# focusedmon>>MON,WORKSPACE (also catches some focus-driven changes)
focusedmon\>\>*)
local payload="${1#focusedmon>>}"
local ws="${payload#*,}"
apply_workspace "$ws"
;;
esac
}
ensure_config
apply_initial
log "Listening on $SOCK"
exec socat -U - "UNIX-CONNECT:$SOCK" \
| while IFS= read -r line; do
handle_line "$line" || true
done