Skip to content

Components & Structure

A2UI uses an adjacency list model for component hierarchies. Instead of nested JSON trees, components are a flat list with ID references.

Traditional nested approach:

  • LLM must generate perfect nesting in one pass
  • Hard to update deeply nested components
  • Difficult to stream incrementally

A2UI adjacency list:

  • ✅ Flat structure, easy for LLMs to generate
  • ✅ Send components incrementally
  • ✅ Update any component by ID
  • ✅ Clear separation of structure and data

=== “v0.8”

```json
{
"surfaceUpdate": {
"components": [
{
"id": "root",
"component": {
"Column": {
"children": { "explicitList": ["greeting", "buttons"] }
}
}
},
{
"id": "greeting",
"component": {
"Text": { "text": { "literalString": "Hello" } }
}
},
{
"id": "buttons",
"component": {
"Row": {
"children": { "explicitList": ["cancel-btn", "ok-btn"] }
}
}
},
{
"id": "cancel-btn",
"component": {
"Button": {
"child": "cancel-text",
"action": { "name": "cancel" }
}
}
},
{
"id": "cancel-text",
"component": {
"Text": { "text": { "literalString": "Cancel" } }
}
},
{
"id": "ok-btn",
"component": {
"Button": {
"child": "ok-text",
"action": { "name": "ok" }
}
}
},
{
"id": "ok-text",
"component": {
"Text": { "text": { "literalString": "OK" } }
}
}
]
}
}
```

=== “v0.9”

```json
{
"version": "v0.9",
"updateComponents": {
"surfaceId": "main",
"components": [
{
"id": "root",
"component": "Column",
"children": ["greeting", "buttons"]
},
{
"id": "greeting",
"component": "Text",
"text": "Hello"
},
{
"id": "buttons",
"component": "Row",
"children": ["cancel-btn", "ok-btn"]
},
{
"id": "cancel-btn",
"component": "Button",
"child": "cancel-text",
"action": { "event": { "name": "cancel" } }
},
{
"id": "cancel-text",
"component": "Text",
"text": "Cancel"
},
{
"id": "ok-btn",
"component": "Button",
"child": "ok-text",
"action": { "event": { "name": "ok" } }
},
{
"id": "ok-text",
"component": "Text",
"text": "OK"
}
]
}
}
```
v0.9 uses a flatter component format: `"component": "Text"` instead of nested `{"Text": {...}}`, and children are simple arrays instead of `{"explicitList": [...]}`.

Components reference children by ID, not by nesting.

Every component has:

  1. ID: Unique identifier ("welcome")
  2. Type: Component type (Text, Button, Card)
  3. Properties: Configuration specific to that type

=== “v0.8”

```json
{
"id": "welcome",
"component": {
"Text": {
"text": { "literalString": "Hello" },
"usageHint": "h1"
}
}
}
```

=== “v0.9”

```json
{
"id": "welcome",
"component": "Text",
"text": "Hello",
"variant": "h1"
}
```

A2UI defines a standard catalog of components organized by purpose:

  • Layout: Row, Column, List - arrange other components
  • Display: Text, Image, Icon, Video, Divider - show information
  • Interactive: Button, TextField, CheckBox, DateTimeInput, Slider - user input
  • Container: Card, Tabs, Modal - group and organize content

For the complete component gallery with examples, see Component Reference.

Static (explicitList) - Fixed list of child IDs:

{
"children": {
"explicitList": ["back-btn", "title", "menu-btn"]
}
}

Dynamic (template) - Generate children from data array:

{
"children": {
"template": {
"dataBinding": "/items",
"componentId": "item-template"
}
}
}

For each item in /items, render the item-template. See Data Binding for details.

Components get their values two ways:

  • Literal - Fixed value: {"text": {"literalString": "Welcome"}}
  • Data-bound - From data model: {"text": {"path": "/user/name"}}

LLMs can generate components with literal values or bind them to data paths for dynamic content.

Components compose into surfaces (widgets):

=== “v0.8”

1. LLM generates component definitions via `surfaceUpdate`
2. LLM populates data via `dataModelUpdate`
3. LLM signals render via `beginRendering`
4. Client renders all components as native widgets

=== “v0.9”

1. LLM creates a surface via `createSurface` (specifying catalog)
2. LLM generates component definitions via `updateComponents`
3. LLM populates data via `updateDataModel`
4. Client renders all components as native widgets

A surface is a complete, cohesive UI (form, dashboard, chat, etc.).

  • Add - Send new component definitions with new IDs
  • Update - Send component definitions with existing ID and new properties
  • Remove - Update parent’s children list to exclude removed IDs

The flat structure makes all updates simple ID-based operations.

Beyond the standard catalog, clients can define custom components for domain-specific needs:

  • How: Register custom component types in your renderer
  • What: Charts, maps, custom visualizations, specialized widgets
  • Security: Custom components still part of the client’s trusted catalog

Custom components are advertised from the client’s renderer to the LLM. The LLM can then use them in addition to the standard catalog.

See Custom Components Guide for implementation details.

  1. Descriptive IDs: Use "user-profile-card" not "c1"
  2. Shallow hierarchies: Avoid deep nesting
  3. Separate structure from content: Use data bindings, not literals
  4. Reuse with templates: One template, many instances via dynamic children