Skip to content

Commit 75a45e5

Browse files
committed
feat(clap_complete): Support multiple values after flags in native completions
1 parent 5d8c84b commit 75a45e5

File tree

2 files changed

+42
-21
lines changed

2 files changed

+42
-21
lines changed

clap_complete/src/dynamic/completer.rs

+28-7
Original file line numberDiff line numberDiff line change
@@ -92,7 +92,7 @@ pub fn complete(
9292
if value.is_some() {
9393
ParseState::ValueDone
9494
} else {
95-
ParseState::Opt(opt.unwrap())
95+
ParseState::Opt((opt.unwrap(), 1))
9696
}
9797
}
9898
Some(clap::ArgAction::SetTrue) | Some(clap::ArgAction::SetFalse) => {
@@ -115,7 +115,7 @@ pub fn complete(
115115
Some(opt) => {
116116
state = match short.next_value_os() {
117117
Some(_) => ParseState::ValueDone,
118-
None => ParseState::Opt(opt),
118+
None => ParseState::Opt((opt, 1)),
119119
};
120120
}
121121
None => {
@@ -128,9 +128,19 @@ pub fn complete(
128128
pos_index += 1;
129129
state = ParseState::ValueDone;
130130
}
131-
ParseState::Opt(_) => {
132-
state = ParseState::ValueDone;
133-
}
131+
ParseState::Opt((ref opt, count)) => match opt.get_num_args() {
132+
Some(range) => {
133+
let max = range.max_values();
134+
if count < max {
135+
state = ParseState::Opt((opt.clone(), count + 1));
136+
} else {
137+
state = ParseState::ValueDone;
138+
}
139+
}
140+
None => {
141+
state = ParseState::ValueDone;
142+
}
143+
},
134144
}
135145
}
136146
}
@@ -150,7 +160,7 @@ enum ParseState<'a> {
150160
Pos(usize),
151161

152162
/// Parsing a optional flag argument
153-
Opt(&'a clap::Arg),
163+
Opt((&'a clap::Arg, usize)),
154164
}
155165

156166
fn complete_arg(
@@ -298,8 +308,19 @@ fn complete_arg(
298308
completions.extend(complete_arg_value(arg.to_value(), positional, current_dir));
299309
}
300310
}
301-
ParseState::Opt(opt) => {
311+
ParseState::Opt((opt, count)) => {
302312
completions.extend(complete_arg_value(arg.to_value(), opt, current_dir));
313+
let min = opt.get_num_args().map(|r| r.min_values()).unwrap_or(0);
314+
if count > min {
315+
// Also complete this raw_arg as a positional argument, flags, options and subcommand.
316+
completions.extend(complete_arg(
317+
arg,
318+
cmd,
319+
current_dir,
320+
pos_index,
321+
ParseState::ValueDone,
322+
)?);
323+
}
303324
}
304325
}
305326
if completions.iter().any(|a| a.is_visible()) {

clap_complete/tests/testsuite/dynamic.rs

+14-14
Original file line numberDiff line numberDiff line change
@@ -469,12 +469,9 @@ val3"
469469
assert_data_eq!(
470470
complete!(cmd, "--certain-num val1 [TAB]"),
471471
snapbox::str![
472-
"--certain-num
473-
--uncertain-num
474-
--help\tPrint help
475-
-Y
476-
-N
477-
-h\tPrint help"
472+
"val1
473+
val2
474+
val3"
478475
]
479476
);
480477

@@ -502,7 +499,10 @@ val3"
502499
assert_data_eq!(
503500
complete!(cmd, "--uncertain-num val1 [TAB]"),
504501
snapbox::str![
505-
"--certain-num
502+
"val1
503+
val2
504+
val3
505+
--certain-num
506506
--uncertain-num
507507
--help\tPrint help
508508
-Y
@@ -535,12 +535,9 @@ val3"
535535
assert_data_eq!(
536536
complete!(cmd, "-Y val1 [TAB]"),
537537
snapbox::str![
538-
"--certain-num
539-
--uncertain-num
540-
--help\tPrint help
541-
-Y
542-
-N
543-
-h\tPrint help"
538+
"val1
539+
val2
540+
val3"
544541
]
545542
);
546543

@@ -568,7 +565,10 @@ val3"
568565
assert_data_eq!(
569566
complete!(cmd, "-N val1 [TAB]"),
570567
snapbox::str![
571-
"--certain-num
568+
"val1
569+
val2
570+
val3
571+
--certain-num
572572
--uncertain-num
573573
--help\tPrint help
574574
-Y

0 commit comments

Comments
 (0)