mirror of
https://github.com/nix-community/home-manager.git
synced 2026-06-05 21:02:51 +00:00
`literalExpression` is intended just to signify code that needs to stay a string that gets represented exactly as-is for docs. It has been misused heavily and people get confused repeatedly on when or not to use it because of the rampant misuse.
639 lines
18 KiB
Nix
639 lines
18 KiB
Nix
{
|
|
config,
|
|
lib,
|
|
options,
|
|
pkgs,
|
|
...
|
|
}:
|
|
|
|
let
|
|
inherit (lib)
|
|
literalExpression
|
|
mkEnableOption
|
|
mkIf
|
|
mkOption
|
|
mkPackageOption
|
|
optionals
|
|
types
|
|
;
|
|
|
|
cfg = config.programs.neovim;
|
|
|
|
inherit
|
|
(
|
|
(import ../../lib/file-type.nix {
|
|
inherit (config.home) homeDirectory;
|
|
inherit lib pkgs;
|
|
})
|
|
)
|
|
fileType
|
|
;
|
|
|
|
inherit (pkgs) neovimUtils;
|
|
jsonFormat = pkgs.formats.json { };
|
|
|
|
pluginTypeStateVersion = lib.hm.deprecations.mkStateVersionOptionDefault {
|
|
inherit (config.home) stateVersion;
|
|
since = "26.05";
|
|
optionPath = [
|
|
"programs"
|
|
"neovim"
|
|
"plugins"
|
|
"PLUGIN"
|
|
"type"
|
|
];
|
|
legacy.value = "viml";
|
|
current.value = "lua";
|
|
};
|
|
in
|
|
{
|
|
meta.maintainers = with lib.maintainers; [ khaneliman ];
|
|
|
|
imports = [
|
|
./fennel.nix
|
|
(lib.mkRenamedOptionModule
|
|
[ "programs" "neovim" "extraLuaConfig" ]
|
|
[ "programs" "neovim" "initLua" ]
|
|
)
|
|
];
|
|
|
|
options = {
|
|
programs.neovim = {
|
|
enable = mkEnableOption "Neovim";
|
|
|
|
package = mkPackageOption pkgs "neovim" { default = "neovim-unwrapped"; };
|
|
|
|
finalPackage = mkOption {
|
|
type = types.package;
|
|
readOnly = true;
|
|
description = "Resulting customized neovim package.";
|
|
};
|
|
|
|
# Aliases
|
|
viAlias = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Symlink {command}`vi` to {command}`nvim` binary.
|
|
'';
|
|
};
|
|
|
|
vimAlias = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Symlink {command}`vim` to {command}`nvim` binary.
|
|
'';
|
|
};
|
|
|
|
vimdiffAlias = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Alias {command}`vimdiff` to {command}`nvim -d`.
|
|
'';
|
|
};
|
|
|
|
defaultEditor = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Whether to configure {command}`nvim` as the default
|
|
editor using the {env}`EDITOR` and {env}`VISUAL`
|
|
environment variables.
|
|
'';
|
|
};
|
|
|
|
# Providers & Runtimes
|
|
withNodeJs = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Enable node provider. Set to `true` to
|
|
use Node plugins.
|
|
'';
|
|
};
|
|
|
|
withPerl = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Enable perl provider. Set to `true` to
|
|
use Perl plugins.
|
|
'';
|
|
};
|
|
|
|
withPython3 = mkOption {
|
|
type = types.bool;
|
|
inherit
|
|
(lib.hm.deprecations.mkStateVersionOptionDefault {
|
|
inherit (config.home) stateVersion;
|
|
since = "26.05";
|
|
optionPath = [
|
|
"programs"
|
|
"neovim"
|
|
"withPython3"
|
|
];
|
|
legacy.value = true;
|
|
current.value = false;
|
|
})
|
|
default
|
|
defaultText
|
|
;
|
|
description = ''
|
|
Enable Python 3 provider. Set to `true` to
|
|
use Python 3 plugins.
|
|
'';
|
|
};
|
|
|
|
withRuby = mkOption {
|
|
type = types.bool;
|
|
inherit
|
|
(lib.hm.deprecations.mkStateVersionOptionDefault {
|
|
inherit (config.home) stateVersion;
|
|
since = "26.05";
|
|
optionPath = [
|
|
"programs"
|
|
"neovim"
|
|
"withRuby"
|
|
];
|
|
legacy.value = true;
|
|
current.value = false;
|
|
})
|
|
default
|
|
defaultText
|
|
;
|
|
description = ''
|
|
Enable ruby provider.
|
|
'';
|
|
};
|
|
|
|
extraPython3Packages = mkOption {
|
|
type = types.functionTo (types.listOf types.package);
|
|
default = _: [ ];
|
|
defaultText = literalExpression "ps: [ ]";
|
|
example = literalExpression "pyPkgs: with pyPkgs; [ python-language-server ]";
|
|
description = ''
|
|
The extra Python 3 packages required for your plugins to work.
|
|
This option accepts a function that takes a Python 3 package set as an argument,
|
|
and selects the required Python 3 packages from this package set.
|
|
See the example for more info.
|
|
'';
|
|
};
|
|
|
|
extraLuaPackages = mkOption {
|
|
type = types.functionTo (types.listOf types.package);
|
|
default = _: [ ];
|
|
defaultText = literalExpression "ps: [ ]";
|
|
example = literalExpression "luaPkgs: with luaPkgs; [ luautf8 ]";
|
|
description = ''
|
|
The extra Lua packages required for your plugins to work.
|
|
This option accepts a function that takes a Lua package set as an argument,
|
|
and selects the required Lua packages from this package set.
|
|
See the example for more info.
|
|
'';
|
|
};
|
|
|
|
# Wrapper Configuration
|
|
extraName = mkOption {
|
|
type = types.str;
|
|
default = "";
|
|
description = ''
|
|
Extra name appended to the wrapper package name.
|
|
'';
|
|
};
|
|
|
|
autowrapRuntimeDeps = mkOption {
|
|
type = types.bool;
|
|
default = true;
|
|
description = ''
|
|
Whether to automatically wrap the binary with the runtime dependencies of the plugins.
|
|
'';
|
|
};
|
|
|
|
waylandSupport = mkOption {
|
|
type = types.bool;
|
|
default = pkgs.stdenv.isLinux;
|
|
defaultText = literalExpression "pkgs.stdenv.isLinux";
|
|
description = ''
|
|
Whether to enable Wayland clipboard support.
|
|
'';
|
|
};
|
|
|
|
extraWrapperArgs = mkOption {
|
|
type = types.listOf types.str;
|
|
default = [ ];
|
|
example = literalExpression ''
|
|
[
|
|
"--suffix"
|
|
"LIBRARY_PATH"
|
|
":"
|
|
"''${lib.makeLibraryPath [ pkgs.stdenv.cc.cc pkgs.zlib ]}"
|
|
"--suffix"
|
|
"PKG_CONFIG_PATH"
|
|
":"
|
|
"''${lib.makeSearchPathOutput "dev" "lib/pkgconfig" [ pkgs.stdenv.cc.cc pkgs.zlib ]}"
|
|
]
|
|
'';
|
|
description = ''
|
|
Extra arguments to be passed to the neovim wrapper.
|
|
This option sets environment variables required for building and running binaries
|
|
with external package managers like mason.nvim.
|
|
'';
|
|
};
|
|
|
|
extraPackages = mkOption {
|
|
type = types.listOf types.package;
|
|
default = [ ];
|
|
example = literalExpression "[ pkgs.shfmt ]";
|
|
description = "Extra packages available to nvim.";
|
|
};
|
|
|
|
# Configuration & Plugins
|
|
extraConfig = mkOption {
|
|
type = types.lines;
|
|
default = "";
|
|
example = ''
|
|
set nobackup
|
|
'';
|
|
description = ''
|
|
Custom vimrc lines.
|
|
'';
|
|
};
|
|
|
|
sideloadInitLua = mkOption {
|
|
type = types.bool;
|
|
default = false;
|
|
description = ''
|
|
Enable to avoid writing the content of {var}`initLua` to the default
|
|
location {file}`$XDG_CONFIG_HOME/nvim/init.lua` and load it through
|
|
neovim wrapper arguments instead.
|
|
|
|
|
|
This is useful if you want to manage your own {file}`init.lua` imperatively.
|
|
'';
|
|
};
|
|
|
|
initLua = mkOption {
|
|
type = types.lines;
|
|
default = "";
|
|
example = lib.literalExpression ''
|
|
let
|
|
nvimEarlyInit = lib.mkOrder 500 "set rtp+=vim.opt.rtp:prepend('/home/user/myplugin')";
|
|
nvimLateInit = lib.mkAfter "vim.opt.signcolumn = 'auto:1-3'";
|
|
in
|
|
lib.mkMerge [ nvimEarlyInit nvimLateInit ];
|
|
'';
|
|
description = ''
|
|
Content to be added to {file}`init.lua`.
|
|
|
|
Automatically contains the [advised plugin config](https://nixos.org/manual/nixpkgs/stable/#neovim-custom-configuration)
|
|
|
|
To specify the order, use `lib.mkOrder`, `lib.mkBefore`, `lib.mkAfter`.
|
|
'';
|
|
};
|
|
|
|
plugins =
|
|
let
|
|
pluginWithConfigType = types.submodule {
|
|
options = {
|
|
config = mkOption {
|
|
type = types.nullOr types.lines;
|
|
description = "Script to configure this plugin. The scripting language should match type.";
|
|
default = null;
|
|
};
|
|
|
|
type = mkOption {
|
|
type = types.either (types.enum [
|
|
"lua"
|
|
"viml"
|
|
"teal"
|
|
"fennel"
|
|
]) types.str;
|
|
description = "Language used in config. Configurations are aggregated per-language.";
|
|
default = pluginTypeStateVersion.effectiveDefault;
|
|
inherit (pluginTypeStateVersion) defaultText;
|
|
};
|
|
|
|
optional = mkEnableOption "optional" // {
|
|
description = "Don't load by default (load with :packadd)";
|
|
};
|
|
|
|
plugin = mkPackageOption pkgs.vimPlugins "plugin" {
|
|
default = null;
|
|
example = "pkgs.vimPlugins.nvim-treesitter";
|
|
pkgsText = "pkgs.vimPlugins";
|
|
};
|
|
|
|
runtime = mkOption {
|
|
default = { };
|
|
# passing actual "${xdg.configHome}/nvim" as basePath was a bit tricky
|
|
# due to how fileType.target is implemented
|
|
type = fileType "programs.neovim.plugins._.runtime" "{var}`xdg.configHome/nvim`" "nvim";
|
|
example = {
|
|
"ftplugin/c.vim".text = "setlocal omnifunc=v:lua.vim.lsp.omnifunc";
|
|
};
|
|
description = ''
|
|
Set of files that have to be linked in nvim config folder.
|
|
'';
|
|
};
|
|
};
|
|
};
|
|
|
|
in
|
|
mkOption {
|
|
type = types.listOf (types.either types.package pluginWithConfigType);
|
|
default = [ ];
|
|
example = literalExpression ''
|
|
with pkgs.vimPlugins;
|
|
[
|
|
yankring
|
|
vim-nix
|
|
{ plugin = vim-startify;
|
|
config = "let g:startify_change_to_vcs_root = 0";
|
|
}
|
|
]
|
|
'';
|
|
description = ''
|
|
List of vim plugins to install optionally associated with
|
|
configuration to be placed in init.vim.
|
|
|
|
This option is mutually exclusive with {var}`configure`.
|
|
'';
|
|
};
|
|
|
|
coc = {
|
|
enable = mkEnableOption "Coc";
|
|
|
|
package = mkPackageOption pkgs "coc-nvim" {
|
|
default = [
|
|
"vimPlugins"
|
|
"coc-nvim"
|
|
];
|
|
};
|
|
|
|
settings = mkOption {
|
|
inherit (jsonFormat) type;
|
|
default = { };
|
|
example = {
|
|
"suggest.noselect" = true;
|
|
"suggest.enablePreview" = true;
|
|
"suggest.enablePreselect" = false;
|
|
"suggest.disableKind" = true;
|
|
languageserver = {
|
|
haskell = {
|
|
command = "haskell-language-server-wrapper";
|
|
args = [ "--lsp" ];
|
|
rootPatterns = [
|
|
"*.cabal"
|
|
"stack.yaml"
|
|
"cabal.project"
|
|
"package.yaml"
|
|
"hie.yaml"
|
|
];
|
|
filetypes = [
|
|
"haskell"
|
|
"lhaskell"
|
|
];
|
|
};
|
|
};
|
|
};
|
|
description = ''
|
|
Extra configuration lines to add to
|
|
{file}`$XDG_CONFIG_HOME/nvim/coc-settings.json`
|
|
See
|
|
<https://github.com/neoclide/coc.nvim/wiki/Using-the-configuration-file>
|
|
for options.
|
|
'';
|
|
};
|
|
|
|
pluginConfig = mkOption {
|
|
type = types.lines;
|
|
default = "";
|
|
description = "Script to configure CoC. Must be viml.";
|
|
};
|
|
};
|
|
|
|
# Generated / Read-Only
|
|
generatedConfigViml = mkOption {
|
|
type = types.lines;
|
|
visible = true;
|
|
readOnly = true;
|
|
description = ''
|
|
Generated vimscript config.
|
|
'';
|
|
};
|
|
|
|
generatedConfigs = mkOption {
|
|
type = types.attrsOf types.lines;
|
|
visible = true;
|
|
readOnly = true;
|
|
example = literalExpression ''
|
|
{
|
|
viml = '''
|
|
" Generated by home-manager
|
|
map <leader> ,
|
|
''';
|
|
|
|
lua = '''
|
|
-- Generated by home-manager
|
|
vim.opt.background = "dark"
|
|
''';
|
|
}
|
|
'';
|
|
description = ''
|
|
Generated configurations with as key their language (set via type).
|
|
'';
|
|
};
|
|
};
|
|
};
|
|
|
|
config = mkIf cfg.enable (
|
|
let
|
|
legacyPluginTypeWarnings = lib.hm.deprecations.mkStateVersionListSubmoduleWarnings {
|
|
inherit (config.home) stateVersion;
|
|
since = "26.05";
|
|
baseWarning = pluginTypeStateVersion.warning;
|
|
definitions = options.programs.neovim.plugins.definitionsWithLocations;
|
|
shouldWarnEntry = lib.hm.deprecations.mkListEntryOmittedFieldPredicate {
|
|
omittedField = "type";
|
|
triggerField = "config";
|
|
};
|
|
describeEntry =
|
|
entry:
|
|
if entry.value ? plugin && entry.value.plugin != null then
|
|
"plugin `${lib.getName entry.value.plugin}`"
|
|
else
|
|
"a plugin entry";
|
|
extraEntryWarning =
|
|
_entry:
|
|
''Set `type = "viml"` or `type = "lua"` on that plugin entry to make the config language explicit.'';
|
|
};
|
|
|
|
allPlugins =
|
|
cfg.plugins
|
|
++ lib.optional cfg.coc.enable {
|
|
type = "viml";
|
|
plugin = cfg.coc.package;
|
|
config = cfg.coc.pluginConfig;
|
|
optional = false;
|
|
};
|
|
|
|
defaultPlugin = {
|
|
type = pluginTypeStateVersion.effectiveDefault;
|
|
plugin = null;
|
|
config = null;
|
|
optional = false;
|
|
runtime = { };
|
|
};
|
|
|
|
# transform all plugins into a standardized attrset
|
|
pluginsNormalized = map (
|
|
x: defaultPlugin // (if x ? plugin then x else { plugin = x; })
|
|
) allPlugins;
|
|
|
|
# remove attributes not understood by nixpkgs' "makeVimPackageInfo"
|
|
suppressIncompatibleConfig =
|
|
p:
|
|
lib.filterAttrs (
|
|
n: _v:
|
|
builtins.elem n [
|
|
"plugin"
|
|
"optional"
|
|
"config"
|
|
]
|
|
) (if p.type != "viml" then p // { config = null; } else p);
|
|
|
|
# Wrapper Arguments Construction
|
|
extraMakeWrapperArgs = optionals (cfg.extraPackages != [ ]) [
|
|
"--suffix"
|
|
"PATH"
|
|
":"
|
|
(lib.makeBinPath cfg.extraPackages)
|
|
];
|
|
|
|
nixpkgsCompatiblePlugins = map suppressIncompatibleConfig pluginsNormalized;
|
|
vimPackageInfo = neovimUtils.makeVimPackageInfo nixpkgsCompatiblePlugins;
|
|
|
|
wrappedNeovim' =
|
|
(pkgs.wrapNeovimUnstable cfg.package {
|
|
withNodeJs = cfg.withNodeJs || cfg.coc.enable;
|
|
plugins = nixpkgsCompatiblePlugins;
|
|
|
|
inherit (cfg)
|
|
extraLuaPackages
|
|
extraName
|
|
withPython3
|
|
withRuby
|
|
withPerl
|
|
viAlias
|
|
vimAlias
|
|
autowrapRuntimeDeps
|
|
waylandSupport
|
|
;
|
|
|
|
extraPython3Packages =
|
|
ps: (cfg.extraPython3Packages ps) ++ (lib.concatMap (f: f ps) vimPackageInfo.pluginPython3Packages);
|
|
neovimRcContent = cfg.extraConfig;
|
|
wrapperArgs = cfg.extraWrapperArgs ++ extraMakeWrapperArgs;
|
|
wrapRc = false;
|
|
}).overrideAttrs
|
|
{
|
|
|
|
# nixpkgs implementation dependend: avoid nixpkgs adding rtp/packpath wrapping arguments
|
|
packpathDirs.myNeovimPackages = {
|
|
start = [ ];
|
|
opt = [ ];
|
|
};
|
|
};
|
|
in
|
|
{
|
|
warnings = legacyPluginTypeWarnings;
|
|
|
|
programs.neovim = {
|
|
generatedConfigViml = cfg.extraConfig;
|
|
|
|
generatedConfigs =
|
|
let
|
|
grouped = lib.groupBy (x: x.type) pluginsNormalized;
|
|
configsOnly = lib.foldl (acc: p: if p.config != null then acc ++ [ p.config ] else acc) [ ];
|
|
in
|
|
lib.mapAttrs (_name: vals: lib.concatStringsSep "\n" (configsOnly vals)) grouped;
|
|
|
|
finalPackage = wrappedNeovim';
|
|
};
|
|
|
|
home = {
|
|
packages = [ cfg.finalPackage ];
|
|
|
|
sessionVariables = mkIf cfg.defaultEditor {
|
|
EDITOR = "nvim";
|
|
VISUAL = "nvim";
|
|
};
|
|
|
|
shellAliases = mkIf cfg.vimdiffAlias { vimdiff = "nvim -d"; };
|
|
};
|
|
|
|
programs.neovim.extraPackages = mkIf cfg.autowrapRuntimeDeps vimPackageInfo.runtimeDeps;
|
|
|
|
programs.neovim.extraWrapperArgs = mkIf (cfg.sideloadInitLua && cfg.initLua != "") [
|
|
"--add-flags"
|
|
''--cmd 'lua dofile("${pkgs.writeText "wrapper-init-lua" cfg.initLua}")' ''
|
|
];
|
|
|
|
programs.neovim.initLua =
|
|
let
|
|
# using default 'foldmarker', to be used with foldmethod=marker
|
|
foldedLuaBlock =
|
|
title: content:
|
|
if (content != "") then
|
|
''
|
|
-- ${title} {{{
|
|
${content}
|
|
-- }}}
|
|
''
|
|
else
|
|
null;
|
|
in
|
|
lib.mkMerge [
|
|
(lib.mkIf (wrappedNeovim'.luaRcContent != "") (
|
|
# we want it to appear rather early
|
|
lib.mkOrder 200 wrappedNeovim'.luaRcContent
|
|
))
|
|
(lib.mkIf (lib.hasAttr "lua" cfg.generatedConfigs && cfg.generatedConfigs.lua != "") (
|
|
lib.mkAfter (foldedLuaBlock "user-associated plugin config" cfg.generatedConfigs.lua)
|
|
))
|
|
|
|
];
|
|
|
|
# link the packpath in expected folder so that even unwrapped neovim can pick
|
|
# home-manager's plugins
|
|
xdg.dataFile."nvim/site/pack/hm" =
|
|
let
|
|
packpathDirs.hm = vimPackageInfo.vimPackage;
|
|
in
|
|
{
|
|
enable = allPlugins != [ ];
|
|
source = "${pkgs.neovimUtils.packDir packpathDirs}/pack/hm";
|
|
};
|
|
|
|
xdg.configFile = lib.mkMerge (
|
|
# writes runtime
|
|
(map (x: x.runtime) pluginsNormalized)
|
|
++ [
|
|
{
|
|
"nvim/init.lua" = mkIf (cfg.initLua != "") {
|
|
enable = !cfg.sideloadInitLua;
|
|
text = cfg.initLua;
|
|
};
|
|
|
|
"nvim/coc-settings.json" = mkIf cfg.coc.enable {
|
|
source = jsonFormat.generate "coc-settings.json" cfg.coc.settings;
|
|
};
|
|
}
|
|
]
|
|
);
|
|
}
|
|
);
|
|
}
|