Learn-skills.dev electrobun-best-practices

Use when building or maintaining Electrobun desktop apps in TypeScript, including electrobun.config.ts, electrobun/bun or electrobun/view imports, BrowserWindow/BrowserView usage, updater flows, and distribution artifacts.

install
source · Clone the upstream repo
git clone https://github.com/NeverSight/learn-skills.dev
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/NeverSight/learn-skills.dev "$T" && mkdir -p ~/.claude/skills && cp -r "$T/data/skills-md/0xbigboss/claude-code/electrobun-best-practices" ~/.claude/skills/neversight-learn-skills-dev-electrobun-best-practices && rm -rf "$T"
manifest: data/skills-md/0xbigboss/claude-code/electrobun-best-practices/SKILL.md
source content

Electrobun Best Practices

Electrobun builds cross-platform desktop apps with TypeScript and Bun. This skill gives safe defaults, typed RPC patterns, and operational guidance for build/update/distribution.

Docs: https://blackboard.sh/electrobun/docs/

Pair with TypeScript Best Practices

Always load

typescript-best-practices
alongside this skill.

Version and Freshness

Electrobun APIs evolve quickly. Before relying on advanced options or platform-specific behavior, verify against current docs and CLI output.

Architecture

Electrobun apps run as Bun apps:

  • Bun process (main): imports from
    electrobun/bun
  • Browser context (views): imports from
    electrobun/view
  • Shared types: RPC schemas shared between both contexts

IPC between bun and browser contexts uses postMessage, FFI, and (in some paths) encrypted WebSockets.

Quick Start

bunx electrobun init
bun install
bun start

Recommended scripts:

{
  "scripts": {
    "start": "electrobun run",
    "dev": "electrobun dev",
    "dev:watch": "electrobun dev --watch",
    "build:dev": "bun install && electrobun build",
    "build:canary": "electrobun build --env=canary",
    "build:stable": "electrobun build --env=stable"
  }
}

Secure Defaults

Use this baseline for untrusted or third-party content:

import { BrowserWindow } from "electrobun/bun";

const win = new BrowserWindow({
  title: "External Content",
  url: "https://example.com",
  sandbox: true,                  // disables RPC, events still work
  partition: "persist:external",
});

win.webview.setNavigationRules([
  "^*",                          // block everything by default
  "*://example.com/*",           // allow only trusted domain(s)
  "^http://*",                   // enforce HTTPS
]);

win.webview.on("will-navigate", (e) => {
  console.log("nav", e.data.url, "allowed", e.data.allowed);
});

Security checklist:

  • Use
    sandbox: true
    for untrusted content.
  • Apply strict navigation allowlists.
  • Use separate
    partition
    values for isolation.
  • Validate all
    host-message
    payloads from
    <electrobun-webview>
    preload scripts.
  • Do not write to
    PATHS.RESOURCES_FOLDER
    at runtime; use
    Utils.paths.userData
    .

Typed RPC (Minimal Pattern)

// src/shared/types.ts
import type { RPCSchema } from "electrobun/bun";

export type MyRPC = {
  bun: RPCSchema<{
    requests: {
      getUser: { params: { id: string }; response: { name: string } };
    };
    messages: {
      logToBun: { msg: string };
    };
  }>;
  webview: RPCSchema<{
    requests: {
      updateUI: { params: { html: string }; response: boolean };
    };
    messages: {
      notify: { text: string };
    };
  }>;
};
// bun side
import { BrowserView, BrowserWindow } from "electrobun/bun";
import type { MyRPC } from "../shared/types";

const rpc = BrowserView.defineRPC<MyRPC>({
  handlers: {
    requests: {
      getUser: ({ id }) => ({ name: `user-${id}` }),
    },
    messages: {
      logToBun: ({ msg }) => console.log(msg),
    },
  },
});

const win = new BrowserWindow({
  title: "App",
  url: "views://mainview/index.html",
  rpc,
});

await win.webview.rpc.updateUI({ html: "<p>Hello</p>" });
// browser side
import { Electroview } from "electrobun/view";
import type { MyRPC } from "../shared/types";

const rpc = Electroview.defineRPC<MyRPC>({
  handlers: {
    requests: {
      updateUI: ({ html }) => {
        document.body.innerHTML = html;
        return true;
      },
    },
    messages: {
      notify: ({ text }) => console.log(text),
    },
  },
});

const electroview = new Electroview({ rpc });
await electroview.rpc.request.getUser({ id: "1" });
electroview.rpc.send.logToBun({ msg: "hello" });

Events and Shutdown

Use

before-quit
for shutdown cleanup instead of relying on
process.on("exit")
for async work.

import Electrobun from "electrobun/bun";

Electrobun.events.on("before-quit", async (e) => {
  await saveState();
  // e.response = { allow: false }; // optional: cancel quit
});

Important caveat:

  • Linux currently has a caveat where some system-initiated quit paths (for example Ctrl+C/window-manager/taskbar quit) may not fire
    before-quit
    . Programmatic quit via
    Utils.quit()
    /
    process.exit()
    is reliable.

Common Patterns

  • Keyboard shortcuts (copy/paste/undo): define an Edit
    ApplicationMenu
    with role-based items.
  • Tray-only app: set
    runtime.exitOnLastWindowClosed: false
    , then drive UX from
    Tray
    .
  • Multi-account isolation: use separate
    partition
    values per account.
  • Chromium consistency: set
    bundleCEF: true
    and
    defaultRenderer: "cef"
    in platform config.

Troubleshooting

  • RPC calls fail unexpectedly:
    • Check whether the target webview is sandboxed (
      sandbox: true
      disables RPC).
    • Confirm shared RPC types match both bun and browser handlers.
  • Navigation blocks legitimate URLs:
    • Review
      setNavigationRules
      ordering; last match wins.
    • Keep
      ^*
      first only when you intentionally run strict allowlist mode.
  • Updater says no update:
    • Verify
      release.baseUrl
      and uploaded
      artifacts/
      naming (
      {channel}-{os}-{arch}-...
      ).
    • Confirm channel/build env alignment (
      canary
      vs
      stable
      ).
  • User sessions leak across accounts:
    • Use explicit per-account partitions and manage cookies via
      Session.fromPartition(...)
      .
  • Build hooks not running:
    • Ensure hook paths are correct and executable via Bun.
    • Inspect hook env vars (for example
      ELECTROBUN_BUILD_ENV
      ,
      ELECTROBUN_OS
      ,
      ELECTROBUN_ARCH
      ).

Reference Files