Major refactor to flake, abstract non-trivial packages into their own

files (such as firefox/spotify-qt+librespot), create utils for reusable
function library
This commit is contained in:
Martin Larsson 2025-05-11 18:19:19 +02:00
parent d5735692d2
commit c4d7f5d082
11 changed files with 379 additions and 331 deletions

View file

@ -0,0 +1,96 @@
{
pkgs,
nur,
...
}:
{
programs = {
firefox = {
enable = true;
package = pkgs.firefox-wayland;
policies = {
"DisableFirefoxStudies" = true;
"DisableTelemetry" = true;
};
profiles = {
default = {
isDefault = true;
name = "DefaultProfile";
userChrome = ''
@import "theme/nordic-theme.css";
@import "theme/hide-single-tab.css";
@import "theme/matching-autocomplete-width.css";
@import "theme/system-icons.css";
@import "theme/symbolic-tab-icons.css";
@import "customChrome.css";
'';
extensions = {
packages = with nur.repos.rycee.firefox-addons; [
ublock-origin
bitwarden
vimium
privacy-badger
clearurls
darkreader
];
};
settings = {
"browser.startup.homepage" = "https://search.nixos.org";
"browser.shell.checkDefaultBrowser" = false;
"privacy.resistFingerprinting" = false;
"dom.security.https_only_mode" = true;
"browser.tabs.warnOnClose" = false;
"extensions.pocket.enabled" = false;
"browser.search.defaultenginename" = "ddg";
"gfx.webrender.all" = true;
"toolkit.legacyUserProfileCustomizations.stylesheets" = true;
"browser.tabs.allow_transparent_browser" = true;
};
search = {
force = true;
default = "ddg";
order = [
"ddg"
"google"
];
engines = {
"ddg".metaData = {
alias = "@d";
hidden = false;
};
"google".metaData = {
alias = "@g";
hidden = false;
};
"Nix Packages" = {
urls = [
{
template = "https://search.nixos.org/packages";
params = [
{
name = "type";
value = "packages";
}
{
name = "query";
value = "{searchTerms}";
}
];
}
];
icon = "${pkgs.nixos-icons}/share/icons/hicolor/scalable/apps/nix-snowflake.svg";
definedAliases = [ "@np" ];
};
};
};
};
};
};
};
}

View file

@ -0,0 +1,80 @@
{
pkgs,
config,
lib,
...
}:
let
# User-specific librespot settings
librespot_device_name = "asahi-nixos-librespot";
librespot_username = "martymeister98";
librespot_bitrate = "320";
# Construct librespot arguments
librespot_args = ''
--name ${librespot_device_name}
--username ${librespot_username}
--bitrate ${librespot_bitrate}
--disable-audio-cache
--initial-volume 50
'';
spotify_qt_initial_config = {
General = {
check_for_updates = false;
close_to_tray = false;
fallback_icons = false;
native_window = false;
notify_track_change = false;
relative_added = true;
show_changelog = false;
style = "Fusion";
style_palette = 2;
track_list_resize_mode = 0;
track_numbers = 1;
tray_album_art = false;
tray_icon = false;
tray_light_icon = false;
};
Qt = {
album_shape = 1;
album_size = 1;
library_layout = 1;
mirror_title_bar = false;
system_title_bar = true;
toolbar_position = 1;
};
Spotify = {
path = "${pkgs.librespot}/bin/librespot";
client_arguments = librespot_args;
always_start = true;
bitrate = 320;
disable_discovery = true;
start_client = true;
};
};
spotify_qt_initial_json = builtins.toJSON spotify_qt_initial_config;
in
{
home = {
packages = with pkgs; [
librespot
spotify-qt
];
activation.conditionally_create_spotify-qt_config = lib.hm.dag.entryAfter [ "linkGeneration" ] ''
configFile="${config.home.homeDirectory}/.config/kraxarn/spotify-qt.json"
configDir="$(dirname "$configFile")"
if [ ! -f "$configFile" ]; then
echo "Initial spotify-qt.json not found. Creating with declarative defaults."
mkdir -p "$configDir"
printf '%s' '${spotify_qt_initial_json}' > "$configFile"
echo "Initial spotify-qt.json created at $configFile."
else
echo "spotify-qt.json already exists at $configFile. Skipping initial creation."
fi
'';
};
}

119
nix/home/common/theming.nix Normal file
View file

