mod action;
mod config;
mod prompt;
#[cfg(test)]
#[cfg(feature = "crossterm")]
mod test;
pub use action::*;
use std::fmt::Display;
use crate::{
config::get_configuration,
error::{InquireError, InquireResult},
formatter::OptionFormatter,
list_option::ListOption,
prompts::prompt::Prompt,
terminal::get_default_terminal,
type_aliases::{Scorer, Sorter},
ui::{Backend, RenderConfig, SelectBackend},
};
use self::prompt::SelectPrompt;
#[cfg(feature = "fuzzy")]
use fuzzy_matcher::{skim::SkimMatcherV2, FuzzyMatcher};
#[cfg(feature = "fuzzy")]
use std::sync::LazyLock;
#[cfg(feature = "fuzzy")]
static DEFAULT_MATCHER: LazyLock<SkimMatcherV2> =
LazyLock::new(|| SkimMatcherV2::default().ignore_case());
#[derive(Clone)]
pub struct Select<'a, T> {
pub message: &'a str,
pub options: Vec<T>,
pub help_message: Option<&'a str>,
pub page_size: usize,
pub vim_mode: bool,
pub starting_cursor: usize,
pub starting_filter_input: Option<&'a str>,
pub reset_cursor: bool,
pub filter_input_enabled: bool,
pub scorer: Scorer<'a, T>,
pub sorter: Sorter<'a>,
pub formatter: OptionFormatter<'a, T>,
pub render_config: RenderConfig<'a>,
}
impl<'a, T> Select<'a, T>
where
T: Display,
{
pub const DEFAULT_FORMATTER: OptionFormatter<'a, T> = &|ans| ans.to_string();
#[cfg(feature = "fuzzy")]
pub const DEFAULT_SCORER: Scorer<'a, T> =
&|input, _option, string_value, _idx| -> Option<i64> {
DEFAULT_MATCHER.fuzzy_match(string_value, input)
};
#[cfg(not(feature = "fuzzy"))]
pub const DEFAULT_SCORER: Scorer<'a, T> =
&|input, _option, string_value, _idx| -> Option<i64> {
let filter = input.to_lowercase();
match string_value.to_lowercase().contains(&filter) {
true => Some(0),
false => None,
}
};
pub const DEFAULT_SORTER: Sorter<'a> = &|options: &mut [(usize, i64)]| {
options.sort_unstable_by_key(|(_idx, score)| std::cmp::Reverse(*score));
};
pub const DEFAULT_PAGE_SIZE: usize = crate::config::DEFAULT_PAGE_SIZE;
pub const DEFAULT_VIM_MODE: bool = crate::config::DEFAULT_VIM_MODE;
pub const DEFAULT_STARTING_CURSOR: usize = 0;
pub const DEFAULT_RESET_CURSOR: bool = true;
pub const DEFAULT_FILTER_INPUT_ENABLED: bool = true;
pub const DEFAULT_HELP_MESSAGE: Option<&'a str> =
Some("↑↓ to move, enter to select, type to filter");
pub fn new(message: &'a str, options: Vec<T>) -> Self {
Self {
message,
options,
help_message: Self::DEFAULT_HELP_MESSAGE,
page_size: Self::DEFAULT_PAGE_SIZE,
vim_mode: Self::DEFAULT_VIM_MODE,
starting_cursor: Self::DEFAULT_STARTING_CURSOR,
reset_cursor: Self::DEFAULT_RESET_CURSOR,
filter_input_enabled: Self::DEFAULT_FILTER_INPUT_ENABLED,
scorer: Self::DEFAULT_SCORER,
sorter: Self::DEFAULT_SORTER,
formatter: Self::DEFAULT_FORMATTER,
render_config: get_configuration(),
starting_filter_input: None,
}
}
pub fn with_help_message(mut self, message: &'a str) -> Self {
self.help_message = Some(message);
self
}
pub fn without_help_message(mut self) -> Self {
self.help_message = None;
self
}
pub fn with_page_size(mut self, page_size: usize) -> Self {
self.page_size = page_size;
self
}
pub fn with_vim_mode(mut self, vim_mode: bool) -> Self {
self.vim_mode = vim_mode;
self
}
pub fn with_scorer(mut self, scorer: Scorer<'a, T>) -> Self {
self.scorer = scorer;
self
}
pub fn with_sorter(mut self, sorter: Sorter<'a>) -> Self {
self.sorter = sorter;
self
}
pub fn with_formatter(mut self, formatter: OptionFormatter<'a, T>) -> Self {
self.formatter = formatter;
self
}
pub fn with_starting_cursor(mut self, starting_cursor: usize) -> Self {
self.starting_cursor = starting_cursor;
self
}
pub fn with_starting_filter_input(mut self, starting_filter_input: &'a str) -> Self {
self.starting_filter_input = Some(starting_filter_input);
self
}
pub fn with_reset_cursor(mut self, reset_cursor: bool) -> Self {
self.reset_cursor = reset_cursor;
self
}
pub fn without_filtering(mut self) -> Self {
self.filter_input_enabled = false;
self
}
pub fn with_render_config(mut self, render_config: RenderConfig<'a>) -> Self {
self.render_config = render_config;
self
}
pub fn prompt(self) -> InquireResult<T> {
self.raw_prompt().map(|op| op.value)
}
pub fn prompt_skippable(self) -> InquireResult<Option<T>> {
match self.prompt() {
Ok(answer) => Ok(Some(answer)),
Err(InquireError::OperationCanceled) => Ok(None),
Err(err) => Err(err),
}
}
pub fn raw_prompt_skippable(self) -> InquireResult<Option<ListOption<T>>> {
match self.raw_prompt() {
Ok(answer) => Ok(Some(answer)),
Err(InquireError::OperationCanceled) => Ok(None),
Err(err) => Err(err),
}
}
pub fn raw_prompt(self) -> InquireResult<ListOption<T>> {
let (input_reader, terminal) = get_default_terminal()?;
let mut backend = Backend::new(input_reader, terminal, self.render_config)?;
self.prompt_with_backend(&mut backend)
}
pub(crate) fn prompt_with_backend<B: SelectBackend>(
self,
backend: &mut B,
) -> InquireResult<ListOption<T>> {
SelectPrompt::new(self)?.prompt(backend)
}
}