mirror of
https://github.com/alexpasmantier/television.git
synced 2025-07-28 13:51:41 +00:00

Adds a `--select-1` flag that enables tv to automatically select the only result and exit when such a case appears.
259 lines
7.4 KiB
Rust
259 lines
7.4 KiB
Rust
use rustc_hash::FxHashMap;
|
|
use std::{
|
|
fmt::{self, Display, Formatter},
|
|
ops::Deref,
|
|
};
|
|
|
|
use std::collections::HashSet;
|
|
use std::io::{BufRead, BufReader};
|
|
use std::process::Stdio;
|
|
|
|
use anyhow::Result;
|
|
use regex::Regex;
|
|
use rustc_hash::{FxBuildHasher, FxHashSet};
|
|
use tracing::debug;
|
|
|
|
use crate::channels::entry::{Entry, PreviewCommand, PreviewType};
|
|
use crate::channels::OnAir;
|
|
use crate::matcher::Matcher;
|
|
use crate::matcher::{config::Config, injector::Injector};
|
|
use crate::utils::command::shell_command;
|
|
|
|
#[derive(Debug, Clone, PartialEq)]
|
|
pub enum PreviewKind {
|
|
Command(PreviewCommand),
|
|
Builtin(PreviewType),
|
|
None,
|
|
}
|
|
|
|
#[allow(dead_code)]
|
|
pub struct Channel {
|
|
pub name: String,
|
|
matcher: Matcher<String>,
|
|
entries_command: String,
|
|
preview_kind: PreviewKind,
|
|
selected_entries: FxHashSet<Entry>,
|
|
crawl_handle: tokio::task::JoinHandle<()>,
|
|
}
|
|
|
|
impl Default for Channel {
|
|
fn default() -> Self {
|
|
Self::new(
|
|
"Files",
|
|
"find . -type f",
|
|
Some(PreviewCommand::new("bat -n --color=always {}", ":")),
|
|
)
|
|
}
|
|
}
|
|
|
|
impl From<CableChannelPrototype> for Channel {
|
|
fn from(prototype: CableChannelPrototype) -> Self {
|
|
Self::new(
|
|
&prototype.name,
|
|
&prototype.source_command,
|
|
match prototype.preview_command {
|
|
Some(command) => Some(PreviewCommand::new(
|
|
&command,
|
|
&prototype
|
|
.preview_delimiter
|
|
.unwrap_or(DEFAULT_DELIMITER.to_string()),
|
|
)),
|
|
None => None,
|
|
},
|
|
)
|
|
}
|
|
}
|
|
|
|
pub fn parse_preview_kind(command: &PreviewCommand) -> Result<PreviewKind> {
|
|
debug!("Parsing preview kind for command: {:?}", command);
|
|
let re = Regex::new(r"^\:(\w+)\:$").unwrap();
|
|
if let Some(captures) = re.captures(&command.command) {
|
|
let preview_type = PreviewType::try_from(&captures[1])?;
|
|
Ok(PreviewKind::Builtin(preview_type))
|
|
} else {
|
|
Ok(PreviewKind::Command(command.clone()))
|
|
}
|
|
}
|
|
|
|
impl Channel {
|
|
pub fn new(
|
|
name: &str,
|
|
entries_command: &str,
|
|
preview_command: Option<PreviewCommand>,
|
|
) -> Self {
|
|
let matcher = Matcher::new(Config::default());
|
|
let injector = matcher.injector();
|
|
let crawl_handle = tokio::spawn(load_candidates(
|
|
entries_command.to_string(),
|
|
injector,
|
|
));
|
|
let preview_kind = match preview_command {
|
|
Some(command) => {
|
|
parse_preview_kind(&command).unwrap_or_else(|_| {
|
|
panic!("Invalid preview command: {command}")
|
|
})
|
|
}
|
|
None => PreviewKind::None,
|
|
};
|
|
debug!("Preview kind: {:?}", preview_kind);
|
|
Self {
|
|
matcher,
|
|
entries_command: entries_command.to_string(),
|
|
preview_kind,
|
|
name: name.to_string(),
|
|
selected_entries: HashSet::with_hasher(FxBuildHasher),
|
|
crawl_handle,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[allow(clippy::unused_async)]
|
|
async fn load_candidates(command: String, injector: Injector<String>) {
|
|
debug!("Loading candidates from command: {:?}", command);
|
|
let mut child = shell_command()
|
|
.arg(command)
|
|
.stdout(Stdio::piped())
|
|
.stderr(Stdio::piped())
|
|
.spawn()
|
|
.expect("failed to execute process");
|
|
|
|
if let Some(out) = child.stdout.take() {
|
|
let reader = BufReader::new(out);
|
|
let mut produced_output = false;
|
|
|
|
#[allow(clippy::manual_flatten)]
|
|
for line in reader.lines() {
|
|
if let Ok(l) = line {
|
|
if !l.trim().is_empty() {
|
|
let () = injector.push(l, |e, cols| {
|
|
cols[0] = e.clone().into();
|
|
});
|
|
produced_output = true;
|
|
}
|
|
}
|
|
}
|
|
|
|
if !produced_output {
|
|
let reader = BufReader::new(child.stderr.take().unwrap());
|
|
for line in reader.lines() {
|
|
let line = line.unwrap();
|
|
if !line.trim().is_empty() {
|
|
let () = injector.push(line, |e, cols| {
|
|
cols[0] = e.clone().into();
|
|
});
|
|
}
|
|
}
|
|
}
|
|
}
|
|
let _ = child.wait();
|
|
}
|
|
|
|
impl OnAir for Channel {
|
|
fn find(&mut self, pattern: &str) {
|
|
self.matcher.find(pattern);
|
|
}
|
|
|
|
fn results(&mut self, num_entries: u32, offset: u32) -> Vec<Entry> {
|
|
self.matcher.tick();
|
|
self.matcher
|
|
.results(num_entries, offset)
|
|
.into_iter()
|
|
.map(|item| {
|
|
let path = item.matched_string;
|
|
Entry::new(
|
|
path,
|
|
match &self.preview_kind {
|
|
PreviewKind::Command(ref preview_command) => {
|
|
PreviewType::Command(preview_command.clone())
|
|
}
|
|
PreviewKind::Builtin(preview_type) => {
|
|
preview_type.clone()
|
|
}
|
|
PreviewKind::None => PreviewType::None,
|
|
},
|
|
)
|
|
.with_name_match_indices(&item.match_indices)
|
|
})
|
|
.collect()
|
|
}
|
|
|
|
fn get_result(&self, index: u32) -> Option<Entry> {
|
|
self.matcher.get_result(index).map(|item| {
|
|
let path = item.matched_string;
|
|
Entry::new(
|
|
path,
|
|
match &self.preview_kind {
|
|
PreviewKind::Command(ref preview_command) => {
|
|
PreviewType::Command(preview_command.clone())
|
|
}
|
|
PreviewKind::Builtin(preview_type) => preview_type.clone(),
|
|
PreviewKind::None => PreviewType::None,
|
|
},
|
|
)
|
|
})
|
|
}
|
|
|
|
fn selected_entries(&self) -> &FxHashSet<Entry> {
|
|
&self.selected_entries
|
|
}
|
|
|
|
fn toggle_selection(&mut self, entry: &Entry) {
|
|
if self.selected_entries.contains(entry) {
|
|
self.selected_entries.remove(entry);
|
|
} else {
|
|
self.selected_entries.insert(entry.clone());
|
|
}
|
|
}
|
|
|
|
fn result_count(&self) -> u32 {
|
|
self.matcher.matched_item_count
|
|
}
|
|
|
|
fn total_count(&self) -> u32 {
|
|
self.matcher.total_item_count
|
|
}
|
|
|
|
fn running(&self) -> bool {
|
|
self.matcher.status.running || !self.crawl_handle.is_finished()
|
|
}
|
|
|
|
fn shutdown(&self) {}
|
|
|
|
fn supports_preview(&self) -> bool {
|
|
self.preview_kind != PreviewKind::None
|
|
}
|
|
}
|
|
|
|
#[derive(Clone, Debug, serde::Deserialize, PartialEq)]
|
|
pub struct CableChannelPrototype {
|
|
pub name: String,
|
|
pub source_command: String,
|
|
pub preview_command: Option<String>,
|
|
#[serde(default = "default_delimiter")]
|
|
pub preview_delimiter: Option<String>,
|
|
}
|
|
|
|
pub const DEFAULT_DELIMITER: &str = " ";
|
|
|
|
#[allow(clippy::unnecessary_wraps)]
|
|
fn default_delimiter() -> Option<String> {
|
|
Some(DEFAULT_DELIMITER.to_string())
|
|
}
|
|
|
|
impl Display for CableChannelPrototype {
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
|
write!(f, "{}", self.name)
|
|
}
|
|
}
|
|
|
|
#[derive(Debug, serde::Deserialize, Default)]
|
|
pub struct CableChannels(pub FxHashMap<String, CableChannelPrototype>);
|
|
|
|
impl Deref for CableChannels {
|
|
type Target = FxHashMap<String, CableChannelPrototype>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
&self.0
|
|
}
|
|
}
|