|
|
|
@ -1,8 +1,8 @@
|
|
|
|
|
use anyhow::Result;
|
|
|
|
|
use std::collections::VecDeque;
|
|
|
|
|
use std::env;
|
|
|
|
|
use std::io::{BufRead, BufReader};
|
|
|
|
|
use std::path::PathBuf;
|
|
|
|
|
use std::{env, io};
|
|
|
|
|
|
|
|
|
|
use crate::app;
|
|
|
|
|
|
|
|
|
@ -12,100 +12,93 @@ pub struct Cli {
|
|
|
|
|
pub version: bool,
|
|
|
|
|
pub help: bool,
|
|
|
|
|
pub read_only: bool,
|
|
|
|
|
pub path: Option<PathBuf>,
|
|
|
|
|
pub force_focus: bool,
|
|
|
|
|
pub config: Option<PathBuf>,
|
|
|
|
|
pub extra_config: Vec<PathBuf>,
|
|
|
|
|
pub on_load: Vec<app::ExternalMsg>,
|
|
|
|
|
pub select: Vec<PathBuf>,
|
|
|
|
|
pub paths: Vec<PathBuf>,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Cli {
|
|
|
|
|
fn read_path(&mut self, arg: &str) {
|
|
|
|
|
if arg.is_empty() {
|
|
|
|
|
return;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let path = PathBuf::from(arg);
|
|
|
|
|
if path.exists() {
|
|
|
|
|
self.paths.push(path);
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// Parse arguments from the command-line
|
|
|
|
|
pub fn parse(args: env::Args) -> Result<Self> {
|
|
|
|
|
let mut args: VecDeque<String> = args.skip(1).collect();
|
|
|
|
|
let mut cli = Self::default();
|
|
|
|
|
|
|
|
|
|
let mut flag_ends = false;
|
|
|
|
|
|
|
|
|
|
while let Some(arg) = args.pop_front() {
|
|
|
|
|
match arg.as_str() {
|
|
|
|
|
// Flags
|
|
|
|
|
"-" => {
|
|
|
|
|
let mut path = String::new();
|
|
|
|
|
if io::stdin().read_line(&mut path).is_ok() {
|
|
|
|
|
cli.path =
|
|
|
|
|
Some(path.trim_end_matches("\r\n").trim_end_matches('\n').into());
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
if flag_ends {
|
|
|
|
|
cli.read_path(&arg);
|
|
|
|
|
} else {
|
|
|
|
|
match arg.as_str() {
|
|
|
|
|
// Flags
|
|
|
|
|
"-" => {
|
|
|
|
|
for path in BufReader::new(std::io::stdin())
|
|
|
|
|
.lines()
|
|
|
|
|
.filter_map(|l| l.ok())
|
|
|
|
|
{
|
|
|
|
|
cli.read_path(&path);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
"-h" | "--help" => {
|
|
|
|
|
cli.help = true;
|
|
|
|
|
}
|
|
|
|
|
"-h" | "--help" => {
|
|
|
|
|
cli.help = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
"-V" | "--version" => {
|
|
|
|
|
cli.version = true;
|
|
|
|
|
}
|
|
|
|
|
"-V" | "--version" => {
|
|
|
|
|
cli.version = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
"--" => {
|
|
|
|
|
if cli.path.is_none() {
|
|
|
|
|
cli.path = args.pop_front().map(PathBuf::from);
|
|
|
|
|
"--" => {
|
|
|
|
|
flag_ends = true;
|
|
|
|
|
}
|
|
|
|
|
return Ok(cli);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Options
|
|
|
|
|
"-c" | "--config" => cli.config = args.pop_front().map(PathBuf::from),
|
|
|
|
|
// Options
|
|
|
|
|
"-c" | "--config" => cli.config = args.pop_front().map(PathBuf::from),
|
|
|
|
|
|
|
|
|
|
"-C" | "--extra-config" => {
|
|
|
|
|
while let Some(path) = args.pop_front() {
|
|
|
|
|
if path.starts_with('-') {
|
|
|
|
|
args.push_front(path);
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
cli.extra_config.push(PathBuf::from(path));
|
|
|
|
|
"-C" | "--extra-config" => {
|
|
|
|
|
while let Some(path) = args.pop_front() {
|
|
|
|
|
if path.starts_with('-') {
|
|
|
|
|
args.push_front(path);
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
cli.extra_config.push(PathBuf::from(path));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
"--read-only" => cli.read_only = true,
|
|
|
|
|
"--read-only" => cli.read_only = true,
|
|
|
|
|
|
|
|
|
|
"--on-load" => {
|
|
|
|
|
while let Some(msg) = args.pop_front() {
|
|
|
|
|
if msg.starts_with('-') {
|
|
|
|
|
args.push_front(msg);
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
cli.on_load.push(serde_yaml::from_str(&msg)?);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
"--select" => {
|
|
|
|
|
while let Some(path) = args.pop_front() {
|
|
|
|
|
if path.starts_with('-') && path != "-" {
|
|
|
|
|
for path in BufReader::new(std::io::stdin()).lines() {
|
|
|
|
|
cli.select.push(PathBuf::from(path?));
|
|
|
|
|
"--on-load" => {
|
|
|
|
|
while let Some(msg) = args.pop_front() {
|
|
|
|
|
if msg.starts_with('-') {
|
|
|
|
|
args.push_front(msg);
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
cli.on_load.push(serde_yaml::from_str(&msg)?);
|
|
|
|
|
}
|
|
|
|
|
break;
|
|
|
|
|
} else {
|
|
|
|
|
cli.select.push(PathBuf::from(path));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
"--force-focus" => {
|
|
|
|
|
let path = cli
|
|
|
|
|
.path
|
|
|
|
|
.as_ref()
|
|
|
|
|
.cloned()
|
|
|
|
|
.or_else(|| args.pop_front().map(PathBuf::from))
|
|
|
|
|
.unwrap_or_default();
|
|
|
|
|
cli.on_load.push(app::ExternalMsg::FocusPath(
|
|
|
|
|
path.to_string_lossy().to_string(),
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
"--force-focus" => {
|
|
|
|
|
cli.force_focus = true;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// path
|
|
|
|
|
path => {
|
|
|
|
|
if cli.path.is_none() {
|
|
|
|
|
cli.path = Some(path.into());
|
|
|
|
|
// path
|
|
|
|
|
path => {
|
|
|
|
|
cli.read_path(path);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|