Chapter 10: Spatial UI with UIKitML 
This chapter teaches you how to create immersive spatial user interfaces using pmndrs/uikit, specifically uikitml – IWSDK's spatial UI system.
What You'll Learn 
By the end of this chapter, you'll be able to:
- Understand spatial UI design principles
 - Create UI layouts using UIKitML markup
 - Position and scale interfaces in 3D space
 - Handle user interactions with spatial UI elements
 - Implement common UI patterns for WebXR
 
Spatial UI Principles 
Great spatial interfaces follow these principles:
- World-scale: UI elements have a physical presence in 3D space
 - Natural interaction: Use pointing, grabbing, and gestures
 - Readable at distance: Text and icons scale appropriately
 - Contextual placement: UI appears near relevant objects
 - Comfortable viewing: Positioned to avoid neck strain
 
Introduction to Building Spatial User Interfaces in IWSDK 
The unavailability of HTML in WebXR has been a big challenge for developers, since manually placing user interface elements is very cumbersome. That's why IWSDK uses pmndrs/uikit, a GPU-accelerated UI rendering system that provides an API aligned with HTML and CSS, allowing developers to feel right at home. To make UI authoring even more natural, IWSDK uses the uikitml language, which allows developers to write user interfaces using an HTML-like syntax, including features such as CSS classes. This integration allows IWSDK developers to reuse their HTML knowledge to quickly build high-performance, GPU-accelerated user interfaces for WebXR. Furthermore, IWSDK makes use of the pre-built component collections offered by the uikit project: the Default Kit (based on shadcn) and the Horizon Kit (based on the Reality Labs Design System).
Key Features 
- Declarative markup: Describe UI structure, not implementation
 - 3D layout system: Flexbox-like layouts in 3D space
 - Component Kits: Pre-built buttons, panels, sliders, etc.
 - Event system: Handle clicks, hovers, and other interactions
 - Theming support: Consistent styling across your application
 
Basic Syntax 
UIKitML uses HTML-style markup with CSS properties for styling and layouting:
<!-- Basic panel with text -->
<div class="panel" style="width: 400; height:300; background-color: #2a2a2a">
  <text style="fontSize:24px; color: white">Hello WebXR!</text>
  <button>Click Me</button>
</panel>Setting Up UIKitML with IWSDK 
Vite Plugin Configuration 
IWSDK includes a Vite plugin that compiles UIKitML files:
// vite.config.js
import { defineConfig } from 'vite';
import { uikitml } from '@iwsdk/vite-plugin-uikitml';
export default defineConfig({
  plugins: [
    uikitml({
      // File extensions to process
      include: ['**/*.uikitml'],
      // Hot reload during development
      hotReload: true,
    }),
  ],
});Creating Your First UIKitML File 
Create src/ui/main-menu.uikitml and insert the following content, which uses the Panel, Button, ButtonIcon, and LoginIcon components from the Horizon Kit to design a panel with a button:
<style>
  .panel-root {
    padding: 16px;
    flex-direction: column;
    width: 344px;
  }
</style>
<Panel class="panel-root">
  <button id="xr-button">
    <ButtonIcon>
      <LoginIcon></LoginIcon>
    </ButtonIcon>
    Enter XR
  </button>
</Panel>Loading UI in Your Application 
We can add our panelWithButton uikitml user interface to our IWSDK scene using the PanelUI and PanelDocument components:
export class PanelSystem extends createSystem({
  panelWithButton: {
    required: [PanelUI, PanelDocument],
    where: [eq(PanelUI, 'config', '/ui/main-menu.json')],
  },
}) {}Adding Component Kits to Your Spatial User Interface 
If you'd like to use a different or additional component kit for your uikitml file, you can configure the kits in the spatialUI feature list when creating a World:
import * as horizonKit from "@pmndrs/uikit-horizon";
World.create(document.getElementById("scene-container"), {
  ...
  features: {
    ...
    spatialUI: { kits: [horizonKit] },
  },
})Overview of Properties and Features Available for Building Spatial User Interfaces 
When authoring a User Interface with IWSDK, developers can use almost all the features they know and love from HTML. The following section shows all the available element types and styling methods for designing Spatial User Interfaces.
Element Types 
Container Elements 
Most HTML elements become containers that can hold children and text.
<div>Layout container</div>
<p>Paragraph text</p>
<h1>Main heading</h1>
<button>Click me</button>
<ul>
  <li>List item</li>