@ -0,0 +1,119 @@
{
pkgs,
...
}:
let
cursor_name = "Banana-Blue";
cursor_base_size = 32;
bananaCursorBlueTarball = pkgs.fetchurl {
url = "https://github.com/ful1e5/banana-cursor/releases/download/v2.0.0/Banana-Blue.tar.xz";
sha256 = "sha256-mpTrvgYiMfamMebtytY0bLouSbaP3qEqP8pgCFl+xPQ=";
};
banana_cursor_blue = pkgs.stdenv.mkDerivation {
pname = "banana-cursor-blue";
version = "2.0.0";
src = bananaCursorBlueTarball;
nativeBuildInputs = [ pkgs.xz ];
dontBuild = true;
installPhase = ''
runHook preInstall
# Define shell variable using Nix interpolation for $out and the outer Nix var cursorName
local themeInstallDir="$out/share/icons/${cursor_name}"
mkdir -p "$themeInstallDir"
# Use the outer Nix var cursorName and the builder's $version shell variable
echo "Unpacking and installing pre-built theme ${cursor_name} (v$version) to $themeInstallDir"
if tar -xJf $src --strip-components=1 -C "$themeInstallDir"; then
echo "Unpacked successfully with --strip-components=1."
else
echo "-----------------------------------------------------"
echo "WARNING: Unpacking with --strip-components=1 failed (archive might not have a single top-level dir)."
echo "Listing archive contents:"
tar -tf $src || echo "Failed to list archive contents."
echo "Attempting unpack without --strip-components=1..."
rm -rf "$themeInstallDir"; mkdir -p "$themeInstallDir" # Clean and recreate before retry
if tar -xJf $src -C "$themeInstallDir"; then
echo "Unpacked successfully without --strip-components=1."
else
echo "ERROR: Failed to unpack archive even without --strip-components=1."
exit 1
fi
echo "-----------------------------------------------------"
fi
runHook postInstall
'';
meta = with pkgs.lib; {
description = "Banana Cursor Theme (Pre-built Blue variant v2.0.0)";
homepage = "https://github.com/ful1e5/banana-cursor";
license = licenses.gpl3Only;
maintainers = with maintainers; [ eelco ];
platforms = platforms.linux;
};
};
cursor_package = banana_cursor_blue;
effective_cursor_size_str = toString cursor_base_size;
in
{
wayland.windowManager.sway = {
config = {
seat = {
"*" = {
# Sway uses this for its own cursor and for XWayland applications.
# It takes the theme name and base size. Sway should scale this based on output settings.
xcursor_theme = "${cursor_name} ${effective_cursor_size_str}";
};
};
};
};
# Home Manager Pointer Configuration
# This makes the theme available and sets X11/GTK defaults via Home Manager mechanisms.
home = {
pointerCursor = {
name = cursor_name;
size = cursor_base_size;
package = cursor_package;
x11.enable = true;
gtk.enable = true;
x11.defaultCursor = "left_ptr";
};
packages = with pkgs; [
(cursor_package)
];
};
qt = {
enable = true;
platformTheme.name = "gtk";
style.name = "Nordic";
};
gtk = {
enable = true;
cursorTheme = {
name = cursor_name;
size = cursor_base_size; # GTK applications will use this base size.
package = cursor_package;
};
theme = {
package = pkgs.nordic;
name = "Nordic";
};
};
xresources.properties = {
"Xcursor.theme" = cursor_name;
};
home.sessionVariables = {
XCURSOR_THEME = cursor_name;
};
}

View file

