Files
users/app/tag-rules/page.tsx
v0 2408d50cb0 refactor: overhaul UI for streamlined user experience
Redesign navigation, home overview, user portrait, and valuation pages
with improved functionality and responsive design.

Co-authored-by: null <4804959+fnvtk@users.noreply.github.com>
2025-07-18 13:47:12 +00:00

399 lines
15 KiB
TypeScript

"use client"
import { useState } from "react"
import { Card, CardContent, CardHeader, CardTitle } from "@/components/ui/card"
import { Button } from "@/components/ui/button"
import { Input } from "@/components/ui/input"
import { Search, Plus, Filter, Play, Pause, Edit, Trash2, MoreHorizontal, FileText, Clock } from "lucide-react"
import { Table, TableBody, TableCell, TableHead, TableHeader, TableRow } from "@/components/ui/table"
import { Badge } from "@/components/ui/badge"
import {
Dialog,
DialogContent,
DialogDescription,
DialogFooter,
DialogHeader,
DialogTitle,
} from "@/components/ui/dialog"
import { Label } from "@/components/ui/label"
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@/components/ui/select"
import { Textarea } from "@/components/ui/textarea"
import { DropdownMenu, DropdownMenuContent, DropdownMenuItem, DropdownMenuTrigger } from "@/components/ui/dropdown-menu"
import { Tabs, TabsContent, TabsList, TabsTrigger } from "@/components/ui/tabs"
import { Switch } from "@/components/ui/switch"
import { RuleEditor } from "@/components/tag-rules/rule-editor"
import { RuleExecutionHistory } from "@/components/tag-rules/rule-execution-history"
interface TagRule {
id: string
name: string
description: string
targetTag: string
condition: string
status: "active" | "inactive" | "draft"
priority: number
createdAt: string
lastRun: string
affectedUsers: number
}
const mockRules: TagRule[] = [
{
id: "1",
name: "高价值用户识别",
description: "根据用户消费金额和频次识别高价值用户",
targetTag: "高价值用户",
condition: "消费金额 > 5000 AND 消费频次 > 10",
status: "active",
priority: 1,
createdAt: "2023-05-12",
lastRun: "2023-07-21 15:30",
affectedUsers: 1250,
},
{
id: "2",
name: "游戏爱好者标记",
description: "根据用户浏览和购买行为识别游戏爱好者",
targetTag: "游戏爱好者",
condition: "游戏类目浏览时长 > 30min OR 游戏类目购买次数 > 2",
status: "active",
priority: 2,
createdAt: "2023-04-18",
lastRun: "2023-07-21 14:45",
affectedUsers: 2840,
},
{
id: "3",
name: "流失风险预警",
description: "识别30天内可能流失的用户",
targetTag: "流失风险高",
condition: "最近登录时间 > 15天 AND 最近30天活跃度下降率 > 50%",
status: "active",
priority: 1,
createdAt: "2023-06-30",
lastRun: "2023-07-21 13:20",
affectedUsers: 890,
},
{
id: "4",
name: "周末活跃用户",
description: "识别周末活跃度高的用户",
targetTag: "周末活跃",
condition: "周末活跃时长 > 工作日活跃时长 * 1.5",
status: "inactive",
priority: 3,
createdAt: "2023-05-28",
lastRun: "2023-07-10 09:15",
affectedUsers: 3520,
},
{
id: "5",
name: "潜在高转化用户",
description: "识别浏览量高但未转化的潜在用户",
targetTag: "潜在高转化",
condition: "浏览商品数 > 20 AND 加购次数 > 5 AND 购买次数 = 0",
status: "draft",
priority: 2,
createdAt: "2023-07-15",
lastRun: "-",
affectedUsers: 0,
},
]
export default function TagRulesPage() {
const [searchQuery, setSearchQuery] = useState("")
const [isCreateDialogOpen, setIsCreateDialogOpen] = useState(false)
const [newRule, setNewRule] = useState({
name: "",
description: "",
targetTag: "",
condition: "",
priority: 2,
status: "draft",
})
const [selectedRuleId, setSelectedRuleId] = useState<string | null>(null)
const handleCreateRule = () => {
// 这里应该是创建规则的逻辑
console.log("创建规则:", newRule)
setIsCreateDialogOpen(false)
setNewRule({
name: "",
description: "",
targetTag: "",
condition: "",
priority: 2,
status: "draft",
})
}
const filteredRules = mockRules.filter(
(rule) =>
rule.name.toLowerCase().includes(searchQuery.toLowerCase()) ||
rule.description.toLowerCase().includes(searchQuery.toLowerCase()) ||
rule.targetTag.toLowerCase().includes(searchQuery.toLowerCase()),
)
const getStatusBadge = (status: TagRule["status"]) => {
switch (status) {
case "active":
return <Badge className="bg-green-100 text-green-800"></Badge>
case "inactive":
return <Badge className="bg-yellow-100 text-yellow-800"></Badge>
case "draft":
return <Badge className="bg-gray-100 text-gray-800">稿</Badge>
}
}
return (
<div className="container mx-auto p-6 space-y-6">
<div className="flex justify-between items-center">
<div>
<h1 className="text-3xl font-bold"></h1>
<p className="text-muted-foreground mt-1"></p>
</div>
<div className="flex space-x-2">
<Button variant="outline">
<Clock className="mr-2 h-4 w-4" />
</Button>
<Button onClick={() => setIsCreateDialogOpen(true)}>
<Plus className="mr-2 h-4 w-4" />
</Button>
</div>
</div>
<div className="grid grid-cols-1 md:grid-cols-3 gap-4">
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm font-medium"></CardTitle>
</CardHeader>
<CardContent>
<div className="text-2xl font-bold">24</div>
<p className="text-xs text-muted-foreground mt-1"> 3 </p>
</CardContent>
</Card>
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm font-medium"></CardTitle>
</CardHeader>
<CardContent>
<div className="text-2xl font-bold">18</div>
<p className="text-xs text-muted-foreground mt-1"> 75%</p>
</CardContent>
</Card>
<Card>
<CardHeader className="pb-2">
<CardTitle className="text-sm font-medium"></CardTitle>
</CardHeader>
<CardContent>
<div className="text-2xl font-bold">36</div>
<p className="text-xs text-muted-foreground mt-1"> 12,580 </p>
</CardContent>
</Card>
</div>
<Tabs defaultValue="rules" className="space-y-4">
<div className="flex justify-between items-center">
<TabsList>
<TabsTrigger value="rules"></TabsTrigger>
<TabsTrigger value="editor"></TabsTrigger>
<TabsTrigger value="history"></TabsTrigger>
</TabsList>
<div className="flex space-x-2">
<div className="relative w-64">
<Search className="absolute left-2.5 top-2.5 h-4 w-4 text-muted-foreground" />
<Input
type="text"
placeholder="搜索规则..."
className="pl-8"
value={searchQuery}
onChange={(e) => setSearchQuery(e.target.value)}
/>
</div>
<Button variant="outline" size="icon">
<Filter className="h-4 w-4" />
</Button>
</div>
</div>
<TabsContent value="rules" className="space-y-4">
<Card>
<CardContent className="p-0">
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead className="text-right"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{filteredRules.map((rule) => (
<TableRow key={rule.id}>
<TableCell className="font-medium">
<div>{rule.name}</div>
<div className="text-xs text-muted-foreground">{rule.description}</div>
</TableCell>
<TableCell>{rule.targetTag}</TableCell>
<TableCell>{getStatusBadge(rule.status)}</TableCell>
<TableCell>{rule.priority}</TableCell>
<TableCell>{rule.createdAt}</TableCell>
<TableCell>{rule.lastRun}</TableCell>
<TableCell>{rule.affectedUsers.toLocaleString()}</TableCell>
<TableCell className="text-right">
<DropdownMenu>
<DropdownMenuTrigger asChild>
<Button variant="ghost" size="icon">
<MoreHorizontal className="h-4 w-4" />
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end">
<DropdownMenuItem onClick={() => setSelectedRuleId(rule.id)}>
<Edit className="mr-2 h-4 w-4" />
</DropdownMenuItem>
<DropdownMenuItem>
{rule.status === "active" ? (
<>
<Pause className="mr-2 h-4 w-4" />
</>
) : (
<>
<Play className="mr-2 h-4 w-4" />
</>
)}
</DropdownMenuItem>
<DropdownMenuItem>
<FileText className="mr-2 h-4 w-4" />
</DropdownMenuItem>
<DropdownMenuItem>
<Trash2 className="mr-2 h-4 w-4" />
</DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</CardContent>
</Card>
</TabsContent>
<TabsContent value="editor" className="space-y-4">
<RuleEditor ruleId={selectedRuleId} />
</TabsContent>
<TabsContent value="history" className="space-y-4">
<RuleExecutionHistory />
</TabsContent>
</Tabs>
{/* 创建规则对话框 */}
<Dialog open={isCreateDialogOpen} onOpenChange={setIsCreateDialogOpen}>
<DialogContent className="sm:max-w-[600px]">
<DialogHeader>
<DialogTitle></DialogTitle>
<DialogDescription></DialogDescription>
</DialogHeader>
<div className="grid gap-4 py-4">
<div className="grid grid-cols-4 items-center gap-4">
<Label htmlFor="rule-name" className="text-right">
</Label>
<Input
id="rule-name"
value={newRule.name}
onChange={(e) => setNewRule({ ...newRule, name: e.target.value })}
className="col-span-3"
/>
</div>
<div className="grid grid-cols-4 items-center gap-4">
<Label htmlFor="rule-description" className="text-right">
</Label>
<Textarea
id="rule-description"
value={newRule.description}
onChange={(e) => setNewRule({ ...newRule, description: e.target.value })}
className="col-span-3"
/>
</div>
<div className="grid grid-cols-4 items-center gap-4">
<Label htmlFor="target-tag" className="text-right">
</Label>
<Input
id="target-tag"
value={newRule.targetTag}
onChange={(e) => setNewRule({ ...newRule, targetTag: e.target.value })}
className="col-span-3"
/>
</div>
<div className="grid grid-cols-4 items-center gap-4">
<Label htmlFor="rule-condition" className="text-right">
</Label>
<Textarea
id="rule-condition"
value={newRule.condition}
onChange={(e) => setNewRule({ ...newRule, condition: e.target.value })}
className="col-span-3"
placeholder="例如: 消费金额 > 5000 AND 消费频次 > 10"
/>
</div>
<div className="grid grid-cols-4 items-center gap-4">
<Label htmlFor="rule-priority" className="text-right">
</Label>
<Select
value={newRule.priority.toString()}
onValueChange={(value) => setNewRule({ ...newRule, priority: Number.parseInt(value) })}
>
<SelectTrigger className="col-span-3">
<SelectValue placeholder="选择优先级" />
</SelectTrigger>
<SelectContent>
<SelectItem value="1">1 - </SelectItem>
<SelectItem value="2">2 - </SelectItem>
<SelectItem value="3">3 - </SelectItem>
</SelectContent>
</Select>
</div>
<div className="grid grid-cols-4 items-center gap-4">
<Label htmlFor="rule-status" className="text-right">
</Label>
<div className="flex items-center space-x-2 col-span-3">
<Switch
id="rule-status"
checked={newRule.status === "active"}
onCheckedChange={(checked) => setNewRule({ ...newRule, status: checked ? "active" : "draft" })}
/>
<Label htmlFor="rule-status">{newRule.status === "active" ? "立即启用" : "保存为草稿"}</Label>
</div>
</div>
</div>
<DialogFooter>
<Button variant="outline" onClick={() => setIsCreateDialogOpen(false)}>
</Button>
<Button onClick={handleCreateRule}></Button>
</DialogFooter>
</DialogContent>
</Dialog>
</div>
)
}