/base16.nix

Make applications use your favourite base16 colorschemes on NixOS / home-manager.

Primary LanguageNixMIT LicenseMIT

logo

demo

What's base16 and base16.nix?

  • base16 is a theming standard by which hundreds of colorschemes and application configuration templates were made over the years.
  • base16.nix is a NixOS / home-manager module and a library that makes using base16 / base24 schemes and templates as simple as possible, while leaving the user full flexibility.

Features

With base16.nix, you can:

  • use existing schemes, override them, write a new one in YAML / nix;
  • theme any application with a base16 template in minimal amount of key strokes, or write a new template in mustache / nix.

Nonfeatures

  • base16.nix is simply a ultrasupermega convenient battle-tested nix-y base16 standard implementation, not a ricing engine — but if you want one, please check out every r/unixporn admin's dream — Stylix! built with base16.nix!
  • base16.nix neither aggregates nor vendors schemes / templates to keep data and logic separated.
  • No support for legacy template formats like Embedded Ruby and EJS.

👀 Module example (covers majority of use-cases)

In this example, we will use base16.nix as a NixOS module to theme zathura, neovim and alacritty to use the nord scheme (home-manager module works the same way).

Import and set the scheme (step 1/2)

In your NixOS configuration directory:

flake.nix

{ inputs = {
  # Add base16.nix, base16 schemes and
  # zathura and vim templates to the flake inputs.
  base16.url = github:SenchoPens/base16.nix;

  base16-schemes = {
    url = github:base16-project/base16-schemes;
    flake = false;
  };

  base16-zathura = {
    url = github:haozeke/base16-zathura;
    flake = false;
  };

  base16-vim = {
    url = github:base16-project/base16-vim;
    flake = false;
  };
  ...
};
outputs = { self, ... } @ inputs {
  ...
    nixosSystem {
      modules = [
        # import the base16.nix module
        base16.nixosModule
        # set system's scheme to nord by setting `config.scheme`
        { scheme = "${inputs.base16-schemes}/nord.yaml"; }
        # import `theming.nix`, we will write it in the next, final, step
        ./theming.nix
        ...
      ];
      # so you can use `inputs` in config files
      specialArgs = {
        inherit inputs;
      };
      ...
    };
  ...
};
... }

Theme (step 2/2)

Now that config.scheme is set, we can use it like a function to create themes from templates.

theming.nix

{ config, pkgs, inputs, ... }:
{
  # Theme zathura
  home-manager.users.sencho.programs.zathura.extraConfig =
    builtins.readFile (config.scheme inputs.base16-zathura);

  # Theme `neovim` — more complex, but the principle is the same.
  home-manager.users.sencho.programs.neovim = {
    plugins = [ (pkgs.vimPlugins.base16-vim.overrideAttrs (old:
      let schemeFile = config.scheme inputs.base16-vim;
      in { patchPhase = ''cp ${schemeFile} colors/base16-scheme.vim''; }
    )) ];
    extraConfig = ''
      set termguicolors background=dark
      let base16colorspace=256
      colorscheme base16-scheme
    '';
  };

  # Theme `alacritty`. home-manager doesn't provide an `extraConfig`,
  # but gives us `settings.colors` option of attrs type to set colors. 
  # As alacritty expects colors to begin with `#`, we use an attribute `withHashtag`.
  # Notice that we now use `config.scheme` as an attrset, and that this attrset,
  # besides from having attributes `base00`...`base0F`, has mnemonic attributes (`red`, etc.) -
  # read more on that in the next section.
  home-manager.users.sencho.programs.alacritty.settings.colors =
    with config.scheme.withHashtag; let default = {
        black = base00; white = base07;
        inherit red green yellow blue cyan magenta;
      };
    in {
      primary = { background = base00; foreground = base07; };
      cursor = { text = base02; cursor = base07; };
      normal = default; bright = default; dim = default;
    };
}

