* feat(desktop): add native host scaffold * feat(webui): track turns and usage in gateway * feat(webui): polish desktop chat experience * feat(apps): add ArcGIS and Joplin logos * feat(desktop): polish shell and shared surfaces * fix(webui): avoid preview chips for glob references * test: align CI expectations for token fallback * feat(webui): preview prompt rail entries * feat(webui): add prompt navigator drawer * style(webui): refine prompt navigator placement * style(webui): align prompt navigator with header actions * style(webui): simplify prompt navigator header * refactor(webui): clean thread resource refresh * feat(desktop): add native reply notifications * fix(webui): preserve desktop restart and replay state * fix(desktop): harden gateway proxy startup * fix(web): fall back when readability is unavailable * fix(desktop): hide window instead of closing on macos * fix(webui): unify desktop header actions * fix(webui): simplify prompt history rows * fix(desktop): log notification delivery failures * chore(desktop): clean source package artifacts * fix(cron): support one-time relative reminders * fix(webui): reveal scroll button in place * Revert "fix(cron): support one-time relative reminders" This reverts commit 4c4661da120a3c7283e0768412bae48604e7390b. * refactor(webui): extract token usage heatmap * docs(desktop): clarify contributor guides --------- Co-authored-by: chengyongru <2755839590@qq.com>
4.1 KiB
Desktop Development Guide
This guide is for GitHub contributors who want to change the desktop app. If you are using nanobot rather than developing it, the important bit is simpler: desktop runs the local engine for you and shows the same chat, settings, apps, skills, and workspace UI as the browser WebUI.
desktop is the native host for the shared nanobot WebUI. It is not a fork of
the WebUI, and it should not grow a second copy of product UI.
The healthy mental model is:
nanobot core -> agent runtime, gateway, providers, tools, memory
webui -> shared product UI and runtime-aware UI
desktop -> native host, engine lifecycle, secure host capabilities
Development Workflow
Use this when developing from a source checkout.
Run the shared WebUI dev server:
cd desktop
bun run dev:webui
Run the Electron host in another terminal:
cd desktop
bun run dev:app
In development, Electron loads http://127.0.0.1:5173, so changes under
webui/src hot reload. Changes under desktop/src require restarting
dev:app.
For source checkouts, the host starts the engine with local python3 and
injects the repository root into PYTHONPATH. This means Python changes under
nanobot/ are picked up from the current checkout.
Where Code Goes
Use this table before adding a desktop feature:
| Change | Location |
|---|---|
| Agent behavior, tools, providers, memory, config schema | nanobot/ |
| Shared chat UI, settings UI, reusable product UI | webui/ |
| Runtime-aware UI rows, such as native engine status or open logs buttons | webui/ |
| The implementation behind native capabilities | desktop/src/main.ts |
| The trusted renderer bridge contract | desktop/src/preload.cts and desktop/docs/host-contract.md |
| Electron window, app protocol, native menus, lifecycle, packaging | desktop/src and desktop/package.json |
| WebSocket-over-Unix-socket bridge | desktop/src/unixWebSocket.ts |
| Bundled Python runtime preparation | desktop/scripts/prepare-engine.mjs |
For example, if desktop Settings needs an "Open logs" button, the button belongs
in the shared WebUI settings page because it is product UI. The actual filesystem
operation belongs in the desktop host and is exposed through window.nanobotHost.
Host Contract
The shared WebUI talks to desktop through window.nanobotHost. WebUI code may
check for host capabilities, but it must not import Electron, Node.js modules,
or desktop source files.
Prefer capability-driven UI:
if host can open logs -> show Open logs
if host can restart engine -> show Restart engine
Avoid platform-driven UI:
if desktop -> run Electron-specific logic in WebUI
This keeps the WebUI usable in browsers and leaves room for future native hosts without rewriting product screens.
Adding A Desktop Feature
Before implementing, answer these questions:
- Is this product UI or a native capability?
- Can the WebUI express it through a generic capability instead of a desktop flag?
- Does the host API validate trusted origins and accepted URL schemes?
- Does browser WebUI still work when
window.nanobotHostis missing? - Does the engine behavior belong in nanobot core instead of Electron?
- Does packaged mode use app data for user state instead of app resources?
Anti-Patterns
- Do not copy or fork
webui/srcintodesktop/. - Do not import Electron or Node.js modules from
webui/src. - Do not add provider-specific onboarding screens to
desktop/. - Do not duplicate WebUI settings or login flows in Electron-owned HTML.
- Do not make
desktop/src/main.tsown agent behavior. - Do not commit
desktop/node_modules,desktop/build,desktop/dist, DMGs, ordesktop/resources/nanobot-engine.
Release Shape
Release builds assemble three existing parts:
- the shared WebUI build from
nanobot/web/dist, - the Python engine prepared under
desktop/resources/nanobot-engine, - the Electron host compiled from
desktop/src.
User config, logs, sessions, workspace state, and the default workspace live in the platform app data directory, not inside the app bundle.