superlighttui 0.20.1

Super Light TUI - A lightweight, ergonomic terminal UI library
Documentation
# SLT Migration Guide

A version-by-version upgrade guide for SuperLightTUI users. Pre-1.0, minor versions
(`0.X.0`) may contain breaking changes; patch versions (`0.X.Y`) must not.

For the full per-release notes, see `CHANGELOG.md`. For composition recipes that use
current APIs, see `docs/PATTERNS.md` and `docs/COOKBOOK.md`.

---

## 1. Quick reference: v0.19 → v0.20

The next minor release contains breaking changes from issues #98, #102, #134, #149, #161,
#184, #192, and #193. Most are caught by the compiler — a few are runtime/visual.

| Change | Severity | Migration |
|---|---|---|
| `FilePickerState::selected()``selected_file()` (#98) | Rename — Cargo compile-error | `s/\.selected()/\.selected_file()/g` on `FilePickerState` calls |
| `ContainerBuilder::scroll_offset()` `pub``pub(crate)` (#149) | Compile error if used externally | Use `ScrollState` + `scrollable()` widget instead |
| `scrollbar()` return type `()``Response` (#184) | `unused_must_use` warning if ignored | `let _ = ui.scrollbar(&state);` to discard |
| `virtual_list` cursor stays in viewport (#192) | Visual change, no compile error | If you depended on the old cursor=bottom-row behavior, set `state.viewport_offset = state.selected.saturating_sub(visible_height - 1)` manually |
| `calendar` `h`/`l` = day ±1 (was month) (#193) | Keybinding behavior change | Document for users; `[`/`]` is the new month nav |
| `TextareaState` struct-literal init broken (#102) | Compile error if you used struct literal | Use `TextareaState::new()` constructor (private fields added for undo/redo) |
| `screen_hook_map` keys: `String``&'static str` (#134) | Compile error if you wrote dynamic `String` | Use string literals; for runtime keys, `Box::leak(s.into_boxed_str())` |
| `flex-shrink` global behavior change (#161) | Pixel-exact positions change for overflowing rows/columns | Re-screenshot snapshot tests; only affects layouts that previously overflowed and shrank |

### How to upgrade

1. Bump `superlighttui` in `Cargo.toml` to `0.20.0`
2. `cargo build` — fix compile errors with the table above
3. Re-run any TUI snapshot tests; eyeball overflow layouts and `virtual_list` cursor behavior
4. If your app exposes its own keybindings docs, update them for `calendar`

### Concrete examples

**#98 — `FilePickerState`**

```rust
// Before (v0.19.x)
if let Some(path) = state.selected() {
    open_file(path);
}

// After (v0.20)
if let Some(path) = state.selected_file() {
    open_file(path);
}
```

The struct field `state.selected_file: Option<PathBuf>` already exists in v0.19.x
(`src/widgets/collections.rs` line ~102) — only the accessor method is renamed.
Direct field access (`&state.selected_file`) was always available and keeps working.

**#184 — `scrollbar()` return type**

```rust
// Before (v0.19.x): scrollbar returns ()
ui.scrollbar(&scroll_state);

// After (v0.20): scrollbar returns Response, must be used or discarded
let _ = ui.scrollbar(&scroll_state);
// or, if you want to react to interaction:
if ui.scrollbar(&scroll_state).clicked { /* ... */ }
```

**#192 — `virtual_list` cursor**

The new behavior: cursor stays inside the visible viewport instead of pinning to the
bottom row. If your app relied on the old behavior to surface "newest item at the
cursor" (e.g. a streaming log), opt back in by setting `viewport_offset` manually:

```rust
let visible_height = list_rect.height as usize;
state.viewport_offset = state.selected.saturating_sub(visible_height - 1);
```

**#134 — `screen_hook_map` keys**

```rust
// Before (v0.19.x): HashMap<String, _>
ui.screen_hook_map.insert(format!("screen_{i}"), (start, count));

// After (v0.20): HashMap<&'static str, _>
ui.screen_hook_map.insert("screen_static", (start, count));

// For runtime-generated keys, leak into 'static:
let key: &'static str = Box::leak(format!("screen_{i}").into_boxed_str());
ui.screen_hook_map.insert(key, (start, count));
```

Most callers use static names — the `Box::leak` workaround is for the rare dynamic case.

---

## 2. v0.19.x → v0.19.3 (deprecations only — no code break)

These compiled fine in v0.19.x but now produce deprecation warnings. They will be
**removed in v0.20**:

| Old | New |
|---|---|
| `pad(n)` | `p(n)` |
| `min_width(n)` | `min_w(n)` |
| `max_width(n)` | `max_w(n)` |
| `min_height(n)` | `min_h(n)` |
| `max_height(n)` | `max_h(n)` |

Source-of-truth: `src/context/container.rs` lines ~1050–1069 carry the
`#[deprecated(since = "0.20.0", ...)]` attributes; `pad` is at ~835.

### Codemod (sed-based)

```bash
# Run from your project root. Backs up files before modifying.
find src/ -name "*.rs" -exec sed -i.bak -E '
  s/\.pad\(/\.p(/g;
  s/\.min_width\(/\.min_w(/g;
  s/\.max_width\(/\.max_w(/g;
  s/\.min_height\(/\.min_h(/g;
  s/\.max_height\(/\.max_h(/g;
' {} \;
```

**Warning**: this is a regex codemod, not AST-aware. False positives possible if your
project has unrelated `.pad()` / `.min_width()` etc. methods on non-`ContainerBuilder`
types (e.g. a custom `MyRect::pad`). Review the diff before committing:

```bash
git diff
# Looks good? Drop the backups:
find src/ -name "*.rs.bak" -delete
```

For comparison, React migrations use `npx jscodeshift` (AST-aware). The Rust ecosystem
doesn't have an equivalent for SLT yet — see section 5.

---

## 3. v0.18.x → v0.19.0 (additive, no break)

v0.19.0 added component DX APIs in `src/context/runtime.rs` (`provide`, `use_context`,
`use_state_named`, `with_if`). All existing code keeps compiling. Adopt incrementally:

| Old pattern | New pattern (v0.19.0+) |
|---|---|
| Threading `&theme, &tick, &mut state` parameters through every render fn | `ui.provide(value, \|ui\| ...)` once, then `ui.use_context::<T>()` in nested fns |
| `use_state` inside `if`/`match` (panics on hook-order mismatch) | `use_state_named(id)` (id-keyed, safe in conditionals) |
| `if cond { t.bold(); t.fg(Color::Red); }` | `t.with_if(cond, \|t\| t.bold().fg(Color::Red))` |

See `docs/PATTERNS.md` for the full set of composition patterns.

---

## 4. v0.17.x → v0.18.x

Major architecture refactor — most user-facing APIs unchanged but internal modules split:

- `src/context.rs` → facade (was monolithic)
- `src/widgets.rs` → facade
- `src/layout.rs` → facade

If you had `pub(crate)` references to specific module paths from inside the SLT crate
(forks, or downstream crates that depended on undocumented internals), they may have
moved. Most external users see no break.

Key APIs added in v0.18:

- `NO_COLOR` env var support — automatically disables ANSI colors when set
- `scroll_col` helper — vertical scroll for column layouts
- `draw_with(rect, |buf| { ... })` — escape hatch for raw buffer access
- `try_get(rect, x, y) -> Option<&Cell>` — bounds-safe cell access in raw-draw

v0.18.1 added doc improvements and safety hardening; v0.18.2 added flush coalescing,
the `Command` box pattern, and a flexbox scratch buffer (perf only).

---

## 5. Comparison: how other UI libraries handle migrations

| Framework | Tooling | Convention |
|---|---|---|
| **React** | `npx jscodeshift` AST codemods + RFC | Major every 2–3 years; deprecation warnings ≥1 minor before removal |
| **Vue** | `vue-codemod` AST tool + composition API guide | Major: 3.x → vue-next migration build |
| **Angular** | `ng update` schematic CLI (auto-migration) | Strict semver, deprecation warnings, automatic migration |
| **Flutter** | `dart fix --apply` | Annotated deprecations migrate via `dart fix` |
| **SLT** | sed-based codemods (no AST tool yet) + this guide + deprecation warnings | Strict semver, deprecation warnings ≥1 minor before removal in v0.x |

SLT does **not** yet have an AST-aware codemod tool. PRs welcome — `cargo-slt-migrate`
would be a high-impact contribution. Until then, the sed snippets in section 2 plus
`cargo build` errors are the migration toolchain.

---

## 6. Pre-1.0 versioning policy

- **Pre-1.0 (current)**: minor versions (`0.X.0`) MAY contain breaking changes; patch
  versions (`0.X.Y`) MUST NOT.
- **Post-1.0 (TBD)**: standard semver — major for breaking, minor for features, patch
  for fixes.
- CI runs `cargo-semver-checks` on every PR to verify patch releases stay non-breaking.
  Minor releases are allowed to fail this check; major/minor breakage is documented in
  the release notes and reflected in this file.

Deprecation policy: every API removal in a minor release is preceded by at least one
minor release that emits `#[deprecated(since = "X.Y.Z", note = "...")]` warnings. So if
you keep `cargo build` clean of deprecation warnings before each minor bump, you should
not hit surprise removals.

---

## 7. Reporting migration issues

If you find a migration step that doesn't work, errors out, or is missing from this
guide:

- File an issue: <https://github.com/subinium/SuperLightTUI/issues>
- Tag with `migration` label
- Include:
  - SLT version you're upgrading **from** and **to**
  - Minimal code snippet that reproduces the failure
  - Full error message (compile error, panic, or visual diff screenshot)

PRs that add migration entries here are welcome — keep entries to the same format as
the tables above.

---

## 8. Cross-references

- `CHANGELOG.md` — full per-release notes with every commit
- `docs/COOKBOOK.md` — recipes using current APIs
- `docs/PATTERNS.md` — composition patterns (`provide` / `use_context` / `with_if`)
- `docs/COMPLETE_REFERENCE.md` — full API surface
- `docs/STATE_APIS.md` — state-type method reference