| # Custom Web UI |
|
|
| When `ENABLE_WEB_INTERFACE=true`, the server serves a default Gradio app at `/web` with Reset/Step/Get state, Quick Start, and README. Environment authors can **add** a custom tab by providing a custom Gradio builder. |
|
|
| ## Extension point: `gradio_builder` |
| |
| `create_app()` accepts an optional **`gradio_builder`** callable. When set, the UI at `/web` is built with [Gradio’s TabbedInterface](https://www.gradio.app/4.44.1/docs/gradio/tabbedinterface): the **first tab (“Playground”)** is the default OpenEnv UI, and the **second tab (“Custom”)** is the `gr.Blocks` returned by your builder. Users can switch between the default Playground and your custom interface without losing either. The same `/web/reset`, `/web/step`, `/web/state`, and `/web/metadata` API routes remain available; your custom tab can use the provided `web_manager` in-process or call those endpoints. |
| |
| ### Builder signature |
| |
| ```python |
| def my_gradio_builder( |
| web_manager, # WebInterfaceManager: .reset_environment(), .step_environment(), .get_state() |
| action_fields, # list[dict]: from action schema for form generation |
| metadata, # EnvironmentMetadata | None: name, readme_content, etc. |
| is_chat_env, # bool: True if single message input |
| title, # str: app title (e.g. metadata.name) |
| quick_start_md, # str: Quick Start markdown (class names already replaced) |
| ) -> gr.Blocks: |
| ... |
| ``` |
| |
| Return a `gr.Blocks` instance. It is shown in the **“Custom”** tab of a tabbed interface; the **“Playground”** tab always shows the default OpenEnv UI. Core applies the same theme/css when mounting. |
|
|
| --- |
|
|
| ## Option 1: Add a custom tab |
|
|
| Provide a builder that returns your own `gr.Blocks`; it appears as the second tab (“Custom”) next to the default “Playground” tab: |
|
|
| ```python |
| # server/app.py |
| from openenv.core.env_server.http_server import create_app |
| from .my_environment import MyEnvironment |
| from ..models import MyAction, MyObservation |
| from .gradio_ui import build_my_gradio_app # your module |
| |
| app = create_app( |
| MyEnvironment, |
| MyAction, |
| MyObservation, |
| env_name="my_env", |
| gradio_builder=build_my_gradio_app, |
| ) |
| ``` |
|
|
| In `server/gradio_ui.py` implement `build_my_gradio_app(web_manager, action_fields, metadata, is_chat_env, title, quick_start_md)` returning a `gr.Blocks` (e.g. env-specific visualizations, extra controls). Use `web_manager.reset_environment()`, `web_manager.step_environment(action_data)`, and `web_manager.get_state()` in your Gradio event handlers. The default Playground tab remains available in the first tab. |
|
|
| --- |
|
|
| ## Option 2: Custom tab that wraps or reuses the default |
|
|
| Your builder can call the core `build_gradio_app` to get a Blocks instance and embed it inside your custom tab (e.g. in a `gr.Tabs` or as one section). That way your “Custom” tab can show both the default layout and additional content in one place. |
|
|
| --- |
|
|
| ## Option 3: Custom Quick Start or README only |
|
|
| You don’t need a custom builder only to change text. The default UI uses: |
|
|
| - **Quick Start**: generated from `get_quick_start_markdown(metadata, action_cls, observation_cls)` (init-style class names). |
| - **README**: `metadata.readme_content` (loaded from the env’s README). |
|
|
| So you can influence the default UI by ensuring `metadata` and README are correct. To change the Quick Start template itself (e.g. different wording or placeholders), you would use a custom `gradio_builder` that calls `build_gradio_app` with a custom `quick_start_md` string you build yourself (or by copying and adapting the default template from the core). |
|
|
| --- |
|
|
| ## Migration from custom HTML override (e.g. wildfire) |
|
|
| Environments that currently override `/web` with custom HTML (e.g. by removing the default route and adding a GET `/web` that returns HTML) should migrate to a **gradio_builder** that returns a `gr.Blocks` app. The custom UI then appears in the **“Custom”** tab alongside the default **“Playground”** tab. Benefits: |
| |
| - Single, supported extension point using [TabbedInterface](https://www.gradio.app/4.44.1/docs/gradio/tabbedinterface). |
| - No need to remove or override routes; the default UI stays in the first tab. |
| - Same `/web` path; both tabs can use `web_manager` or `/web/reset`, `/web/step`, `/web/state`. |
| |
| If you need a non-Gradio custom UI (e.g. static HTML/JS), you can still register your own route after `create_app` (e.g. at `/web/custom` or another path), but the main `/web` slot is the Gradio tabbed app when `ENABLE_WEB_INTERFACE=true`. |
| |
| --- |
| |
| ## Summary |
| |
| | Goal | Approach | |
| |-----------------------------|---------------------------------------------------------------------------| |
| | Use default UI only | Do not pass `gradio_builder`. | |
| | Add a custom tab | Pass `gradio_builder=my_builder`; return your own `gr.Blocks` (shown in “Custom” tab). | |
| | Custom tab + default inside | In your builder, call `build_gradio_app(...)` and embed or wrap it in your Blocks. | |
| | Change Quick Start / README | Rely on metadata/README, or custom builder that builds custom markdown. | |
| |
| The default Playground tab is built with `openenv.core.env_server.gradio_ui.build_gradio_app`; you can import and call it with the same arguments if your custom tab needs to embed or extend it. |
| |