3x-ui/frontend
MHSanaei d6f42b3395
refactor(frontend): port inbounds to react+ts and drop vue toolchain
Step 9 — the last entry. Ports the inbounds entry: page shell, list with
desktop table + mobile cards, info modal, qr-code modal, share-link
helpers, and the protocol-aware form modal (basics / protocol /
stream / security / sniffing / advanced JSON). useInbounds replaces
the Vue composable with WebSocket-driven traffic + client-stats merge.

Inbound and DBInbound models stay in JS so the class-driven form keeps
its mutation API; instance access is typed loosely inside the form to
match. FinalMaskForm/JsonEditor/TextModal/PromptModal/InfinityIcon are
the last shared bits to flip; their .vue counterparts go too.

Toolchain cleanup now that no entry needs Vue: drop plugin-vue from
vite.config, remove the .vue lint block + parser, prune vue / vue-i18n
/ ant-design-vue / @ant-design/icons-vue / vue3-persian-datetime-picker
/ moment-jalaali override from package.json, and switch utils/index.js
to import { message } from 'antd' instead of ant-design-vue.
2026-05-21 23:35:23 +02:00
..
src refactor(frontend): port inbounds to react+ts and drop vue toolchain 2026-05-21 23:35:23 +02:00
.gitignore Vue3 migration (#4198) 2026-05-09 17:47:35 +02:00
api-docs.html refactor(frontend): port api-docs to react+ts 2026-05-21 21:26:28 +02:00
clients.html refactor(frontend): port clients to react+ts 2026-05-21 22:03:31 +02:00
eslint.config.js refactor(frontend): port inbounds to react+ts and drop vue toolchain 2026-05-21 23:35:23 +02:00
inbounds.html refactor(frontend): port inbounds to react+ts and drop vue toolchain 2026-05-21 23:35:23 +02:00
index.html refactor(frontend): port index dashboard to react+ts 2026-05-21 22:20:09 +02:00
login.html refactor(frontend): port login to react+ts 2026-05-21 21:19:52 +02:00
nodes.html refactor(frontend): port nodes to react+ts 2026-05-21 21:34:46 +02:00
package-lock.json refactor(frontend): port inbounds to react+ts and drop vue toolchain 2026-05-21 23:35:23 +02:00
package.json refactor(frontend): port inbounds to react+ts and drop vue toolchain 2026-05-21 23:35:23 +02:00
README.md Vue3 migration (#4198) 2026-05-09 17:47:35 +02:00
settings.html refactor(frontend): port settings to react+ts 2026-05-21 21:48:15 +02:00
subpage.html refactor(frontend): port subpage to react+ts 2026-05-21 21:19:38 +02:00
tsconfig.json chore(frontend): add react+typescript toolchain alongside vue 2026-05-21 21:19:09 +02:00
vite.config.js refactor(frontend): port inbounds to react+ts and drop vue toolchain 2026-05-21 23:35:23 +02:00
xray.html refactor(frontend): port xray to react+ts 2026-05-21 22:56:36 +02:00

3x-ui frontend

Vue 3 + Ant Design Vue 4 + Vite. Multi-page app — one HTML entry per panel route — built into ../web/dist/ and embedded into the Go binary via embed.FS.

Dev

npm install
npm run dev

Vite serves on http://localhost:5173/. API calls and /panel/* routes proxy to the Go panel at http://localhost:2053/, so start the Go panel first (go run main.go) and then Vite.

The proxy auto-rewrites /panel, /panel/settings, /panel/inbounds, /panel/xray to the matching Vite-served HTML in dev mode (see MIGRATED_ROUTES in vite.config.js), so the sidebar's production-style links work without round-tripping through Go.

Production build

npm run build

Outputs to ../web/dist/ (HTML at the root, hashed JS/CSS under assets/). The Go binary embeds this directory at compile time and web/controller/dist.go serves the per-page HTML.

Lint

npm run lint

ESLint 10 with eslint.config.js (flat config) — vue3-recommended plus a few rule overrides for the project's formatting style.

Layout

frontend/
├── *.html                 # Vite entry HTML, one per panel route
├── eslint.config.js
├── vite.config.js
└── src/
    ├── entries/           # Per-page bootstrap (createApp + mount)
    ├── pages/             # One folder per route, each with the page
    │   ├── index/         # component + helpers + sub-components
    │   ├── login/
    │   ├── inbounds/
    │   ├── xray/
    │   ├── settings/
    │   └── sub/
    ├── components/        # Cross-page Vue components
    ├── composables/       # Reusable reactive logic (useTheme, …)
    ├── api/               # Axios setup, CSRF interceptor
    ├── i18n/              # vue-i18n init (locales live in web/translation/)
    ├── models/            # Inbound, Outbound, Status, … domain classes
    └── utils/             # HttpUtil, ObjectUtil, LanguageManager, …

Adding a new page

  1. Add frontend/<page>.html referencing /src/entries/<page>.js.
  2. Add src/entries/<page>.js that imports the page component and mounts it.
  3. Add the page component under src/pages/<page>/.
  4. Register the entry in rollupOptions.input in vite.config.js.
  5. If the page is reachable from the sidebar at /panel/<route>, add it to MIGRATED_ROUTES so the dev proxy serves the Vite HTML.
  6. Wire the Go controller to serveDistPage(c, "<page>.html").