diff --git a/modules/fonts/default.nix b/modules/fonts/default.nix index ba775f3a..f218cbeb 100644 --- a/modules/fonts/default.nix +++ b/modules/fonts/default.nix @@ -1,57 +1,72 @@ { config, lib, pkgs, ... }: with lib; -with builtins; let cfg = config.fonts; - readDirsRec = path: let - home = readDir path; - list = mapAttrsToList (name: type: - let newPath = "${path}/${name}"; - in if (type == "directory" || type == "symlink") && !(isFont name) then readDirsRec newPath else [ newPath ]) home; - in flatten list; - isFont = name: let - fontExt = [ ".ttf" ".otf" ]; - hasExt = exts: name: foldl' (acc: ext: (hasSuffix ext name) || acc) false exts; - in hasExt fontExt name; - fontFiles = dir: filter isFont (readDirsRec dir); - libraryLink = font: "ln -fn '/run/current-system/sw/share/fonts/${baseNameOf font}' '/Library/Fonts/${baseNameOf font}'"; - outLink = font: "ln -sfn -t $out/share/fonts/ '${font}'"; - fontLinks = link: dir: concatMapStringsSep "\n" link (fontFiles dir); - systemFontsDir = pkgs.runCommand "systemFontsDir" {} '' - mkdir -p "$out/share/fonts" - echo ${toString config.fonts.fonts} - ${concatMapStringsSep "\n" (fontLinks outLink) config.fonts.fonts} - ''; -in { +in + +{ options = { - fonts = { - enableFontDir = mkOption { - default = false; - description = '' - Whether to enable font directory management and link all fonts in /run/current-system/sw/share/fonts. - Important: removes all manually-added fonts. - ''; - }; - fonts = mkOption { - type = types.listOf types.path; - default = []; - example = literalExample "[ pkgs.dejavu_fonts ]"; - description = "List of primary font paths."; - }; + fonts.enableFontDir = mkOption { + default = false; + description = '' + Whether to enable font management and install configured fonts to + /Library/Fonts. + + NOTE: removes any manually-added fonts. + ''; + }; + + fonts.fonts = mkOption { + type = types.listOf types.path; + default = []; + example = literalExample "[ pkgs.dejavu_fonts ]"; + description = "List of fonts to install."; }; }; - + config = { - system.activationScripts.fonts.text = "" + optionalString cfg.enableFontDir '' + + system.build.fonts = pkgs.runCommandNoCC "fonts" + { paths = cfg.fonts; preferLocalBuild = true; } + '' + mkdir -p $out/Library/Fonts + for path in $paths; do + find -L $path/share/fonts -type f -print0 | while IFS= read -rd "" f; do + ln -s "$f" $out/Library/Fonts + done + done + ''; + + system.activationScripts.fonts.text = optionalString cfg.enableFontDir '' # Set up fonts. - echo "resetting fonts..." >&2 - fontrestore default -n 2>&1 | grep -o '/Library/Fonts/.*' | tr '\n' '\0' | xargs -0 rm || true - echo "updating fonts..." >&2 - ${fontLinks libraryLink systemFontsDir} + echo "configuring fonts..." >&2 + find -L "$systemConfig/Library/Fonts" -type f -print0 | while IFS= read -rd "" l; do + font=''${l##*/} + f=$(readlink -f "$l") + if [ ! -e "/Library/Fonts/$font" ] || [ $(stat -c '%i' "$f") != $(stat -c '%i' "/Library/Fonts/$font") ]; then + echo "updating font $font..." >&2 + # FIXME: hardlink, won't work if nix is on a dedicated filesystem. + ln -fn "$f" /Library/Fonts + fi + done + + fontrestore default -n 2>&1 | while read -r f; do + case $f in + /Library/Fonts/*) + font=''${f##*/} + if [ ! -e "$systemConfig/Library/Fonts/$font" ]; then + echo "removing font $font..." >&2 + rm "/Library/Fonts/$font" + fi + ;; + /*) + # ignoring unexpected fonts + ;; + esac + done ''; - environment.systemPackages = [ systemFontsDir ]; - environment.pathsToLink = [ "/share/fonts" ]; + }; } diff --git a/modules/system/default.nix b/modules/system/default.nix index 9feaf3fb..f5077502 100644 --- a/modules/system/default.nix +++ b/modules/system/default.nix @@ -91,6 +91,7 @@ in mkdir -p $out/Library ln -s ${cfg.build.applications}/Applications $out/Applications + ln -s ${cfg.build.fonts}/Library/Fonts $out/Library/Fonts ln -s ${cfg.build.launchd}/Library/LaunchAgents $out/Library/LaunchAgents ln -s ${cfg.build.launchd}/Library/LaunchDaemons $out/Library/LaunchDaemons diff --git a/tests/fonts.nix b/tests/fonts.nix index f3285921..be485270 100644 --- a/tests/fonts.nix +++ b/tests/fonts.nix @@ -12,9 +12,13 @@ in fonts.fonts = [ font ]; test = '' - echo checking installed fonts >&2 - grep -o "fontrestore default -n" ${config.out}/activate - grep -o "ln -fn '/run/current-system/sw/share/fonts/Font.ttf' '/Library/Fonts/Font.ttf'" ${config.out}/activate + echo "checking fonts in /Library/Fonts" >&2 + test -e ${config.out}/Library/Fonts/Font.ttf + + echo "checking activation of fonts in /activate" >&2 + grep "fontrestore default -n 2>&1" ${config.out}/activate + grep 'ln -fn ".*" /Library/Fonts' ${config.out}/activate + grep 'rm "/Library/Fonts/.*"' ${config.out}/activate ''; }