declarative
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.5.2'
features = ['builder-mode']
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"] } ~~~
In addition to macro features, the examples also show some usage patterns (templates, components, Elm, etc.). GTK has a pattern of its own due to its object orientation and declarative
integrates well, but there is no example about it (it would be verbose and exclusive to GTK, while declarative
is not GTK based).
In the following I manually implement the Elm pattern. The macro does not require any specific pattern.
~~~ rust use declarative::{block as view, builder_mode, clone}; use gtk::{glib, prelude::*};
enum Msg { Increase, Decrease }
// syntactic sugar for sending messages: macro_rules! send { [$msg:expr => $tx:expr] => [$tx.send($msg).unwrap()] }
fn start(app: >k::Application) { let (tx, rx) = glib::MainContext::channel(glib::PRIORITY_DEFAULT); let mut count = 0; // the state
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)
}
@refresh = move |count| bindings!()
}
}
}
let update = |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(&mut count, msg); // the state is updated
refresh(count); // now the view is refreshed
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 ~~~
The following commands must be executed and must not give any problems:
~~~ bash cargo test -p declarative-macros cargo test -p declarative-macros --features builder-mode cargo check -p declarative-macros cargo check -p declarative-macros --features builder-mode cargo clippy -p declarative-macros cargo clippy -p declarative-macros --features builder-mode cargo test --features gtk-rs cargo check cargo check --features gtk-rs cargo clippy cargo clippy --features gtk-rs
~~~
If you need a changelog, maybe the commit log will help (the last ones try to have the most important details).
Licensed under either of
at your option.
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.