3x-ui/frontend/src/components/FinalMaskForm.tsx
Sanaei dc37f9b731
Migrate frontend models/api/utils to TypeScript and modernize AntD theming (#4563)
* refactor(frontend): port api/* and reality-targets to TypeScript

Phase 1 of the JS→TS migration: convert three small, isolated files
(axios-init, websocket, reality-targets) to typed sources so future
phases can lean on their interfaces.

- api/axios-init.ts: typed CSRF cache, interceptors, request retry
- api/websocket.ts: typed listener map, message envelope guard,
  reconnect timer
- models/reality-targets.ts: RealityTarget interface, readonly list
- env.d.ts: minimal qs module shim (stringify/parse)
- consumers: drop ".js" extension from @/api imports

* refactor(frontend): port utils/index to TypeScript

Phase 2 of the JS→TS migration: convert the 858-line utility module
that 30+ pages and hooks depend on.

- Msg<T = any> generic with success/msg/obj shape preserved
- HttpUtil get/post/postWithModal generic over response shape
- RandomUtil, Wireguard, Base64 fully typed
- SizeFormatter/CPUFormatter/TimeFormatter/NumberFormatter typed
- ColorUtils.usageColor returns 'green'|'orange'|'red'|'purple' union
- LanguageManager.supportedLanguages readonly typed
- IntlUtil.formatDate/formatRelativeTime accept null/undefined
- ObjectUtil.clone/deepClone/cloneProps/equals kept as `any`-shaped
  to preserve the prior JS contract used by class-instance callers
  (AllSetting.cloneProps(this, data), etc.)

* refactor(frontend): port models/outbound to TypeScript (hybrid typing)

Phase 4 of the JS→TS migration: rename outbound.js to outbound.ts and
make it compile under strict mode with a minimal hybrid type pass.

- Enum-like constants kept as typed objects (Protocols, SSMethods, …)
- Top-level DNS helpers strictly typed
- CommonClass gets [key: string]: any so all subclasses can keep their
  loose this.foo = bar assignments without per-field declarations
- Constructor / fromJson / toJson signatures typed as any to preserve
  the prior JS contract used by consumers and parsers
- Outbound declares static fields for the dynamically-attached Settings
  subclasses (Settings, FreedomSettings, VmessSettings, …)
- urlParams.get() results that feed parseInt now use the non-null
  assertion since the surrounding has() check already guards them
- File-level eslint-disable for no-explicit-any/no-var/prefer-const to
  keep the JS-derived code building without churn

* refactor(frontend): port models/inbound to TypeScript (hybrid typing)

Phase 5 of the JS→TS migration. Same hybrid approach as outbound.ts:
constants typed strictly, classes get [key: string]: any from
XrayCommonClass, constructor / fromJson / toJson signatures use any.

- XrayCommonClass gains [key: string]: any plus typed static helpers
  (toJsonArray, fallbackToJson, toHeaders, toV2Headers)
- TcpStreamSettings/TlsStreamSettings/RealityStreamSettings/Inbound
  declare static fields for their dynamically-attached subclasses
  (TcpRequest, TcpResponse, Cert, Settings, ClientBase, Vmess/VLESS/
  Trojan/Shadowsocks/Hysteria/Tunnel/Mixed/Http/Wireguard/TunSettings)
- All gen*Link, applyXhttpExtra*, applyExternalProxyTLS*, applyFinalMask*
  and related helpers explicitly any-typed
- Constructor positional client-args (email, limitIp, totalGB, …) typed
  as optional any across Vmess/VLESS/Trojan/Shadowsocks/Hysteria.VMESS|
  VLESS|Trojan|Shadowsocks|Hysteria
- File-level eslint-disable for no-explicit-any/prefer-const/
  no-case-declarations/no-array-constructor to silence churn without
  changing behavior

* refactor(frontend): port models/dbinbound to TypeScript

Phase 6 — final phase of the JS→TS migration. Frontend src/ no
longer contains any *.js files.

- DBInbound declares all fields explicitly (id, userId, up, down,
  total, …, nodeId, fallbackParent) with proper types
- _expiryTime getter/setter typed against dayjs.Dayjs
- coerceInboundJsonField takes unknown, returns any
- Private cache fields (_cachedInbound, _clientStatsMap) declared
- Consumers (InboundFormModal, InboundsPage, useInbounds): drop ".js"
  extension from @/models/dbinbound imports

* refactor(frontend): drop .js extensions from TS-resolved imports

Cleanup after the JS→TS migration:

- All consumers that imported @/models/{inbound,outbound,dbinbound}.js
  now drop the .js extension (TS module resolution lands on the .ts
  file automatically)
- eslint.config.js: remove the **/*.js block since the only remaining
  JS file under src/ is endpoints.js (build-script consumed only) and
  js.configs.recommended already covers it correctly

* refactor(frontend): tighten inbound.ts cleanup wins

Checkpoint before the full any → typed pass:
- Wrap 15 case bodies in braces (no-case-declarations)
- Convert 14 let → const in genLink helpers (prefer-const)
- new Array() → [] for shadowsocks passwords (no-array-constructor)
- XrayCommonClass: HeaderEntry, FallbackEntry, JsonObject interfaces;
  fromJson/toV2Headers/toHeaders typed against them; static methods
  return JsonObject / HeaderEntry[] instead of any
- Reduce file-level eslint-disable scope from 4 rules to just
  no-explicit-any (the only one still needed)

* refactor(frontend): drop eslint-disable from models/dbinbound

Replace `any` with explicit domain types:
- `coerceInboundJsonField` returns `Record<string, unknown>` (settings/streamSettings/sniffing are always objects).
- Add `RawJsonField`, `ClientStats`, `FallbackParentRef`, `DBInboundInit` types.
- `_cachedInbound: Inbound | null`, `toInbound(): Inbound`.
- `getClientStats(email): ClientStats | undefined`.
- `genInboundLinks(): string` (matches actual return from Inbound.genInboundLinks).
- Constructor now accepts `DBInboundInit`.

* refactor(frontend): drop eslint-disable from InboundsPage

Type all callbacks against DBInbound from @/models/dbinbound:
- state setters use DBInbound | null
- helpers (projectChildThroughMaster, checkFallback, findClientIndex,
  exportInboundLinks, etc.) take DBInbound
- drop `(dbInbounds as any[])` casts; useInbounds already returns DBInbound[]
- introduce ClientMatchTarget for findClientIndex's `client` param
- tighten DBInbound.clientStats to ClientStats[] (default [])
- single boundary cast at <InboundList onRowAction=> to bridge
  InboundList's narrower DBInboundRecord (cleanup belongs with InboundList)

* refactor(frontend): drop file-level eslint-disable from utils/index

- ObjectUtil.clone/deepClone become generic <T>
- cloneProps/delProps accept `object` (cast internally to AnyRecord)
- equals accepts `unknown` with proper narrowing
- ColorUtils.usageColor narrows data/threshold to `number`; total widened
  to `number | { valueOf(): number } | null | undefined` so Dayjs works
- Utils.debounce replaces `const self = this` with lexical arrow
  closure (no-this-alias clean)
- InboundList._expiryTime narrowed from `unknown` to `{ valueOf(): number } | null`
- Single-line eslint-disable remains on `Msg<T = any>` and HttpUtil
  generic defaults (idiomatic API envelope; changing default to unknown
  cascades through 34 consumer files)

* refactor(frontend): drop eslint-disable from OutboundFormModal field section

Replace `type OB = any` with `type OB = Outbound`. Body code still
sees protocol fields as `any` via Outbound's inherited [key: string]: any
index signature (CommonClass) — that escape hatch will narrow as
Phase 6 tightens outbound.ts itself.

The intentional `// eslint-disable-next-line` on `useRef<any>(null)`
at line 72 stays — out of scope per plan.

* refactor(frontend): drop file-level eslint-disable from InboundFormModal

Add minimal local interfaces for protocol-specific shapes the form reads:
- StreamLike, TlsCert, VlessClient, ShadowsocksClient, HttpAccount,
  WireguardPeer (replace with real exports from inbound.ts as Phase 7
  exports them).
- Props typed as DBInbound | null + DBInbound[].
- Drop unnecessary `(Inbound as any).X`, `(RandomUtil as any).X`,
  `(Wireguard as any).X`, `(DBInbound as any)(...)` casts — they are
  already typed classes; only `Inbound.Settings`/`Inbound.HttpSettings`
  remain `any` via static field on Inbound (will tighten in Phase 7).
- inboundRef/dbFormRef retain single-line `// eslint-disable-next-line`
  for `useRef<any>(null)` — nullable narrowing across ~30 callsites
  exceeds Phase 5 scope.
- payload locals typed Record<string, unknown>; setAdvancedAllValue
  parses JSON into a narrowed object instead of `let parsed: any`.

* refactor(frontend): narrow outbound.ts eslint-disable to no-explicit-any only

- Fix all 36 prefer-const violations: convert never-reassigned `let` to
  `const`; for mixed-mutability destructuring (fromParamLink,
  fromHysteriaLink) split into separate `const`/`let` declarations
  by index instead of destructuring.
- Fix both no-var violations: `var stream` / `var settings` → `let`.
- File still carries `/* eslint-disable @typescript-eslint/no-explicit-any */`
  because tightening 223 `any` uses requires removing CommonClass's
  `[key: string]: any` escape hatch and reshaping ~30 dynamically-attached
  subclass patterns into named classes — multi-hour architectural work
  tracked as Phase 7's twin for outbound.

* refactor(frontend): align sub page chrome with login + AntD defaults

- Theme + language buttons now both use AntD `<Button shape="circle"
  size="large" className="toolbar-btn">` with TranslationOutlined and
  the SVG theme icon — identical hover/border behaviour.
- Language popover content switched from hand-rolled `<ul.lang-list>`
  to AntD `<Menu mode="vertical" selectable />`; gains native
  hover/keyboard nav + active highlight.
- Drop `.info-table` `!important` border overrides (8 selectors) so
  Descriptions inherits the AntD theme border colour.
- Drop `.qr-code` padding/background/border-radius overrides; only
  `cursor: pointer` remains (QRCode handles padding/bg itself).
- Remove now-unused `.theme-cycle`, `.lang-list`, `.lang-item*`,
  `.lang-select`, `.settings-popover` rules.

* refactor(frontend): drop CustomStatistic wrapper, move overrides to theme tokens

- Delete `<CustomStatistic>` (a pass-through wrapper over <Statistic>)
  and its unscoped global `.ant-statistic-*` CSS overrides; consumers
  (IndexPage, ClientsPage, InboundsPage, NodesPage) now import AntD
  `<Statistic>` directly.
- Add Statistic component tokens to ConfigProvider so the title (11px)
  and content (17px) font sizes still apply, without `!important`
  global selectors.
- Move dark / ultra-dark card border colours from `body.dark .ant-card`
  + `html[data-theme='ultra-dark'] .ant-card` selectors into Card
  `colorBorderSecondary` tokens; page-cards.css now only carries the
  custom radius/shadow/transition that has no token equivalent.
- Simplify XrayStatusCard badge: remove the custom `xray-pulse` dot
  keyframe and per-state ring-colour overrides; AntD `<Badge
  status="processing" color={…}>` already pulses the ring in the same
  colour, no extra CSS needed.

* refactor(frontend): modernize login page with AntD primitives

- Theme cycle button switched from `<button.theme-cycle>` + custom CSS
  to AntD `<Button shape="circle" className="toolbar-btn">` (matches
  sub page chrome already established).
- Theme icons switched from hand-rolled inline SVG (sun, moon,
  moon+star) to AntD `<SunOutlined />`, `<MoonOutlined />`,
  `<MoonFilled />` for the three light / dark / ultra-dark states.
- Language popover content switched from `<ul.lang-list>` +
  `<button.lang-item>` to AntD `<Menu mode="vertical" selectable />`
  with `selectedKeys=[lang]`; native hover / keyboard nav / active
  highlight come for free.
- Drop CSS for `.theme-cycle`, `.lang-list`, `.lang-item*` (now unused).
  `.toolbar-btn` retained since it sizes both circular buttons.

* refactor(frontend): switch sub page theme icons to AntD primitives

Replace the three hand-rolled SVG theme icons (sun, moon, moon+star)
with AntD `<SunOutlined />`, `<MoonOutlined />`, `<MoonFilled />`
for the light / dark / ultra-dark states. Switch the theme `<Button>`
to use the `icon` prop instead of children so it renders the same
way as the language button. Drop `.toolbar-btn svg` CSS — no longer
needed once the icon comes from AntD.

* refactor(frontend): drop !important overrides from pages CSS (Clients + Log modals + Settings tabs)

- ClientsPage: pagination size-changer `min-width !important` removed;
  the 3-level selector specificity already beats AntD's defaults.
  Scope `body.dark .client-card` to `.clients-page.is-dark .client-card`
  (avoid leaking into other pages).
- LogModal + XrayLogModal: move the mobile full-screen tweaks
  (`top: 0`, `padding-bottom: 0`, `max-width: 100vw`) from `!important`
  class rules to the Modal's `style` prop; keep `.ant-modal-content`
  / `.ant-modal-body` overrides as plain CSS via the className.
- SubscriptionFormatsTab: drop `display: block !important` on
  `.nested-block` — div is already block by default.
- TwoFactorModal: drop `padding/background/border-radius !important`
  on `.qr-code`; AntD QRCode handles those itself.

* refactor(frontend): scope dark overrides and switch list borders to AntD CSS variables

Scope page-level dark overrides:
- inbounds/InboundList: scope `.ant-table` border-radius rules and the
  mobile @media `.ant-card-*` tweaks to `.inbounds-page` (were global
  and leaked into other pages); scope `.inbound-card` dark variant to
  `.inbounds-page.is-dark`.
- nodes/NodeList: scope `.node-card` dark to `.nodes-page.is-dark`.
- xray/RoutingTab, OutboundsTab: scope `.rule-card`, `.criterion-chip`,
  `.criterion-more`, `.address-pill` dark to `.xray-page.is-dark`.

Modernize list borders to use AntD CSS vars instead of body.dark forks:
- index/BackupModal, PanelUpdateModal, VersionModal: replace
  hard-coded `rgba(5,5,5,0.06)` + `body.dark`/`html[data-theme]`
  override pairs with `var(--ant-color-border-secondary)`; replace
  custom text colours with `var(--ant-color-text)` /
  `var(--ant-color-text-tertiary)`.
- xray/DnsPresetsModal: same border-color treatment.
- xray/NordModal, WarpModal: collapse `.row-odd` light + `body.dark`
  pair into a single neutral `rgba(128,128,128,0.06)` that works on
  both themes; scope under `.nord-data-table` / `.warp-data-table`.

* refactor(frontend): switch shared components CSS to AntD CSS variables

Replace body.dark / html[data-theme] forks with AntD CSS variables
in shared components (work in both light and dark, scale to ultra):
- SettingListItem: borders + text colours via
  `--ant-color-border-secondary`, `--ant-color-text`,
  `--ant-color-text-tertiary`.
- InputAddon: bg/border/text via `--ant-color-fill-tertiary`,
  `--ant-color-border`, `--ant-color-text`.
- JsonEditor: host border/bg via `--ant-color-border`,
  `--ant-color-bg-container`; focus border via `--ant-color-primary`.
- Sparkline (SVG): grid/text colours via `--ant-color-text*`
  and `--ant-color-border-secondary`; only the tooltip drop-shadow
  retains a body.dark fork (filter opacity needs explicit value).

* refactor(frontend): swap custom Sparkline SVG for Recharts AreaChart

Replace the 368-line hand-rolled SVG sparkline (with manual
ResizeObserver, gradient/shadow/glow filters, grid + ticks + tooltip,
custom Y-axis label thinning) with a thin Recharts `<AreaChart>`
wrapper that keeps the same prop API.

- Preserved props: data, labels, height, stroke, strokeWidth,
  maxPoints, showGrid, fillOpacity, showMarker, markerRadius,
  showAxes, yTickStep, tickCountX, showTooltip, valueMin, valueMax,
  yFormatter, tooltipFormatter.
- Dropped: `vbWidth`, `gridColor`, `paddingLeft/Right/Top/Bottom` —
  Recharts' ResponsiveContainer handles width, and margins are wired
  to whether axes are visible. Removed the unused `vbWidth` prop from
  SystemHistoryModal, XrayMetricsModal, NodeHistoryPanel callsites.
- Tooltip, grid, and axis text now use AntD CSS variables for
  automatic light/dark adaptation; replaced the SVG body.dark forks
  in Sparkline.css with a single 5-line stylesheet.
- Bundle: vendor +~100KB gzip (Recharts + its d3 deps), trade-off
  for less custom chart code to maintain and a more standard API
  for future charts (multi-series, brush, etc.).

* build(frontend): split Recharts + d3 deps into vendor-recharts chunk

Pulls Recharts (~75KB gzip) and its d3-shape/array/color/path/scale
+ victory-vendor deps out of the catch-all vendor chunk so they
load on demand on the three pages that use Sparkline
(SystemHistoryModal, XrayMetricsModal, NodeHistoryPanel) and cache
independently from the rest of the panel JS.

* refactor(frontend): drop body.dark forks in favor of AntD CSS variables

- ClientInfoModal/InboundInfoModal: link-panel-text and link-panel-anchor now use
  var(--ant-color-fill-tertiary) and color-mix on --ant-color-primary, removing
  the body.dark light/dark background pair.
- InboundFormModal: advanced-panel uses --ant-color-border-secondary and
  --ant-color-fill-quaternary; body.dark/html[data-theme='ultra-dark'] pair gone.
- CustomGeoSection: custom-geo-count, custom-geo-ext-code, custom-geo-copyable:hover
  use --ant-color-fill-tertiary/-secondary; body.dark forks gone.
- SystemHistoryModal: cpu-chart-wrap collapsed from three theme-specific gradients
  into one using color-mix on --ant-color-primary and --ant-color-fill-quaternary.
- page-cards.css: body.dark / html[data-theme='ultra-dark'] selectors renamed to
  page-scoped .is-dark / .is-dark.is-ultra, keeping the same shadow tuning but
  consistent with the page-scoping convention used elsewhere.

* refactor(sidebar): modernize AppSidebar with AntD CSS variables and icons

- Replace hardcoded rgba(0,0,0,X) colors with var(--ant-color-text)
  and var(--ant-color-text-secondary) so light/dark adapt automatically.
- Replace rgba(128,128,128,0.15) borders with var(--ant-color-border-secondary)
  and rgba(128,128,128,0.18) backgrounds with var(--ant-color-fill-tertiary).
- Drop all body.dark/html[data-theme='ultra-dark'] color forks for
  .drawer-brand, .sider-brand, .drawer-close, .sidebar-theme-cycle,
  .sidebar-donate (CSS variables already adapt).
- Drop the body.dark Drawer background !important pair; AntD's
  colorBgElevated token from the dark algorithm handles it now.
- Replace inline sun/moon SVGs in ThemeCycleButton with AntD's
  SunOutlined/MoonOutlined/MoonFilled to match LoginPage/SubPage.
- Convert .sidebar-theme-cycle hover and the menu item selected/hover
  highlights from hardcoded #4096ff to color-mix on --ant-color-primary,
  keeping !important on menu rules to beat AntD's CSS-in-JS specificity.

* refactor(frontend): swap hardcoded AntD palette colors for CSS variables

The dot/badge/pill styles still hardcoded AntD's default palette values
(#52c41a, #1677ff, #ff4d4f, #fa8c16, #ff4d4f). Replace each with its
semantic --ant-color-* equivalent so they auto-adapt to any theme
customization through ConfigProvider.

- ClientsPage: .dot-green/.dot-blue/.dot-red/.dot-orange/.dot-gray now
  use --ant-color-success / -primary / -error / -warning / -text-quaternary.
  .bulk-count / .client-card / .client-card.is-selected backgrounds use
  color-mix on --ant-color-primary and --ant-color-fill-quaternary, which
  also let the body-dark .client-card fork go away.
- XrayMetricsModal: .obs-dot is-alive/is-dead and its pulse keyframe now
  build their box-shadow tint via color-mix on --ant-color-success and
  --ant-color-error instead of rgba literals.
- IndexPage: .action-update warning color uses --ant-color-warning.
- OutboundsTab: .outbound-card border, .address-pill background, and
  .mode-badge tint now use AntD CSS variables; the .xray-page.is-dark
  .address-pill fork is gone.
- InboundFormModal/InboundsPage/ClientBulkAddModal: drop the stale
  `, #1677ff`/`, #1890ff` fallbacks on var(--ant-color-primary), and
  switch .danger-icon to --ant-color-error.

The teal/cyan brand colors (#008771, #3c89e8, #e04141) used by traffic
and pill rows are intentionally kept hardcoded — they are brand-specific
shades, not AntD palette colors.

* refactor(frontend): swap neutral gray rgba literals for AntD CSS variables

Across 12 files the same neutral grays kept reappearing — rgba(128,128,128,
0.06|0.08|0.12|0.15|0.18|0.2|0.25) for borders, dividers, and subtle
backgrounds. Each maps cleanly to an AntD CSS variable that already
adapts to light/dark and to any theme customization through ConfigProvider:

- 0.12–0.18 borders → var(--ant-color-border-secondary)
- 0.2–0.25 borders → var(--ant-color-border)
- 0.06–0.08 backgrounds → var(--ant-color-fill-tertiary)
- 0.02–0.03 card surfaces → var(--ant-color-fill-quaternary)

Card surfaces (InboundList .inbound-card, NodeList .node-card) had a
light/dark fork pair — the variable covers both, so the .is-dark .card
override is gone.

RoutingTab .rule-card.drop-before/after used hardcoded #1677ff for the
inset focus shadow; replaced with var(--ant-color-primary) so reordering
indicators follow the theme primary.

ClientsPage bucketBadgeColor returned hex literals (#ff4d4f, #fa8c16,
#52c41a, rgba gray) for a Badge color prop. Switched to status="error"|
"warning"|"success"|"default" so the dot color now comes from AntD's
semantic palette directly.

* refactor(xray): collapse RoutingTab dark forks into AntD CSS variables

- .criterion-more bg light/dark fork → var(--ant-color-fill-tertiary)
- .xray-page.is-dark .rule-card and .criterion-chip overrides removed;
  the rules already use --bg-card and --ant-color-fill-tertiary that
  adapt to the theme on their own.

* refactor(frontend): inline style hex literals and Alert icon redundancy

- FinalMaskForm: five DeleteOutlined icons used rgb(255,77,79) inline;
  swap for var(--ant-color-error) so they follow theme customization.
- NodesPage: CheckCircleOutlined / CloseCircleOutlined statistic prefixes
  switch to var(--ant-color-success) / -error.
- NodeList: ExclamationCircleOutlined warning icons (two callsites) now
  use var(--ant-color-warning).
- BasicsTab: four <Alert type="warning"> blocks shipped a custom
  ExclamationCircleFilled icon styled to match the warning palette —
  exactly the icon and color AntD Alert renders for type="warning" by
  default. Replace the icon prop with showIcon and drop the now-unused
  ExclamationCircleFilled import.
- JsonEditor: focus-within box-shadow tint now uses color-mix on
  --ant-color-primary instead of an rgba(22,119,255,0.1) literal.

* refactor(logs): collapse log-container dark forks to AntD CSS variables

LogModal and XrayLogModal each had a body.dark fork that overrode the
log container's background, border-color, and text color in addition
to the --log-* severity tokens. Background/border/color all map cleanly
to var(--ant-color-fill-tertiary) / var(--ant-color-border) /
var(--ant-color-text) which already adapt to the theme, so only the
severity color tokens remain inside the dark/ultra-dark blocks.

* refactor(xray): drop stale --ant-primary-color fallbacks and hex literals

- RoutingTab .drop-before/.drop-after box-shadow: #1677ff → var(--ant-color-primary)
- OutboundFormModal .random-icon: drop the --ant-primary-color/#1890ff
  pair (the old AntD v4 token name with stale fallback) for the v6
  --ant-color-primary; .danger-icon hex #ff4d4f → var(--ant-color-error).
- XrayPage .restart-icon: same drop of the --ant-primary-color fallback.

These were all leftovers from the AntD v4 → v6 rename — the v6
--ant-color-primary is already populated by ConfigProvider, so the
fallback hex was dead code that would only trigger if AntD wasn't
mounted.

* refactor(frontend): consolidate margin utility classes into one stylesheet

Page CSS files each carried their own copies of the same atomic margin
utilities (.mt-4, .mt-8, .mb-12, .ml-8, .my-10, ...). The definitions
were identical everywhere they appeared, with each file holding only
the subset it happened to need.

Move all of them into a single styles/utils.css imported once from
main.tsx, and delete the per-page copies from InboundFormModal,
CustomGeoSection, PanelUpdateModal, VersionModal, BasicsTab, NordModal,
OutboundFormModal, and WarpModal. The classes are available globally
on the panel app; login.tsx and subpage.tsx entries do not consume any
of them so they stay untouched.

* refactor(frontend): consolidate shared page-shell rules into one stylesheet

Every panel page CSS file repeated the same wrapper boilerplate — the
--bg-page/--bg-card token triples for light/dark/ultra-dark, the
min-height + background root rule, the .ant-layout transparent reset,
the .content-shell transparent reset, and the .loading-spacer min-height.
That's ~30 identical lines duplicated across IndexPage, ClientsPage,
InboundsPage, XrayPage, SettingsPage, NodesPage, and ApiDocsPage.

Move all of it into styles/page-shell.css and import it once from
main.tsx alongside utils.css and page-cards.css. Each page CSS file
now only contains genuinely page-specific rules (content-area padding
overrides, page-specific tokens like ApiDocs's Swagger --sw-* set).

Also drop the per-page `import '@/styles/page-cards.css'` statements
from the 7 page tsx files now that main.tsx loads it globally.

Net: -211 deleted, +6 inserted in the touched files, plus the new
page-shell.css. .zero-margin (Divider override used by Nord/Warp
modals) folded into utils.css alongside the margin classes.

* refactor(frontend): move default content-area padding to page-shell.css

After page-shell.css landed, six of the seven panel pages still kept an
identical `.X-page .content-area { padding: 24px }` desktop rule, plus
three of them kept an identical `padding: 8px` mobile rule. Hoist both
defaults into page-shell.css under a single 6-page selector group and
delete the per-page copies.

What stays page-specific:
- IndexPage keeps its mobile override (padding 12px + padding-top: 64px
  for the fixed drawer handle clearance).
- ApiDocsPage keeps its tighter desktop padding (16px) and its own
  mobile padding-top: 56px.

Settings .ldap-no-inbounds also switches from #999 to
var(--ant-color-text-tertiary) for theme adaptation.

* refactor(frontend): hoist .header-row, .icons-only, .summary-card to page-shell.css

Settings and Xray pages both carried identical .header-row /
.header-actions / .header-info rules and an identical six-rule
.icons-only block that styles tabbed page navigation. Clients, Inbounds,
and Nodes all carried identical .summary-card padding rules with the
same mobile reduction. None of these are page-specific.

Consolidate:
- .header-row family → page-shell scoped to .settings-page, .xray-page
- .icons-only family → page-shell global (the class is a deliberate
  opt-in marker, no scope needed)
- .summary-card → page-shell scoped to .clients-page, .inbounds-page,
  .nodes-page (also fixes InboundsPage's missing scope — its rule was
  global and would have matched stray .summary-card uses elsewhere)

InboundsPage.css and NodesPage.css became empty after the move so the
files and their per-page imports are deleted.

* refactor(frontend): hoist .random-icon to utils.css

Three form modals each carried identical .random-icon styles (small
primary-tinted icon next to randomizable inputs):
  ClientBulkAddModal, InboundFormModal, OutboundFormModal

Single definition lives in utils.css now. ClientBulkAddModal.css was
just this one rule, so the file and its import are deleted along the way.

.danger-icon is left per file — the margin-left differs slightly
between InboundFormModal (6px) and OutboundFormModal (8px), so it
stays as a page-local rule rather than getting averaged into utils.css.

* refactor(frontend): hoist .danger-icon to utils.css and use it everywhere

InboundFormModal (margin-left 6px) and OutboundFormModal (margin-left
8px) each carried their own .danger-icon, and FinalMaskForm wrote the
same color/cursor/marginLeft trio inline five times. Unify on a single
.danger-icon in utils.css with margin-left: 8px — matching the more
generous OutboundFormModal value — and:
- Drop the per-file .danger-icon copies from InboundFormModal.css and
  OutboundFormModal.css.
- Replace the five inline style props in FinalMaskForm.tsx with
  className="danger-icon".

The visible change is a 2px wider gap to the right of the delete icons
on InboundFormModal's protocol/peer dividers.
2026-05-25 14:34:53 +02:00

738 lines
24 KiB
TypeScript

import { useMemo } from 'react';
import { Button, Divider, Form, Input, InputNumber, Select, Switch } from 'antd';
import { DeleteOutlined, PlusOutlined, ReloadOutlined } from '@ant-design/icons';
import { RandomUtil } from '@/utils';
import { Protocols } from '@/models/outbound';
interface StreamShape {
network?: string;
kcp?: { mtu?: number };
finalmask: {
tcp?: MaskRow[];
udp?: MaskRow[];
enableQuicParams?: boolean;
quicParams?: QuicParams;
};
addTcpMask: (type?: string) => void;
delTcpMask: (index: number) => void;
addUdpMask: (type?: string) => void;
delUdpMask: (index: number) => void;
}
interface MaskRow {
type: string;
settings: Record<string, unknown>;
_getDefaultSettings: (type: string, settings: Record<string, unknown>) => Record<string, unknown>;
}
interface ItemRow {
type: string;
packet: string | unknown[];
delay?: number | string;
rand?: number | string;
randRange?: string;
}
interface QuicParams {
congestion: string;
debug?: boolean;
brutalUp?: number | string;
brutalDown?: number | string;
hasUdpHop?: boolean;
udpHop?: { ports: string; interval: string | number };
maxIdleTimeout?: number;
keepAlivePeriod?: number;
disablePathMTUDiscovery?: boolean;
maxIncomingStreams?: number;
initStreamReceiveWindow?: number;
maxStreamReceiveWindow?: number;
initConnectionReceiveWindow?: number;
maxConnectionReceiveWindow?: number;
}
interface FinalMaskFormProps {
stream: StreamShape;
protocol: string;
onChange: () => void;
}
function changeMaskType(mask: MaskRow, type: string) {
mask.type = type;
mask.settings = mask._getDefaultSettings(type, {});
}
function changeItemType(item: ItemRow, type: string) {
item.type = type;
if (type === 'base64') item.packet = RandomUtil.randomBase64();
else if (type === 'array') {
item.rand = 0;
item.packet = [];
} else item.packet = '';
}
function newClientServerItem(): ItemRow {
return { delay: 0, rand: 0, randRange: '0-255', type: 'array', packet: [] };
}
function newUdpClientServerItem(): ItemRow {
return { rand: 0, randRange: '0-255', type: 'array', packet: [] };
}
function newNoiseItem(): ItemRow {
return { rand: '1-8192', randRange: '0-255', type: 'array', packet: [], delay: '10-20' };
}
export default function FinalMaskForm({ stream, protocol, onChange }: FinalMaskFormProps) {
const isHysteria = protocol === Protocols.Hysteria || protocol === 'hysteria';
const network = stream?.network || '';
const showTcp = useMemo(
() => ['raw', 'tcp', 'httpupgrade', 'ws', 'grpc', 'xhttp'].includes(network),
[network],
);
const showUdp = isHysteria || network === 'kcp';
const showQuic = isHysteria || network === 'xhttp';
function notify() {
onChange();
}
function changeUdpMaskType(mask: MaskRow, type: string) {
changeMaskType(mask, type);
if (network === 'kcp' && stream.kcp) {
stream.kcp.mtu = type === 'xdns' ? 900 : 1350;
}
notify();
}
function addUdpMaskWithDefault() {
const def = isHysteria ? 'salamander' : 'mkcp-aes128gcm';
stream.addUdpMask(def);
notify();
}
const tcpMasks = stream.finalmask.tcp || [];
const udpMasks = stream.finalmask.udp || [];
if (!showTcp && !showUdp && !showQuic) return null;
return (
<Form colon={false} labelCol={{ md: { span: 8 } }} wrapperCol={{ md: { span: 14 } }}>
{showTcp && (
<>
<Form.Item label="TCP Masks">
<Button
type="primary"
size="small"
icon={<PlusOutlined />}
onClick={() => {
stream.addTcpMask('fragment');
notify();
}}
/>
</Form.Item>
{tcpMasks.map((mask, mIdx) => (
<div key={`tcp-${mIdx}`}>
<Divider style={{ margin: 0 }}>
TCP Mask {mIdx + 1}
<DeleteOutlined
className="danger-icon"
onClick={() => {
stream.delTcpMask(mIdx);
notify();
}}
/>
</Divider>
<Form.Item label="Type">
<Select
value={mask.type}
onChange={(v) => {
changeMaskType(mask, v);
notify();
}}
options={[
{ value: 'fragment', label: 'Fragment' },
{ value: 'header-custom', label: 'Header Custom' },
{ value: 'sudoku', label: 'Sudoku' },
]}
/>
</Form.Item>
{mask.type === 'fragment' && (
<>
<Form.Item label="Packets">
<Select
value={mask.settings.packets as string}
onChange={(v) => {
(mask.settings as Record<string, unknown>).packets = v;
notify();
}}
options={[
{ value: 'tlshello', label: 'tlshello' },
{ value: '1-3', label: '1-3' },
{ value: '1-5', label: '1-5' },
]}
/>
</Form.Item>
{(['length', 'delay', 'maxSplit'] as const).map((field) => (
<Form.Item key={field} label={field === 'maxSplit' ? 'Max Split' : field.charAt(0).toUpperCase() + field.slice(1)}>
<Input
value={(mask.settings[field] as string) || ''}
onChange={(e) => {
(mask.settings as Record<string, unknown>)[field] = e.target.value;
notify();
}}
/>
</Form.Item>
))}
</>
)}
{mask.type === 'sudoku' && (
<>
{(['password', 'ascii', 'customTable', 'customTables'] as const).map((field) => (
<Form.Item key={field} label={field === 'customTable' ? 'Custom Table' : field === 'customTables' ? 'Custom Tables' : field.charAt(0).toUpperCase() + field.slice(1)}>
<Input
value={(mask.settings[field] as string) || ''}
onChange={(e) => {
(mask.settings as Record<string, unknown>)[field] = e.target.value;
notify();
}}
/>
</Form.Item>
))}
{(['paddingMin', 'paddingMax'] as const).map((field) => (
<Form.Item key={field} label={field === 'paddingMin' ? 'Padding Min' : 'Padding Max'}>
<InputNumber
value={(mask.settings[field] as number) || 0}
min={0}
onChange={(v) => {
(mask.settings as Record<string, unknown>)[field] = Number(v) || 0;
notify();
}}
/>
</Form.Item>
))}
</>
)}
{mask.type === 'header-custom' && (
<HeaderCustomGroups mask={mask} kind="tcp" onChange={notify} />
)}
</div>
))}
</>
)}
{showUdp && (
<>
<Form.Item label="UDP Masks">
<Button type="primary" size="small" icon={<PlusOutlined />} onClick={addUdpMaskWithDefault} />
</Form.Item>
{udpMasks.map((mask, mIdx) => (
<div key={`udp-${mIdx}`}>
<Divider style={{ margin: 0 }}>
UDP Mask {mIdx + 1}
<DeleteOutlined
className="danger-icon"
onClick={() => {
stream.delUdpMask(mIdx);
notify();
}}
/>
</Divider>
<Form.Item label="Type">
<Select
value={mask.type}
onChange={(v) => changeUdpMaskType(mask, v)}
options={
isHysteria
? [{ value: 'salamander', label: 'Salamander (Hysteria2)' }]
: [
{ value: 'mkcp-aes128gcm', label: 'mKCP AES-128-GCM' },
{ value: 'header-dns', label: 'Header DNS' },
{ value: 'header-dtls', label: 'Header DTLS 1.2' },
{ value: 'header-srtp', label: 'Header SRTP' },
{ value: 'header-utp', label: 'Header uTP' },
{ value: 'header-wechat', label: 'Header WeChat Video' },
{ value: 'header-wireguard', label: 'Header WireGuard' },
{ value: 'mkcp-original', label: 'mKCP Original' },
{ value: 'xdns', label: 'xDNS' },
{ value: 'xicmp', label: 'xICMP' },
{ value: 'header-custom', label: 'Header Custom' },
{ value: 'noise', label: 'Noise' },
]
}
/>
</Form.Item>
{['mkcp-aes128gcm', 'salamander'].includes(mask.type) && (
<Form.Item label="Password">
<Input
value={(mask.settings.password as string) || ''}
placeholder="Obfuscation password"
onChange={(e) => {
(mask.settings as Record<string, unknown>).password = e.target.value;
notify();
}}
/>
</Form.Item>
)}
{mask.type === 'header-dns' && (
<Form.Item label="Domain">
<Input
value={(mask.settings.domain as string) || ''}
placeholder="e.g., www.example.com"
onChange={(e) => {
(mask.settings as Record<string, unknown>).domain = e.target.value;
notify();
}}
/>
</Form.Item>
)}
{mask.type === 'xdns' && (
<Form.Item label="Domains">
<Select
mode="tags"
value={(mask.settings.domains as string[]) || []}
style={{ width: '100%' }}
tokenSeparators={[',']}
placeholder="e.g., www.example.com"
onChange={(v) => {
(mask.settings as Record<string, unknown>).domains = v;
notify();
}}
/>
</Form.Item>
)}
{mask.type === 'noise' && (
<NoiseItems mask={mask} onChange={notify} />
)}
{mask.type === 'header-custom' && (
<UdpHeaderCustom mask={mask} onChange={notify} />
)}
{mask.type === 'xicmp' && (
<>
<Form.Item label="IP">
<Input
value={(mask.settings.ip as string) || ''}
placeholder="0.0.0.0"
onChange={(e) => {
(mask.settings as Record<string, unknown>).ip = e.target.value;
notify();
}}
/>
</Form.Item>
<Form.Item label="ID">
<InputNumber
value={(mask.settings.id as number) || 0}
min={0}
onChange={(v) => {
(mask.settings as Record<string, unknown>).id = Number(v) || 0;
notify();
}}
/>
</Form.Item>
</>
)}
</div>
))}
</>
)}
{showQuic && (
<>
<Form.Item label="QUIC Params">
<Switch
checked={!!stream.finalmask.enableQuicParams}
onChange={(v) => {
stream.finalmask.enableQuicParams = v;
notify();
}}
/>
</Form.Item>
{stream.finalmask.enableQuicParams && stream.finalmask.quicParams && (
<QuicParamsForm params={stream.finalmask.quicParams} onChange={notify} />
)}
</>
)}
</Form>
);
}
function HeaderCustomGroups({
mask,
kind: _kind,
onChange,
}: {
mask: MaskRow;
kind: 'tcp';
onChange: () => void;
}) {
const settings = mask.settings as { clients?: ItemRow[][]; servers?: ItemRow[][] };
if (!settings.clients) settings.clients = [];
if (!settings.servers) settings.servers = [];
return (
<>
{(['clients', 'servers'] as const).map((groupKey) => (
<div key={groupKey}>
<Form.Item label={groupKey === 'clients' ? 'Clients' : 'Servers'}>
<Button
type="primary"
size="small"
icon={<PlusOutlined />}
onClick={() => {
(settings[groupKey] as ItemRow[][]).push([newClientServerItem()]);
onChange();
}}
/>
</Form.Item>
{(settings[groupKey] as ItemRow[][]).map((group, gi) => (
<div key={`${groupKey}-${gi}`}>
<Divider style={{ margin: 0 }}>
{groupKey === 'clients' ? 'Clients' : 'Servers'} Group {gi + 1}
<DeleteOutlined
className="danger-icon"
onClick={() => {
(settings[groupKey] as ItemRow[][]).splice(gi, 1);
onChange();
}}
/>
</Divider>
{group.map((item, _ii) => (
<ItemEditor key={_ii} item={item} onChange={onChange} delayAsNumber />
))}
</div>
))}
</div>
))}
</>
);
}
function UdpHeaderCustom({ mask, onChange }: { mask: MaskRow; onChange: () => void }) {
const settings = mask.settings as { client?: ItemRow[]; server?: ItemRow[] };
if (!settings.client) settings.client = [];
if (!settings.server) settings.server = [];
return (
<>
{(['client', 'server'] as const).map((groupKey) => (
<div key={groupKey}>
<Form.Item label={groupKey === 'client' ? 'Client' : 'Server'}>
<Button
type="primary"
size="small"
icon={<PlusOutlined />}
onClick={() => {
(settings[groupKey] as ItemRow[]).push(newUdpClientServerItem());
onChange();
}}
/>
</Form.Item>
{(settings[groupKey] as ItemRow[]).map((item, ci) => (
<div key={ci}>
<Divider style={{ margin: 0 }}>
{groupKey === 'client' ? 'Client' : 'Server'} {ci + 1}
<DeleteOutlined
className="danger-icon"
onClick={() => {
(settings[groupKey] as ItemRow[]).splice(ci, 1);
onChange();
}}
/>
</Divider>
<ItemEditor item={item} onChange={onChange} />
</div>
))}
</div>
))}
</>
);
}
function NoiseItems({ mask, onChange }: { mask: MaskRow; onChange: () => void }) {
const settings = mask.settings as { reset?: number; noise?: ItemRow[] };
if (!settings.noise) settings.noise = [];
return (
<>
<Form.Item label="Reset">
<InputNumber
value={settings.reset || 0}
min={0}
onChange={(v) => {
settings.reset = Number(v) || 0;
onChange();
}}
/>
</Form.Item>
<Form.Item label="Noise">
<Button
type="primary"
size="small"
icon={<PlusOutlined />}
onClick={() => {
(settings.noise as ItemRow[]).push(newNoiseItem());
onChange();
}}
/>
</Form.Item>
{(settings.noise as ItemRow[]).map((n, ni) => (
<div key={ni}>
<Divider style={{ margin: 0 }}>
Noise {ni + 1}
<DeleteOutlined
className="danger-icon"
onClick={() => {
(settings.noise as ItemRow[]).splice(ni, 1);
onChange();
}}
/>
</Divider>
<ItemEditor item={n} onChange={onChange} delayAsString />
</div>
))}
</>
);
}
function ItemEditor({
item,
onChange,
delayAsNumber,
delayAsString,
}: {
item: ItemRow;
onChange: () => void;
delayAsNumber?: boolean;
delayAsString?: boolean;
}) {
return (
<>
<Form.Item label="Type">
<Select
value={item.type}
onChange={(v) => {
changeItemType(item, v);
onChange();
}}
options={[
{ value: 'array', label: 'Array' },
{ value: 'str', label: 'String' },
{ value: 'hex', label: 'Hex' },
{ value: 'base64', label: 'Base64' },
]}
/>
</Form.Item>
{delayAsNumber && (
<Form.Item label="Delay (ms)">
<InputNumber
value={typeof item.delay === 'number' ? item.delay : 0}
min={0}
onChange={(v) => {
item.delay = Number(v) || 0;
onChange();
}}
/>
</Form.Item>
)}
{item.type === 'array' ? (
<>
<Form.Item label="Rand">
{delayAsString ? (
<Input
value={String(item.rand ?? '')}
onChange={(e) => {
item.rand = e.target.value;
onChange();
}}
placeholder="0 or 1-8192"
/>
) : (
<InputNumber
value={typeof item.rand === 'number' ? item.rand : 0}
min={0}
onChange={(v) => {
item.rand = Number(v) || 0;
onChange();
}}
/>
)}
</Form.Item>
<Form.Item label="Rand Range">
<Input
value={item.randRange || ''}
placeholder="0-255"
onChange={(e) => {
item.randRange = e.target.value;
onChange();
}}
/>
</Form.Item>
</>
) : (
<Form.Item label="Packet">
{item.type === 'base64' ? (
<Input.Group compact>
<Input
value={String(item.packet ?? '')}
placeholder="binary data"
style={{ width: 'calc(100% - 32px)' }}
onChange={(e) => {
item.packet = e.target.value;
onChange();
}}
/>
<Button
icon={<ReloadOutlined />}
onClick={() => {
item.packet = RandomUtil.randomBase64();
onChange();
}}
/>
</Input.Group>
) : (
<Input
value={String(item.packet ?? '')}
placeholder="binary data"
onChange={(e) => {
item.packet = e.target.value;
onChange();
}}
/>
)}
</Form.Item>
)}
{delayAsString && (
<Form.Item label="Delay">
<Input
value={typeof item.delay === 'string' ? item.delay : ''}
placeholder="10-20"
onChange={(e) => {
item.delay = e.target.value;
onChange();
}}
/>
</Form.Item>
)}
</>
);
}
function QuicParamsForm({ params, onChange }: { params: QuicParams; onChange: () => void }) {
function update<K extends keyof QuicParams>(key: K, value: QuicParams[K]) {
params[key] = value;
onChange();
}
return (
<>
<Form.Item label="Congestion">
<Select
value={params.congestion}
onChange={(v) => update('congestion', v)}
options={[
{ value: 'reno', label: 'Reno' },
{ value: 'bbr', label: 'BBR' },
{ value: 'brutal', label: 'Brutal' },
{ value: 'force-brutal', label: 'Force Brutal' },
]}
/>
</Form.Item>
<Form.Item label="Debug">
<Switch checked={!!params.debug} onChange={(v) => update('debug', v)} />
</Form.Item>
{['brutal', 'force-brutal'].includes(params.congestion) && (
<>
<Form.Item label="Brutal Up">
<Input
value={String(params.brutalUp ?? '')}
placeholder="65537"
onChange={(e) => update('brutalUp', e.target.value)}
/>
</Form.Item>
<Form.Item label="Brutal Down">
<Input
value={String(params.brutalDown ?? '')}
placeholder="65537"
onChange={(e) => update('brutalDown', e.target.value)}
/>
</Form.Item>
</>
)}
<Form.Item label="UDP Hop">
<Switch checked={!!params.hasUdpHop} onChange={(v) => update('hasUdpHop', v)} />
</Form.Item>
{params.hasUdpHop && params.udpHop && (
<>
<Form.Item label="Hop Ports">
<Input
value={params.udpHop.ports || ''}
placeholder="e.g. 20000-50000"
onChange={(e) => {
params.udpHop!.ports = e.target.value;
onChange();
}}
/>
</Form.Item>
<Form.Item label="Hop Interval (s)">
<InputNumber
value={Number(params.udpHop.interval) || 5}
min={5}
onChange={(v) => {
params.udpHop!.interval = Number(v) || 5;
onChange();
}}
/>
</Form.Item>
</>
)}
{(
[
['maxIdleTimeout', 'Max Idle Timeout (s)', 4, 120],
['keepAlivePeriod', 'Keep Alive Period (s)', 2, 60],
] as const
).map(([key, label, min, max]) => (
<Form.Item key={key} label={label}>
<InputNumber
value={params[key] as number}
min={min}
max={max}
onChange={(v) => update(key, Number(v) || min)}
/>
</Form.Item>
))}
<Form.Item label="Disable Path MTU Dis">
<Switch checked={!!params.disablePathMTUDiscovery} onChange={(v) => update('disablePathMTUDiscovery', v)} />
</Form.Item>
{(
[
['maxIncomingStreams', 'Max Incoming Streams', 8, '1024 = default'],
['initStreamReceiveWindow', 'Init Stream Window', 16384, '8388608 = default'],
['maxStreamReceiveWindow', 'Max Stream Window', 16384, '8388608 = default'],
['initConnectionReceiveWindow', 'Init Conn Window', 16384, '20971520 = default'],
['maxConnectionReceiveWindow', 'Max Conn Window', 16384, '20971520 = default'],
] as const
).map(([key, label, min, placeholder]) => (
<Form.Item key={key} label={label}>
<InputNumber
value={params[key] as number}
min={min}
placeholder={placeholder}
onChange={(v) => update(key, Number(v) || 0)}
/>
</Form.Item>
))}
</>
);
}