visionA 雲端版前端 — 沿用 local-tool 既有 UI(原則 4:先抄 local-tool)+
新增雲端特有的登入 / 配對 / 設定流程,含以下整合階段:
- Phase 0:13 頁 + 30+ 元件 + 雛形 banner
- dashboard / devices / models / workspace / clusters / settings 等頁
- AppShell + Sidebar + Header + tokens + i18n(中英雙語 96 keys)
- API client + 5 stores + 3 hooks
- Phase 0.6:OIDC redirect 改造
- login 頁改為 OIDC redirect(`window.location.href = /api/auth/login`)
- register 改說明頁、account 改唯讀(user 資料來源是 MC)
- api client 改 cookie session(credentials: include)+ 完全清掉 localStorage
- Phase 0.7:stage 部署 + nil guard
- getApiBaseUrl() 修:browser 環境視為 same-origin(與 login 頁一致)
- login 頁加「已登入 → router.replace('/')」effect
- User type email/name 改 optional(MC id_token 不一定回 email/name claim)
- header.tsx UserMenu displayName 4 層 fallback:name → email → id → i18n
- 雛形 banner 文案更新(已接 Innovedus 帳號中心)+ 版號 Phase 0.7
驗證:pnpm lint / test (125/125) / build 全綠
Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
101 lines
4.0 KiB
TypeScript
101 lines
4.0 KiB
TypeScript
"use client";
|
||
|
||
import * as React from "react";
|
||
import { cva, type VariantProps } from "class-variance-authority";
|
||
import { Tabs as TabsPrimitive } from "radix-ui";
|
||
|
||
import { cn } from "@/lib/utils";
|
||
|
||
/**
|
||
* Tabs — Shadcn 風分頁(Radix Tabs 封裝)
|
||
*
|
||
* 來源:local-tool/frontend/src/components/ui/tabs.tsx(100% 沿用)
|
||
*
|
||
* variant:
|
||
* - default:TabsList 帶背景(`bg-muted`),active 項顯示白底陰影
|
||
* - line:TabsList 無背景,active 項底部顯示橫線(搭配 `after:*` utility)
|
||
*/
|
||
function Tabs({
|
||
className,
|
||
orientation = "horizontal",
|
||
...props
|
||
}: React.ComponentProps<typeof TabsPrimitive.Root>) {
|
||
return (
|
||
<TabsPrimitive.Root
|
||
data-slot="tabs"
|
||
data-orientation={orientation}
|
||
orientation={orientation}
|
||
className={cn(
|
||
"group/tabs flex gap-2 data-[orientation=horizontal]:flex-col",
|
||
className,
|
||
)}
|
||
{...props}
|
||
/>
|
||
);
|
||
}
|
||
|
||
const tabsListVariants = cva(
|
||
"rounded-lg p-[3px] group-data-[orientation=horizontal]/tabs:h-9 data-[variant=line]:rounded-none group/tabs-list text-muted-foreground inline-flex w-fit items-center justify-center group-data-[orientation=vertical]/tabs:h-fit group-data-[orientation=vertical]/tabs:flex-col",
|
||
{
|
||
variants: {
|
||
variant: {
|
||
default: "bg-muted",
|
||
line: "gap-1 bg-transparent",
|
||
},
|
||
},
|
||
defaultVariants: {
|
||
variant: "default",
|
||
},
|
||
},
|
||
);
|
||
|
||
function TabsList({
|
||
className,
|
||
variant = "default",
|
||
...props
|
||
}: React.ComponentProps<typeof TabsPrimitive.List> &
|
||
VariantProps<typeof tabsListVariants>) {
|
||
return (
|
||
<TabsPrimitive.List
|
||
data-slot="tabs-list"
|
||
data-variant={variant}
|
||
className={cn(tabsListVariants({ variant }), className)}
|
||
{...props}
|
||
/>
|
||
);
|
||
}
|
||
|
||
function TabsTrigger({
|
||
className,
|
||
...props
|
||
}: React.ComponentProps<typeof TabsPrimitive.Trigger>) {
|
||
return (
|
||
<TabsPrimitive.Trigger
|
||
data-slot="tabs-trigger"
|
||
className={cn(
|
||
"focus-visible:border-ring focus-visible:ring-ring/50 focus-visible:outline-ring text-foreground/60 hover:text-foreground dark:text-muted-foreground dark:hover:text-foreground relative inline-flex h-[calc(100%-1px)] flex-1 items-center justify-center gap-1.5 rounded-md border border-transparent px-2 py-1 text-sm font-medium whitespace-nowrap transition-all group-data-[orientation=vertical]/tabs:w-full group-data-[orientation=vertical]/tabs:justify-start focus-visible:ring-[3px] focus-visible:outline-1 disabled:pointer-events-none disabled:opacity-50 group-data-[variant=default]/tabs-list:data-[state=active]:shadow-sm group-data-[variant=line]/tabs-list:data-[state=active]:shadow-none [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
|
||
"group-data-[variant=line]/tabs-list:bg-transparent group-data-[variant=line]/tabs-list:data-[state=active]:bg-transparent dark:group-data-[variant=line]/tabs-list:data-[state=active]:border-transparent dark:group-data-[variant=line]/tabs-list:data-[state=active]:bg-transparent",
|
||
"data-[state=active]:bg-background dark:data-[state=active]:text-foreground dark:data-[state=active]:border-input dark:data-[state=active]:bg-input/30 data-[state=active]:text-foreground",
|
||
"after:bg-foreground after:absolute after:opacity-0 after:transition-opacity group-data-[orientation=horizontal]/tabs:after:inset-x-0 group-data-[orientation=horizontal]/tabs:after:bottom-[-5px] group-data-[orientation=horizontal]/tabs:after:h-0.5 group-data-[orientation=vertical]/tabs:after:inset-y-0 group-data-[orientation=vertical]/tabs:after:-right-1 group-data-[orientation=vertical]/tabs:after:w-0.5 group-data-[variant=line]/tabs-list:data-[state=active]:after:opacity-100",
|
||
className,
|
||
)}
|
||
{...props}
|
||
/>
|
||
);
|
||
}
|
||
|
||
function TabsContent({
|
||
className,
|
||
...props
|
||
}: React.ComponentProps<typeof TabsPrimitive.Content>) {
|
||
return (
|
||
<TabsPrimitive.Content
|
||
data-slot="tabs-content"
|
||
className={cn("flex-1 outline-none", className)}
|
||
{...props}
|
||
/>
|
||
);
|
||
}
|
||
|
||
export { Tabs, TabsList, TabsTrigger, TabsContent, tabsListVariants };
|