</ul>Image Elements 
Display bitmap images in your 3D UI.
<img src="photo.jpg" alt="Description" />
<img src="icon.png" class="avatar" />
<img src="icon.svg" />Required: src attribute
Inline SVG Elements 
Embed SVG markup directly in your UI.
<svg viewBox="0 0 100 100">
  <circle cx="50" cy="50" r="40" fill="blue" />
  <rect x="10" y="10" width="30" height="30" fill="red" />
</svg>Content: Raw SVG markup is preserved and rendered
Video Elements 
Display video content with standard HTML5 video attributes.
<video src="movie.mp4" controls autoplay /> <video src="demo.webm" loop muted />Required: src attribute Supports: All standard HTML5 video attributes
Input Elements 
Create interactive input fields for user data.
<input type="text" placeholder="Enter your name" />
<input type="email" value="user@example.com" />
<textarea placeholder="Multi-line text input">Default content</textarea>Component Kits 
In addition to these elements, developers can also use the installed kits.
<button id="xr-button">
  <ButtonIcon>
    <LoginIcon></LoginIcon>
  </ButtonIcon>
  Enter XR
</button>Styling System 
Inline Styles 
Use familiar CSS properties with kebab-casing directly on elements:
<div style="background-color: blue; padding: 20px; border-radius: 8px;">
  Styled container
</div>CSS Classes 
Define reusable styles with full pseudo-selector support using the <style> tag:
<style>
  .button {
    background-color: #3b82f6;
    color: white;
    padding: 12px 24px;
    border-radius: 6px;
    cursor: pointer;
  }
  .button:hover {
    background-color: #2563eb;
    transform: scale(1.05);
  }
  .button:active {
    background-color: #1d4ed8;
    transform: scale(0.95);
  }
  /* Responsive styles */
  .button:sm {
    padding: 8px 16px;
    font-size: 14px;
  }
  .button:lg {
    padding: 16px 32px;
    font-size: 18px;
  }
</style>
<button class="button">Interactive Button</button>Supported selectors:
- States: 
:hover,:active,:focus - Responsive: 
:sm,:md,:lg,:xl,:2xl 
ID-Based Styling 
Style specific elements using ID selectors:
<style>
  #header {
    background-color: #ff6b6b;
    padding: 20px;
    justify-content: center;
  }
  #header:hover {
    opacity: 0.9;
  }
</style>
<div id="header">
  <h1>Welcome to uikitml</h1>
</div>Handling User Interactions 
UIKitML provides an event system for handling user interactions:
export class PanelSystem extends createSystem({
  welcomePanel: {
    required: [PanelUI, PanelDocument],
    where: [eq(PanelUI, 'config', '/ui/main-menu.json')],
  },
}) {
  init() {
    this.queries.welcomePanel.subscribe('qualify', (entity) => {
      const document = PanelDocument.data.document[
        entity.index
      ] as UIKitDocument;
      if (!document) return;
      const xrButton = document.getElementById('xr-button') as UIKit.Text;
      xrButton.addEventListener('click', () => {
        // TODO: add your interactivity here
      });
    });
  }
}Troubleshooting 
UI Not Appearing 
UI document loads but nothing shows?
- Check that the position is in front of the player
 - Verify the scale is appropriate (try 0.001 for pixel-based layouts)
 - Ensure UISystem is registered with the world
 - Ensure your elements have a color different then their background
 
Interaction Issues 
Clicks not working?
- Verify event listeners are attached to the UI element
 - Check if anything is blocking the UI
 
Layout Issues 
Elements not positioning correctly?
- Check 
flexDirectionand alignment properties - Verify the parent container has appropriate dimensions
 - Use the UIKitML VSCode extension to understand the size and position of individual elements by hovering over them