Fast designing menus for your Rust CLI programs.
This crate provides a derive(Menu)
procedural macro to easily build menus.
It includes type-checking from the user input, and a formatting customization.
You can find all the crate documentation on Docs.rs. You can also check the examples to learn with a practical way.
Here is an example of how to use it:
```rust use std::io::Write; use ezmenu::{Menu, MenuResult};
fn map(val: u32, w: &mut impl Write) -> MenuResult
struct MyMenu { author: String, #[field(msg = "Give a number", then(map))] number: u32, } ```
To display the menu, you instantiate the struct by calling its from_menu
method:
rust
let MyMenu { author, number } = MyMenu::from_menu();
println!("values provided: author={}, number={}", author, number);
This sample code prints the standard menu like above:
Hello there!
* author: Ahmad
* Give a number: 1000
ok!
values provided: author=Ahmad, number=1000
You can apply several formatting rules on a menu and on a field specifically.
You can edit:
* the chip: *
by default.
* the prefix: :
by default.
* insert a new line before prefix and user input: false
by default.
* display default values or not: true
by default.
For a custom format on a field and a main formatting rule on a menu, you can build this with: ```rust
struct License { #[menu(chip = "- ")] author: String, date: u16, } ```
The custom >>
will be applied on every field except those with custom formatting rules.
In this case, it will format the text like above:
``` - author: ...
date: ... ```
You can provide default values to a field like above:
```rust
struct License { author: String, #[menu(default = 2022)] date: u16, } ```
If the user provided an incorrect input, the program will not re-ask a value to the user, but will directly return the default value instead.
By default, the default value is visible. If you want to hide it, you can do so: ```rust
``` on the struct or on a field.
If you are not using std::io::Stdin
and std::io::Stdout
types, you can provide your own
types by enabling the custom_io
feature in your Cargo.toml file:
toml
[dependencies]
ezmenu = { version = "0.2.3", features = ["custom_io"] }
Then you can instantiate your struct with:
rust
use std::io::stdout;
let input = b"Ahmad\n1000\n" as &[u8];
let values = MyMenu::from_io(input, stdout());
If the user has to provide a value which corresponds to your specific type,
you can use the ezmenu::parsed
on this type.
For example, in the case of a mk-license program, the menu can be built like above:
```rust
enum Type { MIT, BSD, GPL, }
struct License { author: String, date: u16, #[menu(default = "mit")] ty: Type, } ```
This will restrict the user to enter "MIT", "BSD" or "GPL" inputs ignoring the case.
The derive(Menu)
is available with the derive
feature, enabled by default.
You can disable it in your Cargo.toml file:
toml
[dependencies]
ezmenu = { version = "0.2.3", default-features = false }
You can still use the provided library to build your menus.
To ask a simple value, you can use StructField::build
method by giving the Stdin
and Stdout
types.
rust
use std::io::{stdin, stdout};
use ezmenu::StructField;
let age: u8 = StructField::from("How old are you?")
.build(&stdin(), &mut stdout()).unwrap();
If you want to build a menu with all the previous features (default values, formatting rules...), you can refer to this code below: ```rust use ezmenu::{StructField, StructFieldFormatting}; let mut menu = StructMenu::default() .title("-- Mklicense --") .fmt(StructFieldFormatting { chip: "* Give the ", ..Default::default() }) .withfield(StructField::from("project author name")) .withfield(StructField::from("project name")) .withfield(StructField::from("Give the year of the license") .default("2022") .fmt(StructFieldFormatting { prefix: ">> ", newline: true, ..Default::default() }) );
let name: String = menu.nextmap(|s: String, w| { if s.tolowercase() == "ahmad" { w.write(b"Nice name!!")?; } Ok(s) }).unwrap(); let projname: String = menu.next().unwrap(); let projyear: i64 = menu.next().unwrap(); ```
This project is still in development. You can check the EZMenu project to see all the next features.