Bevy tracking Latest version MIT/Apache 2.0 Documentation

Cuicui Layout

A dumb layout algorithm you can rely on, built for and with bevy.

The Cyberpunk 2077 showcase

For some reasons, the Cyberpunk main menu has become the 7GUI of bevy, so here is the Cyberpunk main menu using cuicui_layout_bevy_ui.

https://github.com/nicopap/cuicui_layout/assets/26321040/8a51f9a9-ffa7-4b60-a2ad-3947ff718e27.mp4

Code

```rust use bevy::prelude::*; use cuicuilayoutbevyui::UiDsl as Dsl; use cuicuilayout::{LayoutRootCamera, dsl, dsl_functions::{px, pct}};

fn setup(mut cmds: Commands, serv: Res) {

cmds.spawn((Camera2dBundle::default(), LayoutRootCamera)); let menubuttons = [ "CONTINUE", "NEW GAME", "LOAD GAME", "SETTINGS", "ADDITIONAL CONTENT", "CREDITS", "QUIT GAME", ]; let titlecard = serv.load::("logo.png"); let bg = serv.load("background.png"); let board = serv.load("board.png"); let button = serv.load("button.png");

dsl! { &mut cmds, row(screenroot, "root", mainmargin 100., distribstart, alignstart, image &bg) { column("menu", width px(310), height pct(100), mainmargin 40., image &board) { spawn(image &titlecard, "Title card", width pct(100)); spawnui(titlecard, "Title card 2", width pct(50)); code(let cmds) { for n in &menubuttons { let name = format!("{n} button"); dsl!(cmds, spawnui(*n, named name, image &button, height px(33));); } } } } };

} ```

Running examples

Use the cargo run --bin command to list possible examples, and run them.

We do this because it allows us to have different dependencies between examples.

Stability

This crate is in expansion, use at your own risk, it is extremely likely that a lot of things are going to break a lot.

Using cuicui_layout

First, chose which crate you want to use:

Secondly, add your chosen integration crate to your Cargo.toml:

toml [dependencies] cuicui_layout_bevy_ui = "0.7.0" cuicui_layout = "0.7.1"

Then, use cuicui_layout in your crate with the [dsl!] macro:

```rust,norun use bevy::prelude::*; use cuicuilayout::{dsl, LayoutRootCamera, dslfunctions::*}; use cuicuilayoutbevyui::UiDsl as Dsl;

fn main() { // Do not forget to add cuicuilayoutbevy{ui,sprite}::Plugin App::new().addplugins((DefaultPlugins, cuicuilayoutbevyui::Plugin)) .addsystems(Startup, setup) .run(); } fn setup(mut commands: Commands) { // Use LayoutRootCamera to mark a camera as the screen boundaries. commands.spawn((Camera2dBundle::default(), LayoutRootCamera)); dsl! { &mut commands, // Use screenroot to follow the screen's boundaries row(screenroot) { row(margin 9., border(5, Color::CYAN), bg Color::NAVY) { spawn_ui("Hello world!"); } } }; } ```

That's it! You are now using cuicui_layout, congratulations!

Make sure to check the [LayoutDsl] docs to learn the current capabilities of cuicui_layout.

cuicui_layout crates

This repository contains several crates:

Cargo features

Why cuicui layout

Why not Flexbox

You are writing text to get 2d visual results on screen. The translation from text to screen should be trivial, easy to do in your head. Otherwise you need visual feedback to get what you want. Bevy, even with hot reloading or [bevy-inspector-egui] will always have extremely slow visual feedback.

Flexbox has too many parameters and depends on implicit properties of UI elements, it is not possible to emulate it in your head.

cuicui's layout, in contrast to Flexbox is easy to fit in your head. In fact, I will forecefully push cuicui's layout algorithm in your head in two short bullet points.

That's it. There are some edge cases, but cuicui will ~~yell at you~~ tell you nicely when you hit them and tell you how to handle them properly.

Flexbox FAQ

Q: Where is padding?
A: padding is equivalent to margin in cuicui_layout. margin and border doesn't make conceptual sense.

Q: Why not call it padding then?
A: Look at the dictionary definition of "margin" and "padding".

Q: How do I center a node?
A: nodes are centered by default, make sure the parent's container size has the expected size.

Q: What is the equivalent of flex_direction?
A: use row and column

Q: What are the equivalents of column-reverse and row-reverse?
A: None. Use Alignment::End and swap your elements! Note that the *-reverse flows in flexbox are very useful for internationalization. However, when making a game, it is not enough to just swap the elements! Artistic control is paramount and internationalization needs to be taken as a whole in the context of the UI.

Q: What is the equivalent of flex_wrap?
A: None, do you really need it?

Q: What is the equivalent of align_item, align_self, align_content, justify_content?
A: After 5 years of working with CSS, I still have no clue which one does what, and whether they really do anything, so I wont' adventure an asnwer.

Q: What is the equivalent of flex_grow, flex_shrink, flex_basis, gap?
A: Do you even know what they do?

Q: Why can't child container overflow their parents?
A: It's likely you didn't expect this, so we report it as an error.

Q: How do I make a grid?
A: cuicui_layout is currently not capable of managing a grid of nodes. This might be added in the future.

Why not add \cuicui_layout?

Each flexbox feature is useful taken in isolation, but when combined, they make for a very difficult to grasp whole. It's the combinatorial explosion of interactions between features that makes Flexbox impossible to emulate in your head. In fact, I'm not so sure anything short of a Flexbox implementation can predict what the final output of your CSS will look like.

With this settled, it is natural that I aim to make cuicui_layout as featureless as possible. Ideally, there is exactly one way to do anything, even if it requires a bit of head scratching to get there. Code with less feature is paradoxically smarter. A narrow set of functionalities allow easier inference on the user's expectations, enabling better error messages and suggestions.

Of course, as a library, cuicui_layout must at least have some features. Here is what I look in a new feature:

Here is an example: margin. At first, I didn't even want margins. After all, I can nest a container within another one with padding empty nodes. Right? Well no.

Say you have the following layout:

text dsl! { row(rules(pct(100), pct(100))) { row(margins 10, rules(pct(100), pct(100))) { some_element(); } } }

We can't use padding nodes here. Because the inner row depends on the size of the parent. Adding nodes would make the inner row always overflow the outer row, because it's size will still be 100% that of its parent, in addition to the two 10 pixel empty padding nodes.

The solution to this is to have a distinction between "outer" and "inner" sizes:

The margin now allows specifying the inner row as 100% the "size" of the outer row. In fact it's specifying the size relative to the outer row's size minus the given margin.

Now why limit margin to pixel specification, eschewing percent-based rules?

Paradoxically, it's for usability: It is not clear what the percent is a percentage of. Is it the inner size? The parent's size? children size? The full size after the application of the margin?

We don't know, different people will have different expectations. This avoids any confusion. In any case, this is a situation where empty nodes can be used, since you'll be able to compute the relative size of each node yourself.

Version matrix

| bevy | latest supporting version | |------|-------| | 0.11 | 0.7.1 | | 0.10 | 0.3.0 |

License

Licensed under either of

Contribution

Unless you explicitly state otherwise, any contribution intentionally submitted for inclusion in the work by you, as defined in the Apache-2.0 license, shall be dual licensed as above, without any additional terms or conditions.