That's all, we themed 3 applications!

The attentive reader will notice that after setting config.scheme to a string, we use it as a function (to theme zathura and neovim) and as an attrset (to theme alacritty) — that's base16.nix' magic! Read the Documentation section to see how it works.

🍳 How To

Import a scheme from a YAML file
config.scheme = "${inputs.base16-schemes}/nord.yaml";
Override a scheme

We need to explicitly use mkSchemeAttrs function to use the override field of the resulting scheme attrs:

config.scheme = (config.lib.base16.mkSchemeAttrs "${inputs.base16-schemes}/nord.yaml").override {
  scheme = "Now it's my scheme >:]";
  base00 = "000000";  # make background completely black
};
Declare a scheme in Nix
config.scheme = {
  slug = "balsoftheme"; scheme = "Theme by balsoft"; author = "balsoft";
  base00 = "000000"; base01 = "333333"; base02 = "666666"; base03 = "999999";
  base04 = "cccccc"; base05 = "ffffff"; base06 = "e6e6e6"; base07 = "e6e6e6";
  base08 = "bf4040"; base09 = "bf8040"; base0A = "bfbf40"; base0B = "80bf40";
  base0C = "40bfbf"; base0D = "407fbf"; base0E = "7f40bf"; base0F = "bf40bf";
};

source

Use multiple schemes simultaneously

Achieve this by theming without config.scheme — by calling mkSchemeAttrs:

home-manager.users.sencho.programs.zathura.extraConfig =
  builtins.readFile (config.lib.base16.mkSchemeAttrs inputs.base16-schemes inputs.base16-zathura);

Without importing base16.nix as a module at all:

home-manager.users.sencho.programs.zathura.extraConfig =
  builtins.readFile ((pkgs.callPackage inputs.base16.lib {}).mkSchemeAttrs inputs.base16-schemes inputs.base16-zathura);
Use template variation

Template repositories often define more than one template variation. For example, zathura template repository defines default.mustache (colors only the interface) and recolor.mustache (colors the interface and pdfs).

By default base16.nix uses default.mustache. To use another template, e.g. recolor.mustache:

home-manager.users.sencho.programs.zathura.extraConfig =
  builtins.readFile (config.scheme {
    templateRepo = inputs.base16-zathura; target = "recolor";
  });
Override a template

Sample use-case: suppose you like zathura's default.mustache template, but want to change the background (default-bg) from base00 to base01.

  1. Override the scheme only for zathura:
home-manager.users.sencho.programs.zathura.extraConfig =
  builtins.readFile ((config.scheme.override {
    base00 = config.scheme.base01;
  }) inputs.base16-zathura);

Keep in mind that by doing so you'll change not only default-bg color, but also inputbar-bg, notification-bg, etc.

  1. Copy-paste the template and modify it:
home-manager.users.sencho.programs.zathura.extraConfig =
  builtins.readFile (config.scheme { template = ''
    ... 
    set default-bg   "#{{base01-hex}}"  # <-- we changed this
    set default-fg   "#{{base01-hex}}"

    set statusbar-fg "#{{base04-hex}}"
    set statusbar-bg "#{{base02-hex}}"
    ...
  ''; });

📚 Documentation

Consult the DOCUMENTATION.md to learn about every feature in detail and see how base16.nix works underhood.

🤍 Repositories that use base16.nix

NixOS modules:

  • Stylix — System-wide colorscheming and typography for NixOS.
  • tmux-flake — a flake that configures tmux.

Configs by:

Please feel free to list your repository above, it will make my day :)

