-
-
Notifications
You must be signed in to change notification settings - Fork 2.6k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
command expansions #3393
command expansions #3393
Changes from 4 commits
6f6cb3c
784380f
85d38a7
22d17f9
7e7c0dc
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1822,6 +1822,42 @@ fn run_shell_command( | |
Ok(()) | ||
} | ||
|
||
pub fn process_cmd( | ||
cx: &mut compositor::Context, | ||
input: &str, | ||
event: PromptEvent, | ||
) -> anyhow::Result<()> { | ||
let input = expand_args(cx.editor, input); | ||
let parts = input.split_whitespace().collect::<Vec<&str>>(); | ||
if parts.is_empty() { | ||
return Ok(()); | ||
} | ||
|
||
// If command is numeric, interpret as line number and go there. | ||
if parts.len() == 1 && parts[0].parse::<usize>().ok().is_some() { | ||
if let Err(e) = typed::goto_line_number(cx, &[Cow::from(parts[0])], event) { | ||
cx.editor.set_error(format!("{}", e)); | ||
return Err(e); | ||
} | ||
return Ok(()); | ||
} | ||
|
||
// Handle typable commands | ||
if let Some(cmd) = typed::TYPABLE_COMMAND_MAP.get(parts[0]) { | ||
let shellwords = shellwords::Shellwords::from(input.as_ref()); | ||
let args = shellwords.words(); | ||
|
||
if let Err(e) = (cmd.fun)(cx, &args[1..], event) { | ||
cx.editor.set_error(format!("{}", e)); | ||
return Err(e); | ||
} | ||
} else if event == PromptEvent::Validate { | ||
cx.editor | ||
.set_error(format!("no such command: '{}'", parts[0])); | ||
} | ||
Ok(()) | ||
} | ||
|
||
pub const TYPABLE_COMMAND_LIST: &[TypableCommand] = &[ | ||
TypableCommand { | ||
name: "quit", | ||
|
@@ -2414,31 +2450,7 @@ pub(super) fn command_mode(cx: &mut Context) { | |
} | ||
}, // completion | ||
move |cx: &mut compositor::Context, input: &str, event: PromptEvent| { | ||
let parts = input.split_whitespace().collect::<Vec<&str>>(); | ||
if parts.is_empty() { | ||
return; | ||
} | ||
|
||
// If command is numeric, interpret as line number and go there. | ||
if parts.len() == 1 && parts[0].parse::<usize>().ok().is_some() { | ||
if let Err(e) = typed::goto_line_number(cx, &[Cow::from(parts[0])], event) { | ||
cx.editor.set_error(format!("{}", e)); | ||
} | ||
return; | ||
} | ||
|
||
// Handle typable commands | ||
if let Some(cmd) = typed::TYPABLE_COMMAND_MAP.get(parts[0]) { | ||
let shellwords = Shellwords::from(input); | ||
let args = shellwords.words(); | ||
|
||
if let Err(e) = (cmd.fun)(cx, &args[1..], event) { | ||
cx.editor.set_error(format!("{}", e)); | ||
} | ||
} else if event == PromptEvent::Validate { | ||
cx.editor | ||
.set_error(format!("no such command: '{}'", parts[0])); | ||
} | ||
let _ = process_cmd(cx, input, event); | ||
}, | ||
); | ||
prompt.doc_fn = Box::new(|input: &str| { | ||
|
@@ -2460,3 +2472,55 @@ pub(super) fn command_mode(cx: &mut Context) { | |
prompt.recalculate_completion(cx.editor); | ||
cx.push_layer(Box::new(prompt)); | ||
} | ||
|
||
fn expand_args<'a>(editor: &mut Editor, args: &'a str) -> Cow<'a, str> { | ||
let reg = Regex::new(r"%(\w+)\s*\{(.*)").unwrap(); | ||
reg.replace(args, |caps: ®ex::Captures| { | ||
let remaining = &caps[2]; | ||
let end = find_first_open_right_braces(remaining); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should we auto-complete for users? I think There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. yes, above command won't expand and will be treated literately . The auto-complete is a very nice to have feature, how about we adding it after this PR. |
||
let exp = expand_args(editor, &remaining[..end]); | ||
let doc = doc!(editor); | ||
let rep = match &caps[1] { | ||
"val" => match exp.trim() { | ||
"filename" => doc.path().and_then(|p| p.to_str()).unwrap_or("").to_owned(), | ||
"dirname" => doc | ||
.path() | ||
.and_then(|p| p.parent()) | ||
.and_then(|p| p.to_str()) | ||
.unwrap_or("") | ||
.to_owned(), | ||
_ => "".into(), | ||
}, | ||
"sh" => { | ||
let shell = &editor.config().shell; | ||
if let Ok((output, _)) = shell_impl(shell, &exp, None) { | ||
output.trim().into() | ||
} else { | ||
"".into() | ||
} | ||
} | ||
_ => "".into(), | ||
}; | ||
let next = expand_args(editor, remaining.get(end + 1..).unwrap_or("")); | ||
format!("{rep} {next}") | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The added space between rep and next is a problem when you want to compose a single argument with the substitution. e.g. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Good point, will fix it once I have time. |
||
}) | ||
} | ||
|
||
fn find_first_open_right_braces(str: &str) -> usize { | ||
let mut left_count = 1; | ||
for (i, &b) in str.as_bytes().iter().enumerate() { | ||
match char::from_u32(b as u32) { | ||
Comment on lines
+2891
to
+2892
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Shouldn't we iterate over char directly instead? I think unicode might crash in this case when we are reading bytes within multiple bytes char. Also, the count with unicode characters is incorrect now, There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. the There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. If these patterns are coming from user input there's always a chance that they can be multi-byte, or even multi code-point "characters". Because of that you might want the parser to be aware / tolerant of it. If that is the case then let y = "y̆";
let mut chars = y.chars();
assert_eq!(Some('y'), chars.next()); // not 'y̆' The typical crate I've used / seen used is https://crates.io/crates/unicode-segmentation . Specifically for this implementation you could use grapheme_indices https://unicode-rs.github.io/unicode-segmentation/unicode_segmentation/trait.UnicodeSegmentation.html#tymethod.grapheme_indices . There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The returned index will always be in bounds (the enumerator and len are both byte based). Trying something like the following doesn't cause any issues I think this should remain as is. However, downstream I am seeing something off in shellwords (some multi-byte characters are lost in the generated argument list, I am investigating it now but it is unrelated to this PR) |
||
Some('}') => { | ||
left_count -= 1; | ||
if left_count == 0 { | ||
return i; | ||
} | ||
} | ||
Some('{') => { | ||
left_count += 1; | ||
} | ||
_ => {} | ||
} | ||
} | ||
str.len() | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't this return an error too?