use roff::{bold, italic, roman, Inline, Roff}; pub(crate) fn subcommand_heading(cmd: &clap::Command) -> &str { match cmd.get_subcommand_help_heading() { Some(title) => title, None => "SUBCOMMANDS", } } pub(crate) fn about(roff: &mut Roff, cmd: &clap::Command) { let s = match cmd.get_about().or_else(|| cmd.get_long_about()) { Some(about) => format!("{} - {}", cmd.get_name(), about), None => cmd.get_name().to_string(), }; roff.text([roman(&s)]); } pub(crate) fn description(roff: &mut Roff, cmd: &clap::Command) { if let Some(about) = cmd.get_long_about().or_else(|| cmd.get_about()) { for line in about.lines() { if line.trim().is_empty() { roff.control("PP", []); } else { roff.text([roman(line)]); } } } } pub(crate) fn synopsis(roff: &mut Roff, cmd: &clap::Command) { let mut line = vec![bold(cmd.get_name()), roman(" ")]; for opt in cmd.get_arguments().filter(|i| !i.is_hide_set()) { let (lhs, rhs) = option_markers(opt); match (opt.get_short(), opt.get_long()) { (Some(short), Some(long)) => { line.push(roman(lhs)); line.push(bold(&format!("-{}", short))); line.push(roman("|")); line.push(bold(&format!("--{}", long))); line.push(roman(rhs)); line.push(roman(" ")); } (Some(short), None) => { line.push(roman(lhs)); line.push(bold(&format!("-{} ", short))); line.push(roman(rhs)); line.push(roman(" ")); } (None, Some(long)) => { line.push(roman(lhs)); line.push(bold(&format!("--{}", long))); line.push(roman(rhs)); line.push(roman(" ")); } (None, None) => (), }; } for arg in cmd.get_positionals() { let (lhs, rhs) = option_markers(arg); line.push(roman(lhs)); if let Some(value) = arg.get_value_names() { line.push(italic(value.join(" "))); } else { line.push(italic(arg.get_id().as_str())); } line.push(roman(rhs)); line.push(roman(" ")); } if cmd.has_subcommands() { let (lhs, rhs) = subcommand_markers(cmd); line.push(roman(lhs)); line.push(italic( cmd.get_subcommand_value_name() .unwrap_or_else(|| subcommand_heading(cmd)) .to_lowercase(), )); line.push(roman(rhs)); } roff.text(line); } pub(crate) fn options(roff: &mut Roff, cmd: &clap::Command) { let items: Vec<_> = cmd.get_arguments().filter(|i| !i.is_hide_set()).collect(); for opt in items.iter().filter(|a| !a.is_positional()) { let mut header = match (opt.get_short(), opt.get_long()) { (Some(short), Some(long)) => { vec![short_option(short), roman(", "), long_option(long)] } (Some(short), None) => vec![short_option(short)], (None, Some(long)) => vec![long_option(long)], (None, None) => vec![], }; if let Some(value) = &opt.get_value_names() { header.push(roman("=")); header.push(italic(&value.join(" "))); } if let Some(defs) = option_default_values(opt) { header.push(roman(" ")); header.push(roman(&defs)); } let mut body = vec![]; let mut help_written = false; if let Some(help) = opt.get_long_help().or_else(|| opt.get_help()) { help_written = true; body.push(roman(help)); } let possibles = &opt.get_possible_values(); if !(possibles.is_empty() || opt.is_hide_possible_values_set()) { if help_written { // It looks nice to have a separation between the help and the values body.push(Inline::LineBreak); } let possible_vals = possibles.iter().filter(|pos| !pos.is_hide_set()).collect(); body.append(&mut format_possible_values(possible_vals)); } roff.control("TP", []); roff.text(header); roff.text(body); if let Some(env) = option_environment(opt) { roff.control("RS", []); roff.text(env); roff.control("RE", []); } } for pos in items.iter().filter(|a| a.is_positional()) { let mut header = vec![]; let (lhs, rhs) = option_markers(pos); header.push(roman(lhs)); if let Some(value) = pos.get_value_names() { header.push(italic(value.join(" "))); } else { header.push(italic(pos.get_id().as_str())); }; header.push(roman(rhs)); if let Some(defs) = option_default_values(pos) { header.push(roman(&format!(" {}", defs))); } let mut body = vec![]; if let Some(help) = pos.get_long_help().or_else(|| pos.get_help()) { body.push(roman(&help.to_string())); } roff.control("TP", []); roff.text(header); roff.text(body); if let Some(env) = option_environment(pos) { roff.control("RS", []); roff.text(env); roff.control("RE", []); } } } pub(crate) fn subcommands(roff: &mut Roff, cmd: &clap::Command, section: &str) { for sub in cmd.get_subcommands().filter(|s| !s.is_hide_set()) { roff.control("TP", []); let name = format!("{}-{}({})", cmd.get_name(), sub.get_name(), section); roff.text([roman(&name)]); if let Some(about) = sub.get_about().or_else(|| sub.get_long_about()) { for line in about.lines() { roff.text([roman(line)]); } } } } pub(crate) fn version(cmd: &clap::Command) -> String { format!( "v{}", cmd.get_long_version() .or_else(|| cmd.get_version()) .unwrap() ) } pub(crate) fn after_help(roff: &mut Roff, cmd: &clap::Command) { if let Some(about) = cmd.get_after_long_help().or_else(|| cmd.get_after_help()) { for line in about.lines() { roff.text([roman(line)]); } } } fn subcommand_markers(cmd: &clap::Command) -> (&'static str, &'static str) { markers(cmd.is_subcommand_required_set()) } fn option_markers(opt: &clap::Arg) -> (&'static str, &'static str) { markers(opt.is_required_set()) } fn markers(required: bool) -> (&'static str, &'static str) { if required { ("<", ">") } else { ("[", "]") } } fn short_option(opt: char) -> Inline { bold(&format!("-{}", opt)) } fn long_option(opt: &str) -> Inline { bold(&format!("--{}", opt)) } fn option_environment(opt: &clap::Arg) -> Option> { if opt.is_hide_env_set() { return None; } else if let Some(env) = opt.get_env() { return Some(vec![ roman("May also be specified with the "), bold(env.to_string_lossy().to_owned()), roman(" environment variable. "), ]); } None } fn option_default_values(opt: &clap::Arg) -> Option { if !opt.get_default_values().is_empty() { let values = opt .get_default_values() .iter() .map(|s| s.to_string_lossy()) .collect::>() .join(","); return Some(format!("[default: {}]", values)); } None } /// Generates a Vector of Inline Commands to push to the roff /// to properly format possible values that an option can take. fn format_possible_values(values: Vec<&clap::builder::PossibleValue>) -> Vec { let mut formats: Vec = vec![]; // With Help if values.iter().any(|p| p.get_help().is_some()) { formats.push(Inline::LineBreak); formats.push(roman("Possible values:")); formats.push(Inline::LineBreak); for value in values { formats.push(roman(" - ")); formats.push(roman(value.get_name().as_str())); match value.get_help() { Some(help) => { formats.push(roman(": ")); formats.push(roman(help.as_str())); } None => {} } formats.push(Inline::LineBreak); } } // Without help else { formats.push(Inline::LineBreak); formats.push(roman("[possible values: ")); formats.push(italic( values .iter() .map(|p| p.get_name().as_str()) .collect::>() .join(", "), )); formats.push(roman("]")); } formats }