🎎 Alternatives

  • base16-nix by @atpotts and its forks, notably base16-nix by @AlukardBF and base16-nix by @lukebfox.

  • nix-colors by @misterio. Thanks for the competition spirit! :)) And for the nix-wallpaper function, with which the preview GIF was generated.

    differences:

    Roughly nix-colors can be viewed as an alternative to base16.nix + Stylix, without the mustache template support:

    base16.nix supports the existing ≥ 80 mustache templates, nix-colors does not — instead there are ≥ 4 contributed nix functions and planned (at the time of writing) support for translation from mustache templates to nix functions. Stylix has ≥ 10 Stylix theming nix functions.

    You can generate base16 scheme from a wallpaper — in nix-colors via flavours and in Stylix via home-made CIE-LAB colorspace Haskell genetic algorithm.

    Also, if you use nix-colors without it's nix functions, it does not depend on nixpkgs.

  • theme-base16 by @rycee.

☎️ Troubleshooting

Error / incorrect behavior after updating base16.nix or adding a new source / template

The most probable reason of such an error is either a scheme or a template YAML file. Since version v2.0.0 base16.nix parses the YAML file in pure Nix to bypass IFD issues. The parser work for most base16-<scheme-name>.yaml and templates' config.yaml files, but, as YAML can be quite complicated, sometimes they can be parsed incorrectly.

The exact error depends on the point of failure. It probably will be cryptic if incorrect parsing caused an issue during nix evaluation. Otherwise, if your flake evaluates (nix flake check succeeds), the error will look something like this:

error: builder for '/nix/store/snbbfb43qphzfl6xr1mjs0mr8jny66x9-base16-nix-parse-check.drv' failed with exit code 1;
       last 7 log lines:
       > running tests
       > Output of "jd /nix/store/9jvxabhfx9acrysknblg0r2hzvcwv6ab-fromYAML /nix/store/qwmj9cbg7fpi5fvyd2x3kywfbw7hlm8f-parsed-yaml-as-json":
       > @ ["gotcha"]
       > - ["1 2"]
       > + "[ 1 2 ]"
       > Error: /nix/store/qhdqwj0mfp8qn0gq5s95pgd2i57lb09c-source/base16-kandinsky.yaml was parsed incorrectly during nix evaluation.
       > Please consult https://github.com/SenchoPens/base16.nix/tree/main#%EF%B8%8F-troubleshooting

This check happens by default for templates by installing a special derivation. You can do it for scheme too by adding the config.scheme.check derivation to your NixOS / home-manager package list.

Fix incorrectly parsed YAML file

  • If the problem is with a scheme YAML file and the nix evaluates, add the config.scheme.check derivation to your NixOS / home-manager package list, this will indicate which part of the YAML is being parsed incorrectly.
  • If you think that it is safe to ignore this error when handling a template, turn off the check:
    home-manager.users.sencho.programs.zathura.extraConfig =
      builtins.readFile (config.scheme {
        check-parsed-config-yaml = false;
        templateRepo = inputs.base16-zathura; target = "recolor";
      });
  • Enable IFD (but beware of a possible error described below): If the problem is in the scheme YAML file, parse it with config.lib.base16.yaml2attrs-ifd first:
    config.scheme = config.lib.base16.yaml2attrs-ifd "${inputs.base16-schemes}/nord.yaml";
    If the problem is in the template templates/config.yaml file, turn on use-ifd:
    home-manager.users.sencho.programs.zathura.extraConfig =
      builtins.readFile (config.scheme {
        use-ifd = true;
        templateRepo = inputs.base16-zathura; target = "recolor";
      });
  • Submit an issue.
  • Fix the YAML upstream. Probable causes: trailing spaces, file structure differs from typical config.yaml / scheme YAML files.
  • Fix the Nix parser 😈.
Error on `nix flake check` or `nix flake show`

First, check that you have the most recent version of base16.nix. If updating doesn't help, check that you don't turn on use-ifd in any of the scheme calls (template instantiations) and that you don't invoke yaml2attrs-ifd function.

Relevant issue: #3.

If neither of the above listed solutions do not work for you, please reopen it.

Anyhow, feel free to open an issue!

💙 Acknowledgments

Thanks to:

👩‍💻 Contributing

See CONTRIBUTING.md