egui-controls
is a Rust library that provides a ControlPanel
proc-macro to generate a simple control panel interface using the egui immediate mode graphical user interface library.
You're implementing a Rust algorithm that has various tunable parameters, and would like to inspect the output by tweaking the parameters in real-time.
sh
cargo add egui-controls
Suppose your typical config data that contains the parameters for the algorithm looks like this:
```rust
pub struct CirclePackingAlgorithmConfig { /// The radius of the circles to pack. pub radius: f64, /// If circles overlap, then how many should be allowed /// to overlap at most. pub maxoverlapcount: usize, /// Once we find the circles, label them with the /// given name. pub circlelabel: String, /// Some global constant that should definitely only take on this value. pub nonchangingglobalvalue: i8 }
/// Some initial values for the config that make sense. impl Default for CirclePackingAlgorithmConfig { fn default() -> Self { Self { radius: 12.0, maxoverlapcount: 10, circlelabel: "Some text".tostring(), nonchangingglobal_value: 42 } } } ```
Now, just derive eguicontrols::ControlPanel for your data, and
sprinkle in some #[control]
attributes on the fields you'd like to be interactive in the UI:
```diff
+ use eguicontrols::ControlPanel;
Now, use `config.ui(ui)` to embed that in any UI section you're building with `eframe::egui`.
rust
use eframe::{egui, Frame};pub struct MyApp { settings: CirclePackingAlgorithmConfig }
impl eframe::App for MyApp { fn update(&mut self, ctx: &egui::Context, _frame: &mut Frame) { egui::CentralPanel::default().show(ctx, |ui: &mut egui::Ui| { // Embed the settings panel // directly into your ui. self.settings.ui(ui); // Add this the struct's debug repr if you want // to see the values getting updated as you tweak // the settings via the ui. ui.vertical(|ui| { ui.code(format!("{:#?}", &self.settings)); }); }); } }
// Write the usual eframe entrypoint. pub fn main() { let options = ::eframe::NativeOptions { resizable: true, initialwindowsize: Some(::eframe::egui::vec2(2000.0, 500.0)), ..Default::default() }; let app = MyApp::default(); ::eframe::runnative("readme", options, Box::new(|| Box::new(app))).unwrap(); } ```
This generates a simple control panel interface for the CirclePackingAlgorithmConfig
and we can tune the values via the sliders.
Note: You can run the readme
example to generate the same output.