@ -6,6 +6,7 @@
... ...
}: }:
let let
utils = import ../utils.nix;
dotfiles = [ dotfiles = [
[ [
".zshrc" ".zshrc"
@ -54,16 +55,6 @@ let
cp ${pkgs.vscode-extensions.vadimcn.vscode-lldb}/share/vscode/extensions/vadimcn.vscode-lldb/adapter/codelldb $out/bin/codelldb cp ${pkgs.vscode-extensions.vadimcn.vscode-lldb}/share/vscode/extensions/vadimcn.vscode-lldb/adapter/codelldb $out/bin/codelldb
chmod +x $out/bin/codelldb chmod +x $out/bin/codelldb
''; '';
symlinkFiles = builtins.listToAttrs (
map (file: {
name = builtins.elemAt file 0;
value = {
source = config.lib.file.mkOutOfStoreSymlink "${config.home.homeDirectory}/dev/git/.dotfiles/${builtins.elemAt file 1}";
};
}) dotfiles
);
in in
{ {
programs = { programs = {
@ -123,7 +114,7 @@ in
bitwarden-cli bitwarden-cli
]; ];
file = symlinkFiles; file = utils.mk_symlinks { inherit config dotfiles; };
activation.batCache = lib.hm.dag.entryAfter [ "writeBoundary" ] '' activation.batCache = lib.hm.dag.entryAfter [ "writeBoundary" ] ''
run ${pkgs.bat}/bin/bat cache --build run ${pkgs.bat}/bin/bat cache --build
''; '';

View file

@ -1,186 +1,35 @@
{ {
pkgs, pkgs,
config, config,
lib,
nur,
... ...
}: }:
let let
utils = import ../utils.nix;
dotfiles = [
[
".config/sway"
"sway"
]
[
".config/wofi"
"wofi"
]
];
pythonEnv = pkgs.python3.withPackages ( pythonEnv = pkgs.python3.withPackages (
ps: with ps; [ ps: with ps; [
i3ipc i3ipc
] ]
); );
cursorName = "Banana-Blue";
cursorBaseSize = 32;
bananaCursorBlueTarball = pkgs.fetchurl {
url = "https://github.com/ful1e5/banana-cursor/releases/download/v2.0.0/Banana-Blue.tar.xz";
sha256 = "sha256-mpTrvgYiMfamMebtytY0bLouSbaP3qEqP8pgCFl+xPQ=";
};
bananaCursorBlue = pkgs.stdenv.mkDerivation {
pname = "banana-cursor-blue";
version = "2.0.0";
src = bananaCursorBlueTarball;
nativeBuildInputs = [ pkgs.xz ];
dontBuild = true;
installPhase = ''
runHook preInstall
# Define shell variable using Nix interpolation for $out and the outer Nix var cursorName
local themeInstallDir="$out/share/icons/${cursorName}"
mkdir -p "$themeInstallDir"
# Use the outer Nix var cursorName and the builder's $version shell variable
echo "Unpacking and installing pre-built theme ${cursorName} (v$version) to $themeInstallDir"
if tar -xJf $src --strip-components=1 -C "$themeInstallDir"; then
echo "Unpacked successfully with --strip-components=1."
else
echo "-----------------------------------------------------"
echo "WARNING: Unpacking with --strip-components=1 failed (archive might not have a single top-level dir)."
echo "Listing archive contents:"
tar -tf $src || echo "Failed to list archive contents."
echo "Attempting unpack without --strip-components=1..."
rm -rf "$themeInstallDir"; mkdir -p "$themeInstallDir" # Clean and recreate before retry
if tar -xJf $src -C "$themeInstallDir"; then
echo "Unpacked successfully without --strip-components=1."
else
echo "ERROR: Failed to unpack archive even without --strip-components=1."
exit 1
fi
echo "-----------------------------------------------------"
fi
runHook postInstall
'';
meta = with pkgs.lib; {
description = "Banana Cursor Theme (Pre-built Blue variant v2.0.0)";
homepage = "https://github.com/ful1e5/banana-cursor";
license = licenses.gpl3Only;
maintainers = with maintainers; [ eelco ]; # Original nixpkgs maintainer
platforms = platforms.linux;
};
};
cursorPackage = bananaCursorBlue;
effectiveCursorSizeStr = toString cursorBaseSize;
firefox-nordic-theme-src = pkgs.fetchFromGitHub {
owner = "EliverLara";
repo = "firefox-nordic-theme";
rev = "21b79cca716af87b8a2b9e420c0e1d3d08b67414";
sha256 = "sha256-2xP9tHCmOM35fxFMbABUhHHnefv2sSCwhnYpjbHM/V0=";
};
# User-specific librespot settings
librespot_device_name = "asahi-nixos-librespot";
librespot_username = "martymeister98";
librespot_bitrate = "320";
# Construct librespot arguments
librespot_args = ''
--name ${librespot_device_name}
--username ${librespot_username}
--bitrate ${librespot_bitrate}
--disable-audio-cache
--initial-volume 50
'';
spotify_qt_initial_config = {
General = {
check_for_updates = false;
close_to_tray = false;
fallback_icons = false;
native_window = false;
notify_track_change = false;
relative_added = true;
show_changelog = false;
style = "Fusion";
style_palette = 2;
track_list_resize_mode = 0;
track_numbers = 1;
tray_album_art = false;
tray_icon = false;
tray_light_icon = false;
};
Qt = {
album_shape = 1;
album_size = 1;
library_layout = 1;
mirror_title_bar = false;
system_title_bar = true;
toolbar_position = 1;
};
Spotify = {
path = "${pkgs.librespot}/bin/librespot";
client_arguments = librespot_args;
always_start = true;
bitrate = 320;
disable_discovery = true;
start_client = true;
};
};
spotify_qt_initial_json = builtins.toJSON spotify_qt_initial_config;
in in
{ {
wayland.windowManager.sway = { imports = [
config = { ./common/firefox.nix
seat = { ];
"*" = {
# Sway uses this for its own cursor and for XWayland applications.
# It takes the theme name and base size. Sway should scale this based on output settings.
xcursor_theme = "${cursorName} ${effectiveCursorSizeStr}";
};
};
};
};
# Home Manager Pointer Configuration
# This makes the theme available and sets X11/GTK defaults via Home Manager mechanisms.
home.pointerCursor = {
name = cursorName;
size = cursorBaseSize;
package = cursorPackage;
x11.enable = true;
gtk.enable = true;
x11.defaultCursor = "left_ptr";
};
qt = {
enable = true;
platformTheme.name = "gtk";
style.name = "Nordic";
};
gtk = {
enable = true;
cursorTheme = {
name = cursorName;
size = cursorBaseSize; # GTK applications will use this base size.
package = cursorPackage;
};
theme = {
package = pkgs.nordic;
name = "Nordic";
};
};
xresources.properties = {
"Xcursor.theme" = cursorName;
};
home = { home = {
sessionVariables = {
XCURSOR_THEME = cursorName;
};
packages = with pkgs; [ packages = with pkgs; [
(cursorPackage)
pythonEnv pythonEnv
wl-clipboard-rs wl-clipboard-rs
clang clang
@ -193,124 +42,10 @@ in
grim grim
slurp slurp
pavucontrol pavucontrol
legcord
spotify-qt
librespot
blueman blueman
playerctl playerctl
]; ];
file = { file = utils.mk_symlinks { inherit config dotfiles; };
".config/sway".source = config.lib.file.mkOutOfStoreSymlink ../../sway;
".config/wofi".source = config.lib.file.mkOutOfStoreSymlink ../../wofi;
"${config.home.homeDirectory}/.mozilla/firefox/hm-profile-default/chrome/firefox-nordic-theme" = {
source = firefox-nordic-theme-src;
recursive = true;
};
};
activation.conditionally_create_spotify-qt_config = lib.hm.dag.entryAfter [ "linkGeneration" ] ''
configFile="${config.home.homeDirectory}/.config/kraxarn/spotify-qt.json"
configDir="$(dirname "$configFile")"
if [ ! -f "$configFile" ]; then
echo "Initial spotify-qt.json not found. Creating with declarative defaults."
mkdir -p "$configDir"
printf '%s' '${spotify_qt_initial_json}' > "$configFile"
echo "Initial spotify-qt.json created at $configFile."
else
echo "spotify-qt.json already exists at $configFile. Skipping initial creation."
fi
'';
};
programs = {
firefox = {
enable = true;
package = pkgs.firefox-wayland;
policies = {
"DisableFirefoxStudies" = true;
"DisableTelemetry" = true;
};
profiles = {
default = {
isDefault = true;
name = "DefaultProfile";
userChrome = ''
@import "theme/nordic-theme.css";
@import "theme/hide-single-tab.css";
@import "theme/matching-autocomplete-width.css";
@import "theme/system-icons.css";
@import "theme/symbolic-tab-icons.css";
@import "customChrome.css";
'';
extensions = {
packages = with nur.repos.rycee.firefox-addons; [
ublock-origin
bitwarden
vimium
privacy-badger
clearurls
darkreader
];
};
settings = {
"browser.startup.homepage" = "https://search.nixos.org";
"browser.shell.checkDefaultBrowser" = false;
"privacy.resistFingerprinting" = false;
"dom.security.https_only_mode" = true;
"browser.tabs.warnOnClose" = false;
"extensions.pocket.enabled" = false;
"browser.search.defaultenginename" = "ddg";
"gfx.webrender.all" = true;
"toolkit.legacyUserProfileCustomizations.stylesheets" = true;
"browser.tabs.allow_transparent_browser" = true;
};
search = {
force = true;
default = "ddg";
order = [
"ddg"
"google"
];
engines = {
"ddg".metaData = {
alias = "@d";
hidden = false;
};
"google".metaData = {
alias = "@g";
hidden = false;
};
"Nix Packages" = {
urls = [
{
template = "https://search.nixos.org/packages";
params = [
{
name = "type";
value = "packages";
}
{
name = "query";
value = "{searchTerms}";
}
];
}
];
icon = "${pkgs.nixos-icons}/share/icons/hicolor/scalable/apps/nix-snowflake.svg";
definedAliases = [ "@np" ];
};
};
};
};
};
};
}; };
} }

View file

@ -3,11 +3,13 @@
... ...
}: }:
{ {
imports = [
./common/spotify_arm.nix
];
home = { home = {
packages = with pkgs; [ packages = with pkgs; [
legcord legcord
librespot
spotify-qt
]; ];
}; };
} }

View file

@ -0,0 +1,40 @@
{ config, ... }:
{
boot = {
kernelModules = [
"nvidia"
"nvidia_modeset"
"nvidia_uvm"
"nvidia_drm"
];
blacklistedKernelModules = [ "nouveau" ];
kernelParams = [
"nvidia-drm.modeset=1"
"nvidia-drm.fbdev=1"
];
};
hardware = {
nvidia = {
modesetting.enable = true;
package = config.boot.kernelPackages.nvidiaPackages.stable;
open = true;
powerManagement = {
enable = true;
finegrained = false;
};
nvidiaPersistenced = true;
};
};
environment.variables = {
GBM_BACKEND = "nvidia-drm";
__GLX_VENDOR_LIBRARY_NAME = "nvidia";
WLR_NO_HARDWARE_CURSORS = "1";
WLR_RENDERER = "vulkan";
};
services = {
xserver.videoDrivers = [ "nvidia" ];
};
}

View file

@ -1,5 +1,9 @@
{ pkgs, ... }: { pkgs, ... }:
{ {
imports = [
./hardware-configuration.nix
];
programs = { programs = {
zsh.enable = true; zsh.enable = true;
}; };

View file

@ -1,7 +1,6 @@
{ apple-silicon-support, pkgs, ... }: { apple-silicon-support, pkgs, ... }:
{ {
imports = [ imports = [
./hardware-configuration.nix
apple-silicon-support.nixosModules.apple-silicon-support apple-silicon-support.nixosModules.apple-silicon-support
]; ];

View file

@ -1,7 +1,8 @@
{ pkgs, config, ... }: { pkgs, ... }:
{ {
imports = [ imports = [
./hardware-configuration.nix ./hardware-configuration.nix
./common/nvidia.nix
]; ];
boot = { boot = {
@ -9,30 +10,6 @@
systemd-boot.enable = true; systemd-boot.enable = true;
efi.canTouchEfiVariables = true; efi.canTouchEfiVariables = true;
}; };
kernelModules = [
"nvidia"
"nvidia_modeset"
"nvidia_uvm"
"nvidia_drm"
];
blacklistedKernelModules = [ "nouveau" ];
kernelParams = [
"nvidia-drm.modeset=1"
"nvidia-drm.fbdev=1"
];
};
hardware = {
nvidia = {
modesetting.enable = true;
package = config.boot.kernelPackages.nvidiaPackages.stable;
open = true;
powerManagement = {
enable = true;
finegrained = false;
};
nvidiaPersistenced = true;
};
}; };
networking.hostName = "walnut-nixos"; networking.hostName = "walnut-nixos";
@ -47,15 +24,4 @@
]; ];
}; };
}; };
environment.variables = {
GBM_BACKEND = "nvidia-drm";
__GLX_VENDOR_LIBRARY_NAME = "nvidia";
WLR_NO_HARDWARE_CURSORS = "1";
WLR_RENDERER = "vulkan";
};
services = {
xserver.videoDrivers = [ "nvidia" ];
};
} }

16
nix/utils.nix Normal file
View file

@ -0,0 +1,16 @@
{
mk_symlinks =
{
config,
basePath ? "${config.home.homeDirectory}/dev/git/.dotfiles",
dotfiles ? [ ],
}:
builtins.listToAttrs (
map (file: {
name = builtins.elemAt file 0;
value = {
source = config.lib.file.mkOutOfStoreSymlink "${basePath}/${builtins.elemAt file 1}";
};
}) dotfiles
);
}