gpui-markup 0.4.0

A declarative markup DSL for building GPUI applications
Documentation
# gpui-markup

A declarative markup DSL for building [GPUI](https://gpui.rs) applications.

## Installation

```toml
[dependencies]
gpui-markup = "0.4"
```

## Usage

```rust
use gpui::prelude::*;
use gpui_markup::ui;

fn my_view(cx: &mut ViewContext<Self>) -> impl IntoElement {
    ui! {
        div {
            [flex, flex_col, gap: 2, p: 4, bg: cx.theme().colors().background]
            div {
                [text_size: px(24.0), font_weight: FontWeight::BOLD]
                "Hello, GPUI!",
            },
            div {
                [text_color: cx.theme().colors().text_muted]
                "A declarative way to build UIs",
            },
        }
    }
}
```

## Syntax

### Elements

All elements require braces `{}`. Attributes go inside braces with `[...]`:

```rust
// Empty div
ui! { div {} }
// -> div()

// Div with attributes
ui! { div { [flex, flex_col] } }
// -> div().flex().flex_col()

// Div with children
ui! { div { "content" } }
// -> div().child("content")

// Full form: attributes followed by children
ui! { div { [flex] "content" } }
// -> div().flex().child("content")
```

### Attributes

Attributes go inside `{ [...] }`, comma-separated:

```rust
// Flag attributes (no value)
ui! { div { [flex, flex_col] } }
// -> div().flex().flex_col()

// Key-value attributes
ui! { div { [w: px(200.0), h: px(100.0)] } }
// -> div().w(px(200.0)).h(px(100.0))

// Multi-value attributes (use tuples)
ui! { div { [when: (condition, |d| d.bg(red()))] } }
// -> div().when(condition, |d| d.bg(red()))
```

### Children

Children go inside `{...}`, comma-separated (after optional attributes):

```rust
ui! {
    div {
        "First",
        "Second",
        div { [bold] "Nested" },
    }
}
// -> div().child("First").child("Second").child(div().bold().child("Nested"))
```

### Deferred

The `deferred` element wraps content for deferred rendering:

```rust
ui! {
    deferred {
        div { "Deferred content" },
    }
}
// -> deferred(div().child("Deferred content").into_any_element())
```

### Spread Children

Use `..expr` to spread an iterable as children:

```rust
let items: Vec<Div> = vec![div().child("A"), div().child("B")];

ui! {
    div {
        ..items,
    }
}
// -> div().children(items)

// Can be mixed with regular children
ui! {
    div {
        "Header",
        ..items,
        "Footer",
    }
}
// -> div().child("Header").children(items).child("Footer")
```

### Method Chains

Use `.method(args)` to insert method calls at any position:

```rust
ui! {
    div {
        "static child",
        .when(condition, |d| d.child("dynamic")),
        .flex().gap_2(),
        .map::<Div, _>(|d| d),
    }
}
```

### Comments

Use standard Rust comments inside `ui!`:

```rust
ui! {
    div {
        // This is a comment
        "Visible content",
        /* Multi-line
           comment */
    }
}
// -> div().child("Visible content")
```

### Components

Components are any non-native element names. They automatically call `::new()`:

```rust
// Simple component
ui! { Header {} }
// -> Header::new()

// Component with attributes
ui! { Button { [style: Primary] } }
// -> Button::new().style(Primary)

// Component with children
ui! {
    Container {
        "Content",
        Footer {},
    }
}
// -> Container::new().child("Content").child(Footer::new())
```

### Expression Elements

Any expression can be used as an element. Braces are required:

```rust
// Custom constructor
ui! { Button::with_label("Click") {} }
// -> Button::with_label("Click")

// Expression with attributes
ui! { Button::with_label("Click") { [style: Primary] } }
// -> Button::with_label("Click").style(Primary)

// Builder pattern expression
ui! {
    div().flex() {
        [flex_col]
        "Content",
    }
}
// -> div().flex().flex_col().child("Content")

// Parentheses for complex expressions (braces optional)
ui! { (a + b) }
// -> a + b
```

### Nested Structures

```rust
ui! {
    div {
        [flex, flex_col, gap: 4]
        div {
            [flex, justify_between]
            Label {},
            Button { [on_click: handle_click] },
        },
        div {
            [flex: 1, overflow: hidden]
            ScrollView { content },
        },
    }
}
```

## How It Works

The `ui!` macro transforms the markup syntax into GPUI's builder pattern at compile time:

| Markup | Generated Code |
|--------|----------------|
| `div {}` | `div()` |
| `div { [flex] }` | `div().flex()` |
| `div { [w: x] }` | `div().w(x)` |
| `div { [when: (a, b)] }` | `div().when(a, b)` |
| `div { a, b }` | `div().child(a).child(b)` |
| `div { ..items }` | `div().children(items)` |
| `div { .a().b() }` | `div().a().b()` |
| `deferred { e }` | `deferred(e.into_any_element())` |
| `Header {}` | `Header::new()` |
| `Header { [a] }` | `Header::new().a()` |
| `expr {}` | `expr` |
| `expr { [a] }` | `expr.a()` |
| `(expr)` | `expr` |

## License

[MIT](./LICENSE). Made with ❤️ by [Ray](https://github.com/so1ve)