From 840b7658fdf46c60feeee3335ab55068e24cc27d Mon Sep 17 00:00:00 2001 From: Maxime Coste Date: Thu, 24 Mar 2016 14:01:59 +0000 Subject: [PATCH] Add an alternative -shell-candidates shell completion support -shell-candidates use a shell script that returns all the candidates and then sort them using Kakoune ranked matches system instead of delegating the whole completion to the shell script (as shell-completion does) --- rc/base/ctags.kak | 4 ++-- src/commands.cc | 33 ++++++++++++++++++++++++++++++++- 2 files changed, 34 insertions(+), 3 deletions(-) diff --git a/rc/base/ctags.kak b/rc/base/ctags.kak index ca31cbab..0fadfa3c 100644 --- a/rc/base/ctags.kak +++ b/rc/base/ctags.kak @@ -6,9 +6,9 @@ decl str-list ctagsfiles 'tags' def -params 0..1 \ - -shell-completion ' + -shell-candidates ' ( for tags in $(printf %s "${kak_opt_ctagsfiles}" | tr \':\' \'\n\'); - do readtags -t "${tags}" -p "$1" + do readtags -t "${tags}" -p "" done ) | cut -f 1 | sort | uniq' \ -docstring 'Jump to tag definition' \ tag \ diff --git a/src/commands.cc b/src/commands.cc index a483573b..e6f0af0f 100644 --- a/src/commands.cc +++ b/src/commands.cc @@ -827,6 +827,36 @@ void define_command(const ParametersParser& parser, Context& context, const Shel return Completions{ 0_byte, pos_in_token, split(output, '\n', 0) }; }; } + else if (auto shell_cmd_opt = parser.get_switch("shell-candidates")) + { + String shell_cmd = shell_cmd_opt->str(); + Vector candidates; + int token = -1; + completer = [shell_cmd, candidates, token]( + const Context& context, CompletionFlags flags, CommandParameters params, + size_t token_to_complete, ByteCount pos_in_token) mutable + { + if (token != token_to_complete) + { + if (flags == CompletionFlags::Fast) // no shell on fast completion + return Completions{}; + + ShellContext shell_context{ + params, + { { "token_to_complete", to_string(token_to_complete) } } + }; + String output = ShellManager::instance().eval(shell_cmd, context, {}, + ShellManager::Flags::WaitForStdout, + shell_context).first; + candidates = split(output, '\n', 0); + token = token_to_complete; + } + + return token == token_to_complete ? + Completions{ 0_byte, pos_in_token, complete(params[token_to_complete], pos_in_token, candidates) } + : Completions{}; + }; + } else if (parser.get_switch("command-completion")) { completer = [](const Context& context, CompletionFlags flags, @@ -857,7 +887,8 @@ const CommandDesc define_command_cmd = { { "client-completion", { false, "complete parameters using client name completion" } }, { "buffer-completion", { false, "complete parameters using buffer name completion" } }, { "command-completion", { false, "complete parameters using kakoune command completion" } }, - { "shell-completion", { true, "complete the parameters using the given shell-script" } } }, + { "shell-completion", { true, "complete the parameters using the given shell-script" } }, + { "shell-candidates", { true, "get the parameter candidates using the given shell-script" } } }, ParameterDesc::Flags::None, 2, 2 },