mirror of
https://github.com/nushell/nushell.git
synced 2025-07-13 21:06:06 +02:00
* use arc to avoid cloning entire engine for menus * remove complete import path * remove stack clone * reference in completer
106 lines
3.7 KiB
Rust
106 lines
3.7 KiB
Rust
use nu_engine::documentation::get_flags_section;
|
|
use nu_protocol::{engine::EngineState, levenshtein_distance};
|
|
use reedline::{Completer, Suggestion};
|
|
use std::sync::Arc;
|
|
|
|
pub struct NuHelpCompleter(Arc<EngineState>);
|
|
|
|
impl NuHelpCompleter {
|
|
pub fn new(engine_state: Arc<EngineState>) -> Self {
|
|
Self(engine_state)
|
|
}
|
|
|
|
fn completion_helper(&self, line: &str, pos: usize) -> Vec<Suggestion> {
|
|
let full_commands = self.0.get_signatures_with_examples(false);
|
|
|
|
//Vec<(Signature, Vec<Example>, bool, bool)> {
|
|
let mut commands = full_commands
|
|
.iter()
|
|
.filter(|(sig, _, _, _)| {
|
|
sig.name.to_lowercase().contains(&line.to_lowercase())
|
|
|| sig.usage.to_lowercase().contains(&line.to_lowercase())
|
|
|| sig
|
|
.extra_usage
|
|
.to_lowercase()
|
|
.contains(&line.to_lowercase())
|
|
})
|
|
.collect::<Vec<_>>();
|
|
|
|
commands.sort_by(|(a, _, _, _), (b, _, _, _)| {
|
|
let a_distance = levenshtein_distance(line, &a.name);
|
|
let b_distance = levenshtein_distance(line, &b.name);
|
|
a_distance.cmp(&b_distance)
|
|
});
|
|
|
|
commands
|
|
.into_iter()
|
|
.map(|(sig, examples, _, _)| {
|
|
let mut long_desc = String::new();
|
|
|
|
let usage = &sig.usage;
|
|
if !usage.is_empty() {
|
|
long_desc.push_str(usage);
|
|
long_desc.push_str("\r\n\r\n");
|
|
}
|
|
|
|
let extra_usage = &sig.extra_usage;
|
|
if !extra_usage.is_empty() {
|
|
long_desc.push_str(extra_usage);
|
|
long_desc.push_str("\r\n\r\n");
|
|
}
|
|
|
|
long_desc.push_str(&format!("Usage:\r\n > {}\r\n", sig.call_signature()));
|
|
|
|
if !sig.named.is_empty() {
|
|
long_desc.push_str(&get_flags_section(sig))
|
|
}
|
|
|
|
if !sig.required_positional.is_empty()
|
|
|| !sig.optional_positional.is_empty()
|
|
|| sig.rest_positional.is_some()
|
|
{
|
|
long_desc.push_str("\r\nParameters:\r\n");
|
|
for positional in &sig.required_positional {
|
|
long_desc
|
|
.push_str(&format!(" {}: {}\r\n", positional.name, positional.desc));
|
|
}
|
|
for positional in &sig.optional_positional {
|
|
long_desc.push_str(&format!(
|
|
" (optional) {}: {}\r\n",
|
|
positional.name, positional.desc
|
|
));
|
|
}
|
|
|
|
if let Some(rest_positional) = &sig.rest_positional {
|
|
long_desc.push_str(&format!(
|
|
" ...{}: {}\r\n",
|
|
rest_positional.name, rest_positional.desc
|
|
));
|
|
}
|
|
}
|
|
|
|
let extra: Vec<String> = examples
|
|
.iter()
|
|
.map(|example| example.example.to_string())
|
|
.collect();
|
|
|
|
Suggestion {
|
|
value: sig.name.clone(),
|
|
description: Some(long_desc),
|
|
extra: Some(extra),
|
|
span: reedline::Span {
|
|
start: pos,
|
|
end: pos + line.len(),
|
|
},
|
|
}
|
|
})
|
|
.collect()
|
|
}
|
|
}
|
|
|
|
impl Completer for NuHelpCompleter {
|
|
fn complete(&mut self, line: &str, pos: usize) -> Vec<Suggestion> {
|
|
self.completion_helper(line, pos)
|
|
}
|
|
}
|