nix-community.nixvim/plugins/completion/nvim-cmp/default.nix

678 lines
21 KiB
Nix
Raw Normal View History

{
pkgs,
config,
lib,
...
} @ args:
with lib; let
cfg = config.plugins.nvim-cmp;
helpers = import ../../helpers.nix {inherit lib;};
cmpLib = import ./cmp-helpers.nix args;
snippetEngines = {
"vsnip" = ''vim.fn["vsnip#anonymous"](args.body)'';
"luasnip" = ''require('luasnip').lsp_expand(args.body)'';
"snippy" = ''require('snippy').expand_snippet(args.body)'';
"ultisnips" = ''vim.fn["UltiSnips#Anon"](args.body)'';
};
in {
options.plugins.nvim-cmp = {
2023-01-22 03:32:08 +00:00
enable = mkEnableOption "nvim-cmp";
package = helpers.mkPackageOption "nvim-cmp" pkgs.vimPlugins.nvim-cmp;
general: add package options (#127) * barbar: package option * Base16: package option * gruvbox: package option * nord: package option * one: package option * onedark: package option * tokyonight: package option * nvim-cmp: package option * coq: package option * lspkind: package option * helpers: added package option to mkPlugin * fugitive: package option * gitgutter: package option * gitsigns: package option * neogit: package option * ledger: package option * nix: package option * plantuml-syntax: package option * treesitter-context: package option + formatting * treesitter-refactor: package option + formatting * treesitter: package option * zig: package option * null-ls: package option * null-ls/servers: package option * lsp-lines: package option * lspsaga: package option * trouble: package option * luasnip: added description for package option * airline: package option * lightline: package option * lualine: package option * telescope: package option * telescope/frecency: package option * telescope/fzf-native: package option * telescope/media-files: package option * comment-nvim: package option * vim-commentary: package option * dashboard: package option * easyescape: package option * emmet: package option * endwise: package option * floaterm: package option * goyo: package option * intellitab: package option * mark-radar: package option * notify: package option * nvim-autopairs: package option * nvim-tree: package option * project-nvim: package option * specs: package option * startify: package option * surround: package option * undotree: package option
2023-01-19 10:45:15 +00:00
performance = helpers.mkCompositeOption "Performance options" {
debounce = helpers.defaultNullOpts.mkInt 60 ''
Sets debounce time
This is the interval used to group up completions from different sources
for filtering and displaying.
'';
throttle = helpers.defaultNullOpts.mkInt 30 ''
Sets throttle time.
This is used to delay filtering and displaying completions.
'';
fetchingTimeout = helpers.defaultNullOpts.mkInt 500 ''
Sets the timeout of candidate fetching process.
The nvim-cmp will wait to display the most prioritized source.
'';
};
preselect = helpers.defaultNullOpts.mkEnumFirstDefault ["Item" "None"] ''
- "Item": nvim-cmp will preselect the item that the source specified.
- "None": nvim-cmp will not preselect any items.
'';
mapping = mkOption {
default = null;
type = with types;
nullOr (attrsOf (either str (types.submodule ({...}: {
options = {
action = mkOption {
type = types.nonEmptyStr;
description = "The function the mapping should call";
example = ''"cmp.mapping.scroll_docs(-4)"'';
};
modes = mkOption {
default = null;
type = types.nullOr (types.listOf types.str);
example = ''[ "i" "s" ]'';
};
};
}))));
example = ''
{
"<CR>" = "cmp.mapping.confirm({ select = true })";
"<Tab>" = {
modes = [ "i" "s" ];
action = '${""}'
function(fallback)
if cmp.visible() then
cmp.select_next_item()
elseif luasnip.expandable() then
luasnip.expand()
elseif luasnip.expand_or_jumpable() then
luasnip.expand_or_jump()
elseif check_backspace() then
fallback()
else
fallback()
end
end
'${""}';
};
}
'';
};
mappingPresets = mkOption {
default = [];
type = types.listOf (types.enum [
"insert"
"cmdline"
]);
description = ''
Mapping presets to use; cmp.mapping.preset.
\$\{mappingPreset} will be called with the configured mappings.
'';
example = ''[ "insert" "cmdline" ]'';
};
snippet = helpers.mkCompositeOption "Snippet options" {
expand =
helpers.mkNullOrOption
(
types.either
helpers.rawType
(types.enum (attrNames snippetEngines))
)
''
The snippet expansion function. That's how nvim-cmp interacts with a
particular snippet engine.
You may directly provide one of those four supported engines:
- vsnip
- luasnip
- snippy
- ultisnips
You can also provide a custom function:
```
{
__raw = \'\'
function(args)
vim.fn["vsnip#anonymous"](args.body) -- For `vsnip` users.
-- require('luasnip').lsp_expand(args.body) -- For `luasnip` users.
-- require('snippy').expand_snippet(args.body) -- For `snippy` users.
-- vim.fn["UltiSnips#Anon"](args.body) -- For `ultisnips` users.
end
\'\';
};
```
'';
};
completion = helpers.mkCompositeOption "Completion options" {
keywordLength = helpers.defaultNullOpts.mkInt 1 ''
The number of characters needed to trigger auto-completion.
'';
keywordPattern =
helpers.defaultNullOpts.mkStr
''\%(-\?\d\+\%(\.\d\+\)\?\|\h\w*\%(-\w*\)*\)''
''
The default keyword pattern.
Note: the provided pattern will be embedded as such: `[[PATTERN]]`.
'';
autocomplete =
helpers.defaultNullOpts.mkNullable
(
with types;
either
(listOf str)
(types.enum [false])
)
''[ "TextChanged" ]''
''
The event to trigger autocompletion.
If set to `"false"`, then completion is only invoked manually
(e.g. by calling `cmp.complete`).
'';
completeopt = helpers.defaultNullOpts.mkStr "menu,menuone,noselect" ''
Like vim's completeopt setting. In general, you don't need to change this.
'';
};
confirmation = helpers.mkCompositeOption "Confirmation options" {
getCommitCharacters =
helpers.defaultNullOpts.mkStr
''
function(commit_characters)
return commit_characters
end
''
''
You can append or exclude commitCharacters via this configuration option
function. The commitCharacters are defined by the LSP spec.
'';
};
formatting = helpers.mkCompositeOption "Formatting options" {
expandableIndicator = helpers.defaultNullOpts.mkBool true ''
Boolean to show the `~` expandable indicator in cmp's floating window.
'';
fields =
helpers.defaultNullOpts.mkNullable
(types.listOf types.str)
''[ "kind" "abbr" "menu" ]''
"An array of completion fields to specify their order.";
format =
helpers.defaultNullOpts.mkStr
''
function(_, vim_item)
return vim_item
end
''
''
`fun(entry: cmp.Entry, vim_item: vim.CompletedItem): vim.CompletedItem`
The function used to customize the appearance of the completion menu. See
|complete-items|. This value can also be used to modify the `dup` property.
NOTE: The `vim.CompletedItem` can contain the special properties
`abbr_hl_group`, `kind_hl_group` and `menu_hl_group`.
'';
};
matching = helpers.mkCompositeOption "Matching options" {
disallowFuzzyMatching = helpers.defaultNullOpts.mkBool false ''
Whether to allow fuzzy matching.
'';
disallowFullfuzzyMatching = helpers.defaultNullOpts.mkBool false ''
Whether to allow full-fuzzy matching.
'';
disallowPartialFuzzyMatching = helpers.defaultNullOpts.mkBool true ''
Whether to allow fuzzy matching without prefix matching.
'';
disallowPartialMatching = helpers.defaultNullOpts.mkBool false ''
Whether to allow partial matching.
'';
disallowPrefixUnmatching = helpers.defaultNullOpts.mkBool false ''
Whether to allow prefix unmatching.
'';
};
sorting = helpers.mkCompositeOption "Sorting options" {
priorityWeight = helpers.defaultNullOpts.mkInt 2 ''
Each item's original priority (given by its corresponding source) will be
increased by `#sources - (source_index - 1)` and multiplied by `priority_weight`.
That is, the final priority is calculated by the following formula:
`final_score = orig_score + ((#sources - (source_index - 1)) * sorting.priority_weight)`
'';
comparators =
helpers.defaultNullOpts.mkNullable (types.listOf types.str)
''[ "offset" "exact" "score" "recently_used" "locality" "kind" "length" "order" ]''
''
The function to customize the sorting behavior.
You can use built-in comparators via `cmp.config.compare.*`.
Signature: `(fun(entry1: cmp.Entry, entry2: cmp.Entry): boolean | nil)[]`
'';
};
autoEnableSources = mkOption {
type = types.bool;
default = true;
description = ''
Scans the sources array and installs the plugins if they are known to nixvim.
'';
};
sources = let
source_config = types.submodule ({...}: {
options = {
name = mkOption {
type = types.str;
description = "The name of the source.";
example = ''"buffer"'';
};
option = helpers.mkNullOrOption (types.attrs) ''
Any specific options defined by the source itself.
If direct lua code is needed use `helpers.mkRaw`.
'';
keywordLength = helpers.mkNullOrOption types.int ''
The source-specific keyword length to trigger auto completion.
'';
keywordPattern = helpers.mkNullOrOption types.str ''
The source-specific keyword pattern.
Note: the provided pattern will be embedded as such: `[[PATTERN]]`.
'';
triggerCharacters = helpers.mkNullOrOption (types.listOf types.str) ''
A source-specific keyword pattern.
'';
priority = helpers.mkNullOrOption types.int "The source-specific priority value.";
maxItemCount = helpers.mkNullOrOption types.int "The source-specific item count.";
groupIndex = helpers.mkNullOrOption types.int ''
The source group index.
For instance, you can set the `buffer`'s source `group_index` to a larger number
if you don't want to see `buffer` source items while `nvim-lsp` source is available:
```
cmp.setup {
sources = {
{ name = 'nvim_lsp', group_index = 1 },
{ name = 'buffer', group_index = 2 },
}
}
```
You can also achieve this by using the built-in configuration helper like this:
```
cmp.setup {
sources = cmp.config.sources({
{ name = 'nvim_lsp' },
}, {
{ name = 'buffer' },
})
}
```
'';
entryFilter = helpers.mkNullOrOption types.str ''
A source-specific entry filter, with the following function signature:
`function(entry: cmp.Entry, ctx: cmp.Context): boolean`
Returning `true` will keep the entry, while returning `false` will remove it.
This can be used to hide certain entries from a given source. For instance, you
could hide all entries with kind `Text` from the `nvim_lsp` filter using the
following source definition:
```
{
name = 'nvim_lsp',
entry_filter = function(entry, ctx)
return require('cmp.types').lsp.CompletionItemKind[entry:get_kind()] ~= 'Text'
end
}
```
Using the `ctx` parameter, you can further customize the behaviour of the source.
'';
};
});
in
mkOption {
default = null;
type = with types;
nullOr (
either
(listOf source_config)
(listOf (listOf source_config))
);
description = ''
The sources to use.
Can either be a list of sourceConfigs which will be made directly to a Lua object.
Or it can be a list of lists, which will use the cmp built-in helper function
`cmp.config.sources`.
Default: [ ]
'';
example = ''
[
{ name = "nvim_lsp"; }
{ name = "luasnip"; } #For luasnip users.
{ name = "path"; }
{ name = "buffer"; }
]
'';
};
view = helpers.mkCompositeOption "View options" {
entries =
helpers.defaultNullOpts.mkNullable (with types; either str attrs)
''
{
name = "custom";
selection_order = "top_down";
}
''
''
The view class used to customize nvim-cmp's appearance.
'';
};
window = let
# Reusable options
mkBorderOption = default:
helpers.defaultNullOpts.mkNullable
(with types; either str (listOf str))
default
''
Border characters used for the completion popup menu when |experimental.native_menu| is disabled.
See |nvim_open_win|.
'';
mkWinhighlightOption = default:
helpers.defaultNullOpts.mkStr
default
''
Specify the window's winhighlight option.
See |nvim_open_win|.
'';
zindex = helpers.mkNullOrOption types.int ''
The completion window's zindex.
See |nvim_open_win|.
'';
in
helpers.mkCompositeOption "Windows options" {
completion = helpers.mkCompositeOption "Completion window options" {
border = mkBorderOption ''[ "" "" "" "" "" "" "" "" ]'';
winhighlight =
mkWinhighlightOption
"Normal:Pmenu,FloatBorder:Pmenu,CursorLine:PmenuSel,Search:None";
inherit zindex;
scrolloff = helpers.defaultNullOpts.mkInt 0 ''
Specify the window's scrolloff option.
See |'scrolloff'|.
'';
colOffset = helpers.defaultNullOpts.mkInt 0 ''
Offsets the completion window relative to the cursor.
'';
sidePadding = helpers.defaultNullOpts.mkInt 1 ''
The amount of padding to add on the completion window's sides.
'';
scrollbar = helpers.defaultNullOpts.mkBool true ''
Whether the scrollbar should be enabled if there are more items that fit
'';
};
documentation = helpers.mkCompositeOption "Documentation window options" {
border = mkBorderOption ''[ "" "" "" " " "" "" "" " " ]'';
winhighlight = mkWinhighlightOption "FloatBorder:NormalFloat";
inherit zindex;
maxWidth =
helpers.defaultNullOpts.mkNullable
(types.either types.int types.str)
"math.floor((40 * 2) * (vim.o.columns / (40 * 2 * 16 / 9)))"
"The documentation window's max width.";
maxHeight =
helpers.defaultNullOpts.mkNullable
(types.either types.int types.str)
"math.floor(40 * (40 / vim.o.lines))"
"The documentation window's max height.";
};
};
# This can be kept as types.attrs since experimental features are often removed or completely
# changed after a while
experimental = helpers.mkNullOrOption types.attrs "Experimental features";
};
config = let
options = {
enabled = cfg.enable;
performance = cfg.performance;
preselect =
helpers.ifNonNull' cfg.preselect
(helpers.mkRaw "cmp.PreselectMode.${cfg.preselect}");
# Not very readable sorry
# If null then null
# If an attribute is a string, just treat it as lua code for that mapping
# If an attribute is a module, create a mapping with cmp.mapping() using the action as the first input and the modes as the second.
mapping = let
mappings =
helpers.ifNonNull' cfg.mapping
(mapAttrs
(bind: mapping:
helpers.mkRaw (
if isString mapping
then mapping
else let
modes = mapping.modes;
modesString =
optionalString (!isNull modes && ((length modes) >= 1))
("," + (helpers.toLuaObject mapping.modes));
in "cmp.mapping(${mapping.action}${modesString})"
))
cfg.mapping);
luaMappings = helpers.toLuaObject mappings;
wrapped =
lists.fold
(
presetName: prevString: ''cmp.mapping.preset.${presetName}(${prevString})''
)
luaMappings
cfg.mappingPresets;
in
helpers.mkRaw wrapped;
snippet = helpers.ifNonNull' cfg.snippet {
expand = let
expand = cfg.snippet.expand;
in
if isString expand
then
helpers.mkRaw ''
function(args)
${snippetEngines.${expand}}
end
''
else expand;
};
completion = helpers.ifNonNull' cfg.completion {
keyword_length = cfg.completion.keywordLength;
keyword_pattern = let
keywordPattern = cfg.completion.keywordPattern;
in
helpers.ifNonNull' keywordPattern
(helpers.mkRaw "[[${keywordPattern}]]");
autocomplete = let
autocomplete = cfg.completion.autocomplete;
in
if isList autocomplete
then
map
(triggerEvent:
helpers.mkRaw "require('cmp.types').cmp.TriggerEvent.${triggerEvent}")
autocomplete
# either null or false
else autocomplete;
completeopt = cfg.completion.completeopt;
};
confirmation = helpers.ifNonNull' cfg.confirmation {
get_commit_characters =
if (isString cfg.confirmation.getCommitCharacters)
then helpers.mkRaw cfg.confirmation.getCommitCharacters
else cfg.confirmation.getCommitCharacters;
};
formatting = helpers.ifNonNull' cfg.formatting {
expandable_indicator = cfg.formatting.expandableIndicator;
fields = cfg.formatting.fields;
format =
helpers.ifNonNull' cfg.formatting.format
(helpers.mkRaw cfg.formatting.format);
};
matching = helpers.ifNonNull' cfg.matching {
disallow_fuzzy_matching = cfg.matching.disallowFuzzyMatching;
disallow_fullfuzzy_matching = cfg.matching.disallowFullfuzzyMatching;
disallow_partial_fuzzy_matching = cfg.matching.disallowPartialFuzzyMatching;
disallow_partial_matching = cfg.matching.disallowPartialMatching;
disallow_prefix_unmatching = cfg.matching.disallowPrefixUnmatching;
};
sorting = helpers.ifNonNull' cfg.sorting {
priority_weight = cfg.sorting.priorityWeight;
comparators = let
comparators = cfg.sorting.comparators;
in
helpers.ifNonNull' comparators
(
map
(
funcName:
helpers.mkRaw "require('cmp.config.compare').${funcName}"
)
comparators
);
};
sources = helpers.ifNonNull' cfg.sources (
map
(source: {
inherit (source) name option;
keyword_length = source.keywordLength;
keywordPattern =
helpers.ifNonNull' source.keywordPattern
(helpers.mkRaw "[[${source.keywordPattern}]]");
trigger_characters = source.triggerCharacters;
inherit (source) priority;
max_item_count = source.maxItemCount;
group_index = source.groupIndex;
entry_filter = source.entryFilter;
})
cfg.sources
);
view = cfg.view;
window = helpers.ifNonNull' cfg.window {
completion = helpers.ifNonNull' cfg.window.completion {
inherit
(cfg.window.completion)
border
winhighlight
zindex
scrolloff
scrollbar
;
col_offset = cfg.window.completion.colOffset;
side_padding = cfg.window.completion.sidePadding;
};
documentation = helpers.ifNonNull' cfg.window.completion {
inherit
(cfg.window.completion)
border
winhighlight
zindex
;
max_width = let
maxWidth = cfg.window.documentation.maxWidth;
in
if isInt maxWidth
then maxWidth
else helpers.ifNonNull' maxWidth (helpers.mkRaw maxWidth);
max_height = let
maxHeight = cfg.window.documentation.maxHeight;
in
if isInt maxHeight
then maxHeight
else helpers.ifNonNull' maxHeight (helpers.mkRaw maxHeight);
};
};
experimental = cfg.experimental;
};
in
mkIf cfg.enable {
extraPlugins = [cfg.package];
extraConfigLua = helpers.wrapDo ''
local cmp = require('cmp')
cmp.setup(${helpers.toLuaObject options})
'';
# If auto_enable_sources is set to true, figure out which are provided by the user
# and enable the corresponding plugins.
plugins = let
flattened_sources =
if (isNull cfg.sources)
then []
else flatten cfg.sources;
# Take only the names from the sources provided by the user
found_sources = lists.unique (lists.map (source: source.name) flattened_sources);
# A list of known source names
known_source_names = attrNames cmpLib.pluginAndSourceNames;
attrs_enabled = listToAttrs (map
(name: {
name = cmpLib.pluginAndSourceNames.${name};
value.enable = mkIf (elem name found_sources) true;
})
known_source_names);
in
mkMerge [
(mkIf cfg.autoEnableSources attrs_enabled)
(mkIf (elem "nvim_lsp" found_sources)
{
lsp.capabilities = ''
capabilities = require('cmp_nvim_lsp').default_capabilities()
'';
})
];
};
}