# Backer
data:image/s3,"s3://crabby-images/6bbb1/6bbb1f572fbe3409fdc3e385353b1e8482014bd4" alt="rust"
[data:image/s3,"s3://crabby-images/beb1f/beb1fb81ed5be1114cf9348baf1d76c33fbbf907" alt="crates.io"](https://crates.io/crates/backer)
[data:image/s3,"s3://crabby-images/c495b/c495b5565330142c59df415d474e2b77830bf18e" alt="downloads"](https://crates.io/crates/backer)
[data:image/s3,"s3://crabby-images/bf2b8/bf2b8979c37e65f92118a5a6601013f0cbf10890" alt="license"](https://github.com/ejjonny/backer/blob/main/LICENSE)
A library for straight-forward UI layout.
Dependency free & framework-agnostic. Backer can be used in an index-based layout approach or with inline drawing code.
_This library **only** implements layout & would be most useful along with a GUI library that can do GUI things (like [macroquad](https://github.com/not-fl3/macroquad) or [egui](https://github.com/emilk/egui))._
### Features ✨
- Declarative API: The code should look like the structure it defines
- Minimal interface: No confusing overloads or magic, cascading effects
- Intuitive constraints: Backer adheres to & smoothly resolves size constraints with an advanced algorithm
- Performant: Layout can be comfortably recalculated every frame
- Easy integration & compatibility: Backer should work with just about any UI library with a bit of glue - so it isn't really fine-tuned for any specific UI solution.
This project intends to be a flexible layout tool & not much else.
## Preview
Check out the [demo site](https://ejjonny.github.io/backer/): a mock page showcasing layout capabilities in a realistic interface. Built with [egui](https://github.com/emilk/egui)!
[
](https://ejjonny.github.io/backer/)
Backer relies on simple rules that can compose to create complex, flexible layouts.
data:image/s3,"s3://crabby-images/3386b/3386b11508dfd5a68769e0e22a18876a98a68685" alt="stretched"
See some code
```rust
column_spaced(
10.,
vec![
draw_a(ui),
row_spaced(
10.,
vec![
draw_b(ui).width(180.).align(Align::Leading),
column_spaced(10., vec![draw_a(ui), draw_b(ui), draw_c(ui)]),
],
),
draw_c(ui),
],
)
.pad(10.)
```
# Quick Start
## 1. Create a `Layout` struct with your layout function.
```rust
use backer::layout::Layout;
use backer::layout::Node;
let layout = Layout::new(my_layout_fn);
fn my_layout_fn(state: &mut MyState) -> Node { todo!() }
```
## 2. Implement a `draw` node
For reuse, you can construct your drawable in a function
```rust
fn my_drawable(state: &mut MyState) -> Node {
draw(move |area: Area, state: &mut MyState| {
// The `area` parameter is the space alotted for your view after layout is calculated
// The `state` parameter is *your* mutable state that you pass when you call layout.
// This closure should draw UI based on the alotted area or update state so that drawing can be performed later.
})
}
```
## 3. Combine nodes to define & customize your layout
```rust
fn my_layout_fn(state: &mut MyState) -> Node {
row(vec![
my_drawable(state)
])
}
```
## 4. Draw your layout
```rust
// UI libraries generally will expose methods to get the available screen size
// In a real implementation this should use the real screen size!
let available_area = Area {
x: todo!(),
y: todo!(),
width: todo!(),
height: todo!().
};
let mut my_state = MyState::new();
let layout = Layout::new(my_layout_fn);
// Perform layout & draw all of your drawable nodes.
layout.draw(available_area, &mut my_state);
```
## Status
The crate is currently usable but new! Breaking changes may be relatively frequent as the crate matures.
Contributions are always welcome 🤗