fltk-decl

Use a declarative language (json5, json, yaml, xml, toml) to describe your fltk-rs gui, with support for hot-reloading of your gui file. The crate is designed to be as permissive as possible. So wrong keys or values will be ignored. Normally only changing a widget's id at runtime would cause an error!

Usage

In your Cargo.toml: toml [dependencies] fltk-decl = "0.1"

Create a json file, let's call it gui.json. ```json { "$schema": "https://raw.githubusercontent.com/MoAlyousef/fltk-decl/main/schemas/fltk-schema.json", "widget": "Column", "children": [ { "widget": "Button", "label": "Inc", "fixed": 60, "id": "inc", "labelcolor": "#0000ff"

    },
    {
        "widget": "Row",
        "children": [
            {
                "widget": "Frame",
                "fixed": 30
            },
            {
                "widget": "Frame",
                "label": "0",
                "id": "result",
                "labelcolor": "#ff0000"
            },
            {
                "widget": "Frame",
                "fixed": 30
            }
        ]
    },
    {
        "widget": "Button",
        "label": "Dec",
        "fixed": 60,
        "id": "dec"
    }
]

} ``` Notice we point to the schema to get auto-completion and hinting on vscode, otherwise it's optional.

Note that this crate uses json5, so you could just as easily change your gui.json to gui.json5 (to benefit from comments, trailing commas and unquoted keys!): json5 { // main column widget: "Column", children: [ { // our button widget: "Button", label: "Click me", color: "#ff0000", id: "my_button", } ], } However, you lose vscode's auto-completion since json5 extensions in vscode don't support schemas.

You could also use yaml:

```yaml

widget: Column children: - widget: Button label: Inc fixed: 60 id: inc labelcolor: "#0000ff" - widget: Row children: - widget: Frame fixed: 30 - widget: Frame label: '0' id: result labelcolor: "#ff0000" - widget: Frame fixed: 30 - widget: Button label: Dec fixed: 60 id: dec ```

You could also use xml: gui.xml xml <?xml version="1.0" encoding="UTF-8"?> <root> <widget>Column</widget> <children> <widget>Button</widget> <label>Click Me</label> <id>my_button</id> <labelcolor>#0000ff</labelcolor> </children> </root>

or toml! ``` widget = "Column"

[[children]] widget = "Button" label = "Click Me" id = "my_button" ```

Import it into your app: ```rust use fltk_decl::DeclarativeApp;

fn main() { // use the filetype and extension that you require. // run a callback that runs at least once, or whenever the gui file changes. DeclarativeApp::new(200, 300, "MyApp", "gui.json").run(true, |mainwin| {}); } ```

To handle callbacks: ```rust use fltk::{prelude::*, *}; use fltk_decl::DeclarativeApp;

// use the extension you require! const PATH: &str = "examples/gui.json";

[derive(Clone, Copy)]

struct State { count: i32, }

impl State { pub fn increment(&mut self, val: i32) { let mut result: frame::Frame = app::widgetfromid("result").unwrap(); self.count += val; result.setlabel(&self.count.tostring()); } }

fn btn_cb(b: &mut button::Button) { let state = app::GlobalState::::get(); let val = if b.label() == "Inc" { 1 } else { -1 }; state.with(move |s| s.increment(val)); }

fn main() { app::GlobalState::new(State { count: 0 }); DeclarativeApp::new(200, 300, "MyApp", PATH) .run(|win| { app::setscheme(app::Scheme::Oxy); if let Some(mut btn) = app::widgetfromid::("inc") { btn.setcallback(btncb); } if let Some(mut btn) = app::widgetfromid::("dec") { btn.setcallback(btncb); } }) .unwrap(); } ```

Supported properties:

Supported widgets: