431 lines
15 KiB
Nix
431 lines
15 KiB
Nix
# tests available at pkgs/test/vim
|
|
{ lib, stdenv, vim, vimPlugins, buildEnv, writeText
|
|
, runCommand, makeWrapper
|
|
, python3
|
|
, callPackage, makeSetupHook
|
|
, linkFarm
|
|
}:
|
|
|
|
/*
|
|
|
|
USAGE EXAMPLE
|
|
=============
|
|
|
|
Install Vim like this eg using nixos option environment.systemPackages which will provide
|
|
vim-with-plugins in PATH:
|
|
|
|
vim-full.customize {
|
|
name = "vim-with-plugins"; # optional
|
|
|
|
# add custom .vimrc lines like this:
|
|
vimrcConfig.customRC = ''
|
|
set hidden
|
|
'';
|
|
|
|
# store your plugins in Vim packages
|
|
vimrcConfig.packages.myVimPackage = with pkgs.vimPlugins; {
|
|
# loaded on launch
|
|
start = [ youcompleteme fugitive ];
|
|
# manually loadable by calling `:packadd $plugin-name`
|
|
opt = [ phpCompletion elm-vim ];
|
|
# To automatically load a plugin when opening a filetype, add vimrc lines like:
|
|
# autocmd FileType php :packadd phpCompletion
|
|
};
|
|
};
|
|
|
|
WHAT IS A VIM PLUGIN?
|
|
=====================
|
|
Typical plugin files:
|
|
|
|
plugin/P1.vim
|
|
autoload/P1.vim
|
|
ftplugin/xyz.vim
|
|
doc/plugin-documentation.txt (traditional documentation)
|
|
README(.md) (nowadays thanks to github)
|
|
|
|
|
|
Vim offers the :h rtp setting which works for most plugins. Thus adding
|
|
this to your .vimrc should make most plugins work:
|
|
|
|
set rtp+=~/.nix-profile/share/vim-plugins/youcompleteme
|
|
" or for p in ["youcompleteme"] | exec 'set rtp+=~/.nix-profile/share/vim-plugins/'.p | endfor
|
|
|
|
Learn about about plugin Vim plugin mm managers at
|
|
http://vim-wiki.mawercer.de/wiki/topic/vim%20plugin%20managment.html.
|
|
|
|
The documentation can be accessed by Vim's :help command if it was tagged.
|
|
See vimHelpTags sample code below.
|
|
|
|
CONTRIBUTING AND CUSTOMIZING
|
|
============================
|
|
The example file pkgs/applications/editors/vim/plugins/default.nix provides
|
|
both:
|
|
* manually mantained plugins
|
|
* plugins created by VAM's nix#ExportPluginsForNix implementation
|
|
|
|
I highly recommend to lookup vim plugin attribute names at the [vim-pi] project
|
|
which is a database containing all plugins from
|
|
vim.org and quite a lot of found at github and similar sources. vim-pi's documented purpose
|
|
is to associate vim.org script ids to human readable names so that dependencies
|
|
can be describe easily.
|
|
|
|
How to find a name?
|
|
* http://vam.mawercer.de/ or VAM's
|
|
* grep vim-pi
|
|
* use VAM's completion or :AddonsInfo command
|
|
|
|
It might happen than a plugin is not known by vim-pi yet. We encourage you to
|
|
contribute to vim-pi so that plugins can be updated automatically.
|
|
|
|
|
|
CREATING DERIVATIONS AUTOMATICALLY BY PLUGIN NAME
|
|
==================================================
|
|
Most convenient is to use a ~/.vim-scripts file putting a plugin name into each line
|
|
as documented by [VAM]'s README.md
|
|
It is the same format you pass to vimrcConfig.vam.pluginDictionaries from the
|
|
usage example above.
|
|
|
|
Then create a temp vim file and insert:
|
|
|
|
let opts = {}
|
|
let opts.path_to_nixpkgs = '/etc/nixos/nixpkgs'
|
|
let opts.cache_file = '/tmp/export-vim-plugin-for-nix-cache-file'
|
|
let opts.plugin_dictionaries = map(readfile("vim-plugins"), 'eval(v:val)')
|
|
" add more files
|
|
" let opts.plugin_dictionaries += map(.. other file )
|
|
call nix#ExportPluginsForNix(opts)
|
|
|
|
Then ":source %" it.
|
|
|
|
nix#ExportPluginsForNix is provided by ./vim2nix
|
|
|
|
A buffer will open containing the plugin derivation lines as well list
|
|
fitting the vimrcConfig.vam.pluginDictionaries option.
|
|
|
|
Thus the most simple usage would be:
|
|
|
|
vim_with_plugins =
|
|
let vim = vim-full;
|
|
inherit (vimUtil.override {inherit vim}) rtpPath addRtp buildVimPlugin vimHelpTags;
|
|
vimPlugins = [
|
|
# the derivation list from the buffer created by nix#ExportPluginsForNix
|
|
# don't set which will default to pkgs.vimPlugins
|
|
];
|
|
in vim.customize {
|
|
name = "vim-with-plugins";
|
|
|
|
vimrcConfig.customRC = '' .. '';
|
|
|
|
vimrcConfig.vam.knownPlugins = vimPlugins;
|
|
vimrcConfig.vam.pluginDictionaries = [
|
|
# the plugin list form ~/.vim-scripts turned into nix format added to
|
|
# the buffer created by the nix#ExportPluginsForNix
|
|
];
|
|
}
|
|
|
|
vim_with_plugins can be installed like any other application within Nix.
|
|
|
|
[VAM] https://github.com/MarcWeber/vim-addon-manager
|
|
[vim-pi] https://bitbucket.org/vimcommunity/vim-pi
|
|
*/
|
|
|
|
|
|
let
|
|
inherit lib;
|
|
|
|
# make sure a plugin is a derivation and its dependencies are derivations. If
|
|
# plugin already is a derivation, this is a no-op. If it is a string, it is
|
|
# looked up in knownPlugins.
|
|
pluginToDrv = knownPlugins: plugin:
|
|
let
|
|
drv =
|
|
if builtins.isString plugin then
|
|
# make sure `pname` is set to that we are able to convert the derivation
|
|
# back to a string.
|
|
( knownPlugins.${plugin} // { pname = plugin; })
|
|
else
|
|
plugin;
|
|
in
|
|
# make sure all the dependencies of the plugin are also derivations
|
|
drv // { dependencies = map (pluginToDrv knownPlugins) (drv.dependencies or []); };
|
|
|
|
# transitive closure of plugin dependencies (plugin needs to be a derivation)
|
|
transitiveClosure = plugin:
|
|
[ plugin ] ++ (
|
|
lib.unique (builtins.concatLists (map transitiveClosure plugin.dependencies or []))
|
|
);
|
|
|
|
findDependenciesRecursively = plugins: lib.concatMap transitiveClosure plugins;
|
|
|
|
vamDictToNames = x:
|
|
if builtins.isString x then [x]
|
|
else (lib.optional (x ? name) x.name)
|
|
++ (x.names or []);
|
|
|
|
rtpPath = ".";
|
|
|
|
vimFarm = prefix: name: drvs:
|
|
let mkEntryFromDrv = drv: { name = "${prefix}/${lib.getName drv}"; path = drv; };
|
|
in linkFarm name (map mkEntryFromDrv drvs);
|
|
|
|
/* Generates a packpath folder as expected by vim
|
|
Example:
|
|
packDir (myVimPackage.{ start = [ vimPlugins.vim-fugitive ]; opt = [] })
|
|
=> "/nix/store/xxxxx-pack-dir"
|
|
*/
|
|
packDir = packages:
|
|
let
|
|
packageLinks = packageName: {start ? [], opt ? []}:
|
|
let
|
|
# `nativeImpl` expects packages to be derivations, not strings (as
|
|
# opposed to older implementations that have to maintain backwards
|
|
# compatibility). Therefore we don't need to deal with "knownPlugins"
|
|
# and can simply pass `null`.
|
|
depsOfOptionalPlugins = lib.subtractLists opt (findDependenciesRecursively opt);
|
|
startWithDeps = findDependenciesRecursively start;
|
|
allPlugins = lib.unique (startWithDeps ++ depsOfOptionalPlugins);
|
|
allPython3Dependencies = ps:
|
|
lib.flatten (builtins.map (plugin: (plugin.python3Dependencies or (_: [])) ps) allPlugins);
|
|
python3Env = python3.withPackages allPython3Dependencies;
|
|
|
|
packdirStart = vimFarm "pack/${packageName}/start" "packdir-start" allPlugins;
|
|
packdirOpt = vimFarm "pack/${packageName}/opt" "packdir-opt" opt;
|
|
# Assemble all python3 dependencies into a single `site-packages` to avoid doing recursive dependency collection
|
|
# for each plugin.
|
|
# This directory is only for python import search path, and will not slow down the startup time.
|
|
# see :help python3-directory for more details
|
|
python3link = runCommand "vim-python3-deps" {} ''
|
|
mkdir -p $out/pack/${packageName}/start/__python3_dependencies
|
|
ln -s ${python3Env}/${python3Env.sitePackages} $out/pack/${packageName}/start/__python3_dependencies/python3
|
|
'';
|
|
in
|
|
[ packdirStart packdirOpt ] ++ lib.optional (allPython3Dependencies python3.pkgs != []) python3link;
|
|
in
|
|
buildEnv {
|
|
name = "vim-pack-dir";
|
|
paths = (lib.flatten (lib.mapAttrsToList packageLinks packages));
|
|
};
|
|
|
|
nativeImpl = packages:
|
|
''
|
|
set packpath^=${packDir packages}
|
|
set runtimepath^=${packDir packages}
|
|
'';
|
|
|
|
/* Generates a vimrc string
|
|
|
|
packages is an attrset with {name: { start = [ vim derivations ]; opt = [ vim derivations ]; }
|
|
Example:
|
|
vimrcContent {
|
|
|
|
packages = { home-manager = { start = [vimPlugins.vim-fugitive]; opt = [];};
|
|
beforePlugins = '';
|
|
customRC = ''let mapleader = " "'';
|
|
|
|
};
|
|
*/
|
|
vimrcContent = {
|
|
packages ? null,
|
|
vam ? null, # deprecated
|
|
pathogen ? null, # deprecated
|
|
plug ? null,
|
|
beforePlugins ? ''
|
|
" configuration generated by NIX
|
|
set nocompatible
|
|
'',
|
|
customRC ? null
|
|
}:
|
|
|
|
let
|
|
/* vim-plug is an extremely popular vim plugin manager.
|
|
*/
|
|
plugImpl =
|
|
''
|
|
source ${vimPlugins.vim-plug}/plug.vim
|
|
silent! call plug#begin('/dev/null')
|
|
|
|
'' + (lib.concatMapStringsSep "\n" (pkg: "Plug '${pkg}'") plug.plugins) + ''
|
|
|
|
call plug#end()
|
|
'';
|
|
|
|
# vim-addon-manager = VAM (deprecated)
|
|
vamImpl =
|
|
let
|
|
knownPlugins = vam.knownPlugins or vimPlugins;
|
|
|
|
# plugins specified by the user
|
|
specifiedPlugins = map (pluginToDrv knownPlugins) (lib.concatMap vamDictToNames vam.pluginDictionaries);
|
|
# plugins with dependencies
|
|
plugins = findDependenciesRecursively specifiedPlugins;
|
|
vamPackages.vam = {
|
|
start = plugins;
|
|
};
|
|
in
|
|
nativeImpl vamPackages;
|
|
|
|
entries = [
|
|
beforePlugins
|
|
]
|
|
++ lib.optional (vam != null) (lib.warn "'vam' attribute is deprecated. Use 'packages' instead in your vim configuration" vamImpl)
|
|
++ lib.optional (packages != null && packages != []) (nativeImpl packages)
|
|
++ lib.optional (pathogen != null) (throw "pathogen is now unsupported, replace `pathogen = {}` with `packages.home = { start = []; }`")
|
|
++ lib.optional (plug != null) plugImpl
|
|
++ [ customRC ];
|
|
|
|
in
|
|
lib.concatStringsSep "\n" (lib.filter (x: x != null && x != "") entries);
|
|
|
|
vimrcFile = settings: writeText "vimrc" (vimrcContent settings);
|
|
|
|
in
|
|
|
|
rec {
|
|
inherit vimrcFile;
|
|
inherit vimrcContent;
|
|
inherit packDir;
|
|
|
|
makeCustomizable = let
|
|
mkVimrcFile = vimrcFile; # avoid conflict with argument name
|
|
in vim: vim // {
|
|
# Returns a customized vim that uses the specified vimrc configuration.
|
|
customize =
|
|
{ # The name of the derivation.
|
|
name ? "vim"
|
|
, # A shell word used to specify the names of the customized executables.
|
|
# The shell variable $exe can be used to refer to the wrapped executable's name.
|
|
# Examples: "my-$exe", "$exe-with-plugins", "\${exe/vim/v1m}"
|
|
executableName ?
|
|
if lib.hasInfix "vim" name then
|
|
lib.replaceStrings [ "vim" ] [ "$exe" ] name
|
|
else
|
|
"\${exe/vim/${lib.escapeShellArg name}}"
|
|
, # A custom vimrc configuration, treated as an argument to vimrcContent (see the documentation in this file).
|
|
vimrcConfig ? null
|
|
, # A custom vimrc file.
|
|
vimrcFile ? null
|
|
, # A custom gvimrc file.
|
|
gvimrcFile ? null
|
|
, # If set to true, return the *vim wrappers only.
|
|
# If set to false, overlay the wrappers on top of the original vim derivation.
|
|
# This ensures that things like man pages and .desktop files are available.
|
|
standalone ? name != "vim" && wrapManual != true
|
|
|
|
, # deprecated arguments (TODO: remove eventually)
|
|
wrapManual ? null, wrapGui ? null, vimExecutableName ? null, gvimExecutableName ? null,
|
|
}:
|
|
lib.warnIf (wrapManual != null) ''
|
|
vim.customize: wrapManual is deprecated: the manual is now included by default if `name == "vim"`.
|
|
${if wrapManual == true && name != "vim" then "Set `standalone = false` to include the manual."
|
|
else lib.optionalString (wrapManual == false && name == "vim") "Set `standalone = true` to get the *vim wrappers only."
|
|
}''
|
|
lib.warnIf (wrapGui != null)
|
|
"vim.customize: wrapGui is deprecated: gvim is now automatically included if present"
|
|
lib.throwIfNot (vimExecutableName == null && gvimExecutableName == null)
|
|
"vim.customize: (g)vimExecutableName is deprecated: use executableName instead (see source code for examples)"
|
|
(let
|
|
vimrc =
|
|
if vimrcFile != null then vimrcFile
|
|
else if vimrcConfig != null then mkVimrcFile vimrcConfig
|
|
else throw "at least one of vimrcConfig and vimrcFile must be specified";
|
|
bin = runCommand "${name}-bin" { nativeBuildInputs = [ makeWrapper ]; } ''
|
|
vimrc=${lib.escapeShellArg vimrc}
|
|
gvimrc=${lib.optionalString (gvimrcFile != null) (lib.escapeShellArg gvimrcFile)}
|
|
|
|
mkdir -p "$out/bin"
|
|
for exe in ${
|
|
if standalone then "{,g,r,rg,e}vim {,g}vimdiff vi"
|
|
else "{,g,r,rg,e}{vim,view} {,g}vimdiff ex vi"
|
|
}; do
|
|
if [[ -e ${vim}/bin/$exe ]]; then
|
|
dest="$out/bin/${executableName}"
|
|
if [[ -e $dest ]]; then
|
|
echo "ambiguous executableName: ''${dest##*/} already exists"
|
|
continue
|
|
fi
|
|
makeWrapper ${vim}/bin/"$exe" "$dest" \
|
|
--add-flags "-u ''${vimrc@Q} ''${gvimrc:+-U ''${gvimrc@Q}}"
|
|
fi
|
|
done
|
|
'';
|
|
in if standalone then bin else
|
|
buildEnv {
|
|
inherit name;
|
|
paths = [ (lib.lowPrio vim) bin ];
|
|
});
|
|
|
|
override = f: makeCustomizable (vim.override f);
|
|
overrideAttrs = f: makeCustomizable (vim.overrideAttrs f);
|
|
};
|
|
|
|
vimWithRC = throw "vimWithRC was removed, please use vim.customize instead";
|
|
|
|
vimGenDocHook = callPackage ({ vim }:
|
|
makeSetupHook {
|
|
name = "vim-gen-doc-hook";
|
|
propagatedBuildInputs = [ vim ];
|
|
substitutions = {
|
|
vimBinary = "${vim}/bin/vim";
|
|
inherit rtpPath;
|
|
};
|
|
} ./vim-gen-doc-hook.sh) {};
|
|
|
|
vimCommandCheckHook = callPackage ({ neovim-unwrapped }:
|
|
makeSetupHook {
|
|
name = "vim-command-check-hook";
|
|
propagatedBuildInputs = [ neovim-unwrapped ];
|
|
substitutions = {
|
|
vimBinary = "${neovim-unwrapped}/bin/nvim";
|
|
inherit rtpPath;
|
|
};
|
|
} ./vim-command-check-hook.sh) {};
|
|
|
|
neovimRequireCheckHook = callPackage ({ neovim-unwrapped }:
|
|
makeSetupHook {
|
|
name = "neovim-require-check-hook";
|
|
propagatedBuildInputs = [ neovim-unwrapped ];
|
|
substitutions = {
|
|
nvimBinary = "${neovim-unwrapped}/bin/nvim";
|
|
inherit rtpPath;
|
|
};
|
|
} ./neovim-require-check-hook.sh) {};
|
|
|
|
inherit (import ./build-vim-plugin.nix {
|
|
inherit lib stdenv rtpPath toVimPlugin;
|
|
}) buildVimPlugin buildVimPluginFrom2Nix;
|
|
|
|
|
|
# used to figure out which python dependencies etc. neovim needs
|
|
requiredPlugins = {
|
|
packages ? {},
|
|
plug ? null, ...
|
|
}:
|
|
let
|
|
nativePluginsConfigs = lib.attrsets.attrValues packages;
|
|
nonNativePlugins = (lib.optionals (plug != null) plug.plugins);
|
|
nativePlugins = lib.concatMap (requiredPluginsForPackage) nativePluginsConfigs;
|
|
in
|
|
nativePlugins ++ nonNativePlugins;
|
|
|
|
|
|
# figures out which python dependencies etc. is needed for one vim package
|
|
requiredPluginsForPackage = { start ? [], opt ? []}:
|
|
start ++ opt;
|
|
|
|
toVimPlugin = drv:
|
|
drv.overrideAttrs(oldAttrs: {
|
|
# dont move the "doc" folder since vim expects it
|
|
forceShare = [ "man" "info" ];
|
|
|
|
nativeBuildInputs = oldAttrs.nativeBuildInputs or []
|
|
++ lib.optionals (stdenv.hostPlatform == stdenv.buildPlatform) [
|
|
vimCommandCheckHook vimGenDocHook
|
|
# many neovim plugins keep using buildVimPlugin
|
|
neovimRequireCheckHook
|
|
];
|
|
|
|
passthru = (oldAttrs.passthru or {}) // {
|
|
vimPlugin = true;
|
|
};
|
|
});
|
|
}
|