11 releases (5 breaking)

new 0.6.3 Dec 11, 2024
0.6.2 Dec 11, 2024
0.5.0 Nov 7, 2024
0.4.1 Sep 7, 2024
0.1.0 Aug 20, 2024

#166 in Command-line interface

Download history 403/week @ 2024-08-19 226/week @ 2024-08-26 205/week @ 2024-09-02 1/week @ 2024-09-09 35/week @ 2024-09-16 17/week @ 2024-09-23 9/week @ 2024-09-30 12/week @ 2024-10-07 5/week @ 2024-10-14 121/week @ 2024-11-04 9/week @ 2024-11-11 27/week @ 2024-11-18 63/week @ 2024-11-25 223/week @ 2024-12-02

333 downloads per month
Used in autobib

MIT/Apache

145KB
3K SLoC

Current crates.io release Documentation

nucleo-picker

Yet another fuzzy picker library. This library provides a TUI for the nucleo crate. The picker interface is similar to the very popular fzf command-line tool, but rather than act as a standalone binary, this provides a Rust library which allows you to incorporate a picker interface into your own application.

See the examples directory for implementation examples, or try out the sample find/fzf implementation by cloning the repository and running cargo run --release --example find ~.

If you are looking for documentation for interactive usage of the picker, see the USAGE.md file.

If you are looking for a list of recent changes, see the CHANGELOG.md file.

Features

  • Highly optimized matching.
  • Robust rendering:
    • Full Unicode handling with Unicode text segmentation and Unicode width.
    • Match highlighting with automatic scroll-through.
    • Correctly render multi-line or overflowed items.
    • Responsive interface with batched keyboard input.
  • Ergonomic API:
    • Fully concurrent lock- and wait-free streaming of input items.
    • Generic Picker for any type T which is Send + Sync + 'static.
    • Customizable rendering of crate-local and foreign types with the Render trait.

Example

Implement a heavily simplified fzf clone in 30 lines of code. Try it out with:

cargo build --release --example fzf
cat myfile.txt | ./target/release/examples/fzf

The code to create the binary:

use std::{
    io::{self, BufRead},
    process::exit,
    thread::spawn,
};

use nucleo_picker::{render::StrRenderer, Picker};

fn main() -> io::Result<()> {
    let mut picker = Picker::new(StrRenderer);

    let injector = picker.injector();
    spawn(move || {
        for line in io::stdin().lock().lines() {
            match line {
                Ok(s) => injector.push(s),
                Err(_) => {}
            }
        }
    });

    match picker.pick()? {
        Some(it) => println!("{it}"),
        None => exit(1),
    }
    Ok(())
}

This crate mainly exists as a result of the author's annoyance with pretty much every fuzzy picker TUI in the rust ecosystem.

  • skim's Arc<dyn SkimItem> is inconvenient for a variety of reasons. skim also has a large number of dependencies and is designed more as a binary than a library.
  • fuzzypicker is based on skim and inherits skim's problems.
  • nucleo-ui only has a blocking API and only supports matching on String. It also seems to be un-maintained.
  • fuzzy-select only has a blocking API.
  • dialoguer FuzzySelect only has a blocking API and only supports matching on String. The terminal handling also has a few strange bugs.

Disclaimer

The feature set of this library is quite minimal (by design) but may be expanded in the future. There are a currently a few known problems which have not been addressed (see the issues page on GitHub for a list).

Dependencies

~6–15MB
~205K SLoC