declarative

Matrix REUSE status

A proc-macro library for creating complex reactive views declaratively and quickly.

To use it, add to your Cargo.toml:

~~~ toml [dependencies.declarative] version = '0.3'

for a custom builder mode:

features = ['builder-mode']

if you're going to use it with gtk-rs, you might want to:

features = ['gtk-rs'] # gives a suitable builder_mode! macro ~~~

To learn how to use the macros, it is best to clone the repository, read the source code of the examples in alphabetical order and run them like this:

~~~ bash cargo run --features gtk-rs --example EXAMPLE_NAME ~~~

The examples depend on [gtk-rs], so you should familiarize yourself with [gtk-rs] a bit before:
https://gtk-rs.org/gtk4-rs/stable/latest/book/

You may need to tell rust-analyzer that the examples depend on the gtk-rs feature to avoid false positives. For example, with VS Code it is configured with the following JSON:

~~~ JSON { "rust-analyzer.cargo.features": ["gtk-rs"] } ~~~

Counter application example

In the following I manually implement the Elm pattern. The macro does not require any specific pattern.

Light theme app screenshot Dark theme app screenshot

~~~ rust use declarative::{builder_mode, clone, view}; use gtk::{glib, prelude::*};

macro_rules! send { [$msg:expr => $tx:expr] => [$tx.send($msg).unwrap()] }

enum Msg { Increase, Decrease }

[view {

gtk::ApplicationWindow window !{
    application: app
    title: "My Application"

    gtk::HeaderBar #titlebar(&#) { }

    gtk::Box #child(&#) !{
        orientation: gtk::Orientation::Vertical
        spacing: 6
        margin_top: 6
        margin_bottom: 6
        margin_start: 6
        margin_end: 6 #:

        gtk::Label #append(&#) {
            'bind! set_label: &format!("The count is: {count}")
        }

        gtk::Button::with_label("Increase") #append(&#) {
            connect_clicked: clone![tx; move |_| send!(Msg::Increase => tx)]
        }

        gtk::Button::with_label("Decrease") #append(&#) {
            connect_clicked: move |_| send!(Msg::Decrease => tx)
        }

        @update_view = move |count| bindings!()
    }
}

}]

fn start(app: &gtk::Application) { let (tx, rx) = glib::MainContext::channel(glib::PRIORITY_DEFAULT); let mut count = 0; // the state

expand_view_here! { }

let update_count = |count: &mut u8, msg| match msg {
    Msg::Increase => *count = count.wrapping_add(1),
    Msg::Decrease => *count = count.wrapping_sub(1),
};

rx.attach(None, move |msg| {
    update_count(&mut count, msg);
    update_view(count);
    glib::Continue(true)
});

window.present()

}

fn main() -> glib::ExitCode { let app = gtk::Application::default(); app.connect_activate(start); app.run() } ~~~

To execute, run:

~~~ bash cargo run --features gtk-rs --example y_readme ~~~

Basic maintenance

The following commands must be executed and must not give any problems:

~~~ bash cargo test --features gtk-rs cargo check cargo check --features gtk-rs cargo clippy cargo clippy --features gtk-rs

and now run and check each example

~~~

License

Licensed under either of

at your option.

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.