| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574 |
- "use client"
- import React, {useEffect, useMemo, useState} from "react"
- import "antd/dist/reset.css"
- import {
- BadgeAlertIcon as Alert,
- BarChartIcon as ChartBar,
- BellRing,
- CheckCircle2,
- ChevronsUp,
- FileText,
- LineChart,
- MapPin,
- Settings,
- ShieldAlert,
- Siren,
- SquareGanttChart,
- TimerIcon as Timeline,
- UserRoundCheck,
- } from "lucide-react" // Declare ShieldAlert variable
- import {
- Badge as AntdBadge,
- Button,
- Card,
- Col,
- DatePicker,
- Descriptions,
- Divider,
- Drawer,
- Form,
- Input,
- Modal,
- notification,
- Popconfirm,
- Radio,
- Row,
- Segmented,
- Select,
- Space,
- Statistic,
- Steps,
- Table,
- Tabs,
- Tag,
- Tooltip,
- Upload,
- } from "antd"
- import type {ColumnsType} from "antd/es/table"
- import type {UploadFile} from "antd/es/upload/interface"
- import dayjs, {type Dayjs} from "dayjs"
- import relativeTime from "dayjs/plugin/relativeTime"
- import isBetween from "dayjs/plugin/isBetween"
- import EChart from "@/components/echarts"
- import MatterManagement from "./components/matter-management"
- import globalMessage from "@/app/_modules/globalMessage";
- dayjs.extend(relativeTime)
- dayjs.extend(isBetween)
- type WarningStatus =
- | "draft"
- | "published"
- | "in_process"
- | "supervising"
- | "returned"
- | "upgraded"
- | "resolved"
- | "released"
- type Level = "红色" | "橙色" | "黄色" | "蓝色"
- type WarnType = "燃气" | "供水" | "电力" | "交通" | "综合"
- type Industry = WarnType
- type ProcessLog = {
- time: string
- user: string
- action: string
- remark?: string
- }
- type Alarm = {
- id: string
- device: string
- metric: string
- value: number
- threshold: number
- time: string
- }
- type WarningItem = {
- id: string
- code: string
- name: string
- type: WarnType
- level: Level
- industry: Industry
- status: WarningStatus
- createdAt: string
- releasedAt?: string
- deadline?: string
- assignee?: string
- handler?: string
- location?: string
- coords?: [number, number]
- description?: string
- attachments?: UploadFile[]
- relatedAlarms?: Alarm[]
- process: ProcessLog[]
- }
- type MatterItemStatus = "启用" | "禁用" | "作废"
- type MatterItem = {
- id: string
- code: string
- name: string
- type: WarnType
- level: Level
- status: MatterItemStatus
- createdAt: string
- updatedAt: string
- }
- type ReasonEntry = {
- id: string
- type: WarnType
- level: Level
- reason: string
- createdAt: string
- }
- const LEVELS: Level[] = ["红色", "橙色", "黄色", "蓝色"]
- const TYPES: WarnType[] = ["燃气", "供水", "电力", "交通", "综合"]
- const INDUSTRIES: Industry[] = ["燃气", "供水", "电力", "交通", "综合"]
- const USERS = ["张三", "李四", "王五", "赵六", "钱七", "管理员"]
- const LEADERS = ["总指挥-陈总", "副总指挥-李总", "应急办-王主任"]
- const levelColor = (lvl: Level) => {
- switch (lvl) {
- case "红色":
- return "red"
- case "橙色":
- return "orange"
- case "黄色":
- return "gold"
- case "蓝色":
- return "blue"
- default:
- return "default"
- }
- }
- const statusTag = (status: WarningStatus) => {
- const map: Record<WarningStatus, { color: string; text: string; icon?: React.ReactNode }> = {
- draft: { color: "default", text: "草稿", icon: <FileText size={14} /> },
- published: { color: "processing", text: "已发布", icon: <Siren size={14} /> },
- in_process: {
- color: "blue",
- text: "处置中",
- icon: <Timeline size={14} />,
- },
- supervising: {
- color: "purple",
- text: "督办中",
- icon: <UserRoundCheck size={14} />,
- },
- returned: { color: "warning", text: "已退回", icon: <Alert size={14} /> },
- upgraded: {
- color: "magenta",
- text: "已升级",
- icon: <ChevronsUp size={14} />,
- },
- resolved: {
- color: "success",
- text: "已处置",
- icon: <CheckCircle2 size={14} />,
- },
- released: {
- color: "green",
- text: "已解除",
- icon: <ShieldAlert size={14} />,
- },
- }
- const cfg = map[status]
- return (
- <Tag color={cfg.color}>
- <Space size={4}>
- {cfg.icon}
- <span>{cfg.text}</span>
- </Space>
- </Tag>
- )
- }
- function randPick<T>(arr: T[]): T {
- return arr[Math.floor(Math.random() * arr.length)]
- }
- function genId(prefix = "id") {
- return `${prefix}_${Math.random().toString(36).slice(2, 10)}`
- }
- function autoCode(type: WarnType, level: Level, seq: number) {
- const date = dayjs().format("YYYYMMDD")
- const t = { 燃气: "GAS", 供水: "WTR", 电力: "ELE", 交通: "TRF", 综合: "COM" }[type]
- const l = { 红色: "R", 橙色: "O", 黄色: "Y", 蓝色: "B" }[level]
- return `YW-${t}-${l}-${date}-${seq.toString().padStart(3, "0")}`
- }
- function isToday(iso?: string) {
- if (!iso) return false
- return dayjs(iso).isSame(dayjs(), "day")
- }
- function daysAgo(n: number) {
- return dayjs().subtract(n, "day").toISOString()
- }
- // demo seeds (client-only)
- const seedAlarms: Alarm[] = new Array(24).fill(0).map((_, i) => ({
- id: genId("alarm"),
- device: `设备-${(i % 8) + 1}`,
- metric: ["压力", "温度", "流量"][i % 3],
- value: +(80 + Math.random() * 50).toFixed(1),
- threshold: 100,
- time: dayjs()
- .subtract(Math.floor(Math.random() * 72), "hour")
- .toISOString(),
- }))
- const seedWarnings: WarningItem[] = (() => {
- const arr: WarningItem[] = []
- for (let i = 0; i < 26; i++) {
- const type = randPick(TYPES)
- const level = randPick(LEVELS)
- const status = randPick<WarningStatus>([
- "draft",
- "published",
- "in_process",
- "supervising",
- "returned",
- "upgraded",
- "resolved",
- "released",
- ])
- const createdAt = dayjs()
- .subtract(Math.floor(Math.random() * 15), "day")
- .subtract(Math.floor(Math.random() * 24), "hour")
- .toISOString()
- const releasedAt =
- status === "released" || status === "resolved"
- ? dayjs(createdAt)
- .add(Math.floor(Math.random() * 48), "hour")
- .toISOString()
- : undefined
- arr.push({
- id: genId("warn"),
- code: autoCode(type, level, i + 1),
- name: `${type}专项-${["一号", "二号", "三号", "四号"][i % 4]}预警`,
- type,
- level,
- industry: type,
- status,
- createdAt,
- releasedAt,
- deadline: dayjs(createdAt).add(2, "day").toISOString(),
- assignee: randPick(USERS),
- handler: randPick(USERS),
- location: ["城区A", "城区B", "园区C", "沿线D"][i % 4],
- coords: [120.1 + Math.random(), 30.2 + Math.random()],
- description: "设备异常波动,指标超限,建议立即排查。包含监测曲线、现场图片等信息。",
- attachments: [],
- relatedAlarms: seedAlarms.slice(i, i + 3),
- process: [
- { time: createdAt, user: randPick(USERS), action: "创建预警" },
- ...(status !== "draft"
- ? [{ time: dayjs(createdAt).add(1, "hour").toISOString(), user: randPick(USERS), action: "发布预警" }]
- : []),
- ...(status === "resolved" || status === "released"
- ? [
- {
- time: dayjs(createdAt).add(2, "day").toISOString(),
- user: randPick(USERS),
- action: status === "released" ? "解除预警" : "完成处置",
- },
- ]
- : []),
- ],
- })
- }
- return arr
- })()
- const seedMatters: MatterItem[] = new Array(12).fill(0).map((_, i) => {
- const type = randPick(TYPES)
- const level = randPick(LEVELS)
- return {
- id: genId("matter"),
- code: autoCode(type, level, i + 1),
- name: `${type}专项事项-${i + 1}`,
- type,
- level,
- status: randPick(["启用", "禁用", "作废"]),
- createdAt: daysAgo(30 - i),
- updatedAt: daysAgo(20 - i),
- }
- })
- const seedReasons: ReasonEntry[] = [
- { id: genId("reason"), type: "燃气", level: "红色", reason: "主干管压力骤降", createdAt: daysAgo(40) },
- { id: genId("reason"), type: "供水", level: "黄色", reason: "水质浊度升高", createdAt: daysAgo(25) },
- { id: genId("reason"), type: "电力", level: "橙色", reason: "变压器过载", createdAt: daysAgo(12) },
- ]
- export default function WarningDashboard() {
- const [currentUser] = useState("刘昊林")
- const [warnings, setWarnings] = useState<WarningItem[]>([])
- const [matters, setMatters] = useState<MatterItem[]>([])
- const [reasons, setReasons] = useState<ReasonEntry[]>([])
- useEffect(() => {
- // 仅在客户端生成随机数据
- setWarnings(seedWarnings)
- setMatters(seedMatters)
- setReasons(seedReasons)
- }, [])
- const [activeTab, setActiveTab] = useState("overview")
- const [detail, setDetail] = useState<WarningItem | null>(null)
- const [publishForm] = Form.useForm()
- const [publishOpen, setPublishOpen] = useState(false)
- const [superviseOpen, setSuperviseOpen] = useState<WarningItem | null>(null)
- const [upgradeOpen, setUpgradeOpen] = useState<WarningItem | null>(null)
- const [leaderOpen, setLeaderOpen] = useState<WarningItem | null>(null)
- const [returnOpen, setReturnOpen] = useState<WarningItem | null>(null)
- const [todoFilter, setTodoFilter] = useState<{ q?: string; status?: "todo" | "done" }>({ status: "todo" })
- const [mgmtFilters, setMgmtFilters] = useState<{
- type?: WarnType
- level?: Level
- industry?: Industry
- status?: WarningStatus
- date?: [Dayjs, Dayjs]
- q?: string
- }>({})
- const [todayOnly, setTodayOnly] = useState(true)
- // Derived stats
- const todayStats = useMemo(() => {
- const todayList = warnings.filter((w) => isToday(w.createdAt))
- const released = todayList.filter((w) => w.status === "released").length
- const unresolved = todayList.filter((w) => !["released"].includes(w.status)).length
- return { total: todayList.length, released, unresolved, list: todayList }
- }, [warnings])
- const todayUnhandled = useMemo(() => {
- return warnings.filter(
- (w) =>
- isToday(w.createdAt) && ["published", "in_process", "supervising", "upgraded", "returned"].includes(w.status),
- )
- }, [warnings])
- const historyByType = useMemo(() => {
- const map = new Map<WarnType, number>()
- TYPES.forEach((t) => map.set(t, 0))
- warnings.forEach((w) => map.set(w.type, (map.get(w.type) || 0) + 1))
- return Array.from(map.entries()).map(([name, value]) => ({ name, value }))
- }, [warnings])
- const historyByLevel = useMemo(() => {
- const map = new Map<Level, number>()
- LEVELS.forEach((l) => map.set(l, 0))
- warnings.forEach((w) => map.set(w.level, (map.get(w.level) || 0) + 1))
- return Array.from(map.entries()).map(([name, value]) => ({ name, value }))
- }, [warnings])
- const mineTodo = useMemo(() => {
- return warnings.filter(
- (w) =>
- w.assignee === currentUser &&
- ["published", "in_process", "supervising", "upgraded", "returned"].includes(w.status),
- )
- }, [warnings, currentUser])
- const mineDone = useMemo(() => {
- return warnings.filter((w) => w.assignee === currentUser && ["resolved", "released"].includes(w.status))
- }, [warnings, currentUser])
- // Actions
- function pushProcess(w: WarningItem, action: string, remark?: string) {
- w.process = [...w.process, { time: new Date().toISOString(), user: currentUser, action, remark }]
- }
- function updateWarning(wid: string, patch: Partial<WarningItem>, processAction?: string, remark?: string) {
- setWarnings((prev) =>
- prev.map((w) => {
- if (w.id !== wid) return w
- const next = { ...w, ...patch }
- if (processAction) {
- const cp = { ...next }
- pushProcess(cp, processAction, remark)
- return cp
- }
- return next
- }),
- )
- }
- function handleRelease(w: WarningItem) {
- updateWarning(w.id, { status: "released", releasedAt: new Date().toISOString() }, "解除预警")
- globalMessage.success("预警已解除")
- }
- function handleResolve(w: WarningItem) {
- updateWarning(w.id, { status: "resolved" }, "完成处置")
- globalMessage.success("预警已标记为已处置")
- }
- function handleReturn(w: WarningItem, reason: string) {
- updateWarning(w.id, { status: "returned" }, "退回预警", reason)
- globalMessage.warning("预警已退回处置单位")
- }
- function handleUpgrade(w: WarningItem, newLevel: Level, reason?: string) {
- updateWarning(w.id, { status: "upgraded", level: newLevel }, "升级预警", reason)
- notification.info({ message: "预警已升级", description: `${w.code} -> ${newLevel}` })
- }
- function handleSupervise(w: WarningItem, payload: { people: string[]; opinion: string; channels: string[] }) {
- updateWarning(
- w.id,
- { status: "supervising" },
- "督办",
- `人员: ${payload.people.join(",")} | 方式: ${payload.channels.join(",")} | 意见: ${payload.opinion}`,
- )
- notification.warning({
- message: "已发起督办",
- description: `${w.name} - ${payload.people.join(", ")}`,
- })
- }
- function handleLeaderInstruction(w: WarningItem, payload: { leaders: string[]; sms: string }) {
- updateWarning(w.id, {}, "领导批示", `已短信推送给: ${payload.leaders.join(", ")};内容:${payload.sms}`)
- notification.success({
- message: "已推送领导批示",
- description: payload.sms,
- })
- }
- function handlePublish(values: any) {
- const seq = warnings.length + 1
- const code = autoCode(values.type, values.level, seq)
- const newItem: WarningItem = {
- id: genId("warn"),
- code,
- name: values.name,
- type: values.type,
- level: values.level,
- industry: values.type,
- status: "published",
- createdAt: values.releasedAt?.toISOString?.() ?? new Date().toISOString(),
- releasedAt: undefined,
- deadline: dayjs().add(2, "day").toISOString(),
- assignee: values.assignee,
- handler: currentUser,
- location: values.location,
- coords: [120.18 + Math.random() * 0.1, 30.25 + Math.random() * 0.1],
- description: values.description,
- attachments: values.attachments?.fileList ?? [],
- relatedAlarms: seedAlarms.slice(0, Math.min(3, seedAlarms.length)),
- process: [
- {
- time: new Date().toISOString(),
- user: currentUser,
- action: "发布预警",
- remark: values.description,
- },
- ],
- }
- setWarnings((prev) => [newItem, ...prev])
- setPublishOpen(false)
- publishForm.resetFields()
- globalMessage.success("预警已发布")
- setActiveTab("publish")
- }
- // Filters
- const filteredMgmtList = useMemo(() => {
- return warnings.filter((w) => {
- if (todayOnly && !isToday(w.createdAt)) return false
- if (mgmtFilters.type && w.type !== mgmtFilters.type) return false
- if (mgmtFilters.level && w.level !== mgmtFilters.level) return false
- if (mgmtFilters.industry && w.industry !== mgmtFilters.industry) return false
- if (mgmtFilters.status && w.status !== mgmtFilters.status) return false
- if (mgmtFilters.date && mgmtFilters.date.length === 2) {
- const [s, e] = mgmtFilters.date
- if (!dayjs(w.createdAt).isBetween(s, e, "day", "[]")) return false
- }
- if (mgmtFilters.q) {
- const q = mgmtFilters.q.trim()
- if (!q) return true
- return w.name.includes(q) || w.code.includes(q) || w.location?.includes(q) || w.description?.includes(q)
- }
- return true
- })
- }, [warnings, mgmtFilters, todayOnly])
- // Tables
- const warnColumns: ColumnsType<WarningItem> = [
- {
- title: "预警名称",
- dataIndex: "name",
- key: "name",
- render: (text, record) => (
- <Space>
- <Tooltip title={record.type}>
- <AntdBadge color="geekblue" />
- </Tooltip>
- <a onClick={() => setDetail(record)}>{text}</a>
- {statusTag(record.status)}
- </Space>
- ),
- },
- { title: "编码", dataIndex: "code", key: "code" },
- {
- title: "类型/行业",
- key: "type",
- render: (_, r) => (
- <Space size={6}>
- <Tag>{r.type}</Tag>
- <Tag color={levelColor(r.level)}>{r.level}</Tag>
- </Space>
- ),
- },
- { title: "位置", dataIndex: "location", key: "location" },
- {
- title: "创建时间",
- dataIndex: "createdAt",
- key: "createdAt",
- render: (t: string) => dayjs(t).format("YYYY-MM-DD HH:mm"),
- },
- { title: "责任人", dataIndex: "assignee", key: "assignee" },
- {
- title: "操作",
- key: "actions",
- fixed: "right",
- render: (_, r) => (
- <Space wrap>
- <Button size="small" onClick={() => setDetail(r)}>
- 详情
- </Button>
- <Button size="small" type="default" onClick={() => setUpgradeOpen(r)} icon={<ChevronsUp size={14} />}>
- 升级
- </Button>
- <Button size="small" type="dashed" onClick={() => setSuperviseOpen(r)} icon={<UserRoundCheck size={14} />}>
- 督办
- </Button>
- <Button size="small" type="dashed" onClick={() => setLeaderOpen(r)} icon={<BellRing size={14} />}>
- 批示
- </Button>
- <Button size="small" danger onClick={() => setReturnOpen(r)} icon={<Alert size={14} />}>
- 退回
- </Button>
- <Popconfirm title="确认解除该预警?" onConfirm={() => handleRelease(r)} okText="解除" cancelText="取消">
- <Button size="small" type="primary" ghost icon={<ShieldAlert size={14} />}>
- 解除
- </Button>
- </Popconfirm>
- <Popconfirm title="标记为已处置?" onConfirm={() => handleResolve(r)} okText="确定" cancelText="取消">
- <Button size="small" type="primary" icon={<CheckCircle2 size={14} />}>
- 已处置
- </Button>
- </Popconfirm>
- </Space>
- ),
- },
- ]
- // Charts options
- const todayPieOption = useMemo(() => {
- return {
- tooltip: { trigger: "item" },
- legend: { top: "bottom" },
- series: [
- {
- name: "今日预警",
- type: "pie",
- radius: ["40%", "60%"],
- label: { formatter: "{b}: {c} ({d}%)" },
- data: [
- { name: "未解除", value: todayStats.unresolved },
- { name: "已解除", value: todayStats.released },
- ],
- },
- ],
- color: ["#f97316", "#10b981"],
- }
- }, [todayStats])
- const historyBarOption = useMemo(() => {
- return {
- tooltip: { trigger: "axis" },
- legend: { data: ["各专项累计"] },
- xAxis: { type: "category", data: historyByType.map((d) => d.name) },
- yAxis: { type: "value" },
- series: [
- {
- name: "各专项累计",
- type: "bar",
- data: historyByType.map((d) => d.value),
- itemStyle: { color: "#6b7280" },
- },
- ],
- }
- }, [historyByType])
- const levelRingOption = useMemo(() => {
- return {
- tooltip: { trigger: "item" },
- series: [
- {
- name: "等级分布",
- type: "pie",
- radius: ["30%", "55%"],
- label: { formatter: "{b}: {c}" },
- data: historyByLevel,
- color: ["#ef4444", "#fb923c", "#facc15", "#3b82f6"],
- },
- ],
- }
- }, [historyByLevel])
- const trendOption = useMemo(() => {
- const days = Array.from({ length: 14 }, (_, i) =>
- dayjs()
- .subtract(13 - i, "day")
- .format("MM-DD"),
- )
- const series = TYPES.map((t) => ({
- name: t,
- type: "line",
- smooth: true,
- data: days.map(() => Math.floor(Math.random() * 6)),
- areaStyle: { opacity: 0.08 },
- emphasis: { focus: "series" as const },
- }))
- return {
- tooltip: { trigger: "axis" },
- legend: { top: 0 },
- grid: { left: 24, right: 16, bottom: 24, top: 40 },
- xAxis: { type: "category", data: days },
- yAxis: { type: "value" },
- series,
- }
- }, [])
- const efficiencyTopOption = useMemo(() => {
- const data = TYPES.map((t) => ({
- name: t,
- value: +(80 + Math.random() * 20).toFixed(1),
- })).sort((a, b) => b.value - a.value)
- return {
- tooltip: { trigger: "axis" },
- xAxis: { type: "value", max: 100 },
- yAxis: { type: "category", data: data.map((d) => d.name), inverse: true },
- series: [
- {
- type: "bar",
- data: data.map((d) => d.value),
- itemStyle: {
- color: (params: any) => ["#10b981", "#22c55e", "#84cc16", "#eab308", "#f97316"][params.dataIndex % 5],
- },
- label: { show: true, position: "right", formatter: "{c}%" },
- },
- ],
- }
- }, [])
- // Modals
- function UpgradeModal() {
- const [form] = Form.useForm<{ level: Level; reason: string }>()
- const rec = upgradeOpen
- return (
- <Modal
- title="预警升级"
- open={!!rec}
- onCancel={() => setUpgradeOpen(null)}
- onOk={() => {
- form.validateFields().then((vals) => {
- if (rec) handleUpgrade(rec, vals.level, vals.reason)
- setUpgradeOpen(null)
- })
- }}
- okText="升级"
- >
- <Form form={form} layout="vertical" initialValues={{ level: rec?.level }}>
- <Form.Item label="新等级" name="level" rules={[{ required: true, message: "请选择新等级" }]}>
- <Select options={LEVELS.map((l) => ({ label: l, value: l }))} placeholder="选择新等级" />
- </Form.Item>
- <Form.Item label="升级原因" name="reason" rules={[{ required: true }]}>
- <Input.TextArea placeholder="说明升级原因..." rows={3} />
- </Form.Item>
- </Form>
- </Modal>
- )
- }
- function SuperviseModal() {
- const [form] = Form.useForm<{ people: string[]; channels: string[]; opinion: string }>()
- const rec = superviseOpen
- return (
- <Modal
- title="预警督办"
- open={!!rec}
- onCancel={() => setSuperviseOpen(null)}
- onOk={() => {
- form.validateFields().then((vals) => {
- if (rec) handleSupervise(rec, vals)
- setSuperviseOpen(null)
- })
- }}
- okText="督办"
- >
- <Form form={form} layout="vertical" initialValues={{ channels: ["平台通知"] }}>
- <Form.Item label="督办人员" name="people" rules={[{ required: true, message: "请选择督办人员" }]}>
- <Select mode="multiple" options={USERS.map((u) => ({ label: u, value: u }))} placeholder="选择人员" />
- </Form.Item>
- <Form.Item label="通知方式" name="channels">
- <Select mode="multiple" options={["短信", "邮件", "平台通知"].map((c) => ({ label: c, value: c }))} />
- </Form.Item>
- <Form.Item label="督办意见" name="opinion" rules={[{ required: true }]}>
- <Input.TextArea rows={3} placeholder="输入督办意见..." />
- </Form.Item>
- </Form>
- </Modal>
- )
- }
- function LeaderModal() {
- const [form] = Form.useForm<{ leaders: string[]; sms: string }>()
- const rec = leaderOpen
- return (
- <Modal
- title="添加领导批示(短信推送)"
- open={!!rec}
- onCancel={() => setLeaderOpen(null)}
- onOk={() => {
- form.validateFields().then((vals) => {
- if (rec) handleLeaderInstruction(rec, vals)
- setLeaderOpen(null)
- })
- }}
- okText="推送"
- >
- <Form form={form} layout="vertical" initialValues={{ leaders: [LEADERS[0]] }}>
- <Form.Item label="接收领导" name="leaders" rules={[{ required: true, message: "请选择接收人" }]}>
- <Select mode="multiple" options={LEADERS.map((l) => ({ label: l, value: l }))} />
- </Form.Item>
- <Form.Item label="短信内容" name="sms" rules={[{ required: true, message: "请输入短信内容" }]}>
- <Input.TextArea rows={3} maxLength={200} showCount />
- </Form.Item>
- </Form>
- </Modal>
- )
- }
- function ReturnModal() {
- const [form] = Form.useForm<{ reason: string }>()
- const rec = returnOpen
- return (
- <Modal
- title="退回预警"
- open={!!rec}
- onCancel={() => setReturnOpen(null)}
- onOk={() => {
- form.validateFields().then((vals) => {
- if (rec) handleReturn(rec, vals.reason)
- setReturnOpen(null)
- })
- }}
- okText="退回"
- >
- <Form form={form} layout="vertical">
- <Form.Item label="退回原因" name="reason" rules={[{ required: true, message: "请输入退回原因" }]}>
- <Input.TextArea rows={3} placeholder="说明退回原因..." />
- </Form.Item>
- </Form>
- </Modal>
- )
- }
- function DetailDrawer() {
- const rec = detail
- const stepIndex: number = useMemo(() => {
- if (!rec) return 0
- const stepMap: Record<WarningStatus, number> = {
- draft: 0,
- published: 1,
- in_process: 2,
- supervising: 2,
- returned: 2,
- upgraded: 2,
- resolved: 3,
- released: 4,
- }
- return stepMap[rec.status]
- }, [rec])
- return (
- <Drawer
- title={
- <Space>
- <Siren />
- <span>预警详情</span>
- {rec ? statusTag(rec.status) : null}
- </Space>
- }
- width={720}
- open={!!rec}
- onClose={() => setDetail(null)}
- >
- {!rec ? null : (
- <Space direction="vertical" size={16} className="w-full">
- <Descriptions bordered size="small" column={2}>
- <Descriptions.Item label="预警名称">{rec.name}</Descriptions.Item>
- <Descriptions.Item label="编码">{rec.code}</Descriptions.Item>
- <Descriptions.Item label="类型">{rec.type}</Descriptions.Item>
- <Descriptions.Item label="等级">
- <Tag color={levelColor(rec.level)}>{rec.level}</Tag>
- </Descriptions.Item>
- <Descriptions.Item label="行业">{rec.industry}</Descriptions.Item>
- <Descriptions.Item label="责任人">{rec.assignee}</Descriptions.Item>
- <Descriptions.Item label="创建时间">{dayjs(rec.createdAt).format("YYYY-MM-DD HH:mm")}</Descriptions.Item>
- <Descriptions.Item label="截止时间">
- {rec.deadline ? dayjs(rec.deadline).format("YYYY-MM-DD HH:mm") : "-"}
- </Descriptions.Item>
- <Descriptions.Item label="位置" span={2}>
- <Space>
- <MapPin size={16} />
- <span>{rec.location}</span>
- </Space>
- </Descriptions.Item>
- <Descriptions.Item label="描述" span={2}>
- {rec.description}
- </Descriptions.Item>
- </Descriptions>
- <Card size="small" title="GIS 定位(示意)">
- <img
- src="/placeholder.svg?height=220&width=660"
- alt="GIS 定位示意图"
- className="w-full rounded-md border"
- />
- </Card>
- <Card size="small" title="报警关联">
- <Table<Alarm>
- size="small"
- rowKey="id"
- pagination={false}
- dataSource={rec.relatedAlarms}
- columns={[
- { title: "设备", dataIndex: "device" },
- { title: "指标", dataIndex: "metric" },
- { title: "数值/阈值", render: (_, a) => `${a.value} / ${a.threshold}` },
- { title: "时间", dataIndex: "time", render: (t) => dayjs(t).format("YYYY-MM-DD HH:mm") },
- ]}
- />
- </Card>
- <Card size="small" title="处置流程(可视化)">
- <Steps
- current={stepIndex}
- items={[
- { title: "草稿" },
- { title: "已发布" },
- { title: "处置中/督办" },
- { title: "已处置" },
- { title: "已解除" },
- ]}
- />
- <Divider />
- <Table<ProcessLog>
- size="small"
- pagination={false}
- rowKey={(r) => `${r.time}-${r.action}`}
- dataSource={[...rec.process].reverse()}
- columns={[
- { title: "时间", dataIndex: "time", render: (t) => dayjs(t).format("YYYY-MM-DD HH:mm") },
- { title: "人员", dataIndex: "user" },
- { title: "动作", dataIndex: "action" },
- { title: "备注", dataIndex: "remark" },
- ]}
- />
- </Card>
- </Space>
- )}
- </Drawer>
- )
- }
- // Upload handler
- const normFile = (e: any) => {
- if (Array.isArray(e)) return e
- return e?.fileList
- }
- // Sections
- function Overview() {
- return (
- <Space direction="vertical" size={16} className="w-full">
- <Row gutter={16}>
- <Col xs={24} md={6}>
- <Card>
- <Statistic title="今日预警总数" value={todayStats.total} prefix={<Siren className="text-orange-500" />} />
- </Card>
- </Col>
- <Col xs={24} md={6}>
- <Card>
- <Statistic
- title="未解除"
- value={todayStats.unresolved}
- valueStyle={{ color: "#f97316" }}
- prefix={<Alert className="text-orange-500" />}
- />
- </Card>
- </Col>
- <Col xs={24} md={6}>
- <Card>
- <Statistic
- title="已解除"
- value={todayStats.released}
- valueStyle={{ color: "#10b981" }}
- prefix={<ShieldAlert className="text-emerald-500" />}
- />
- </Card>
- </Col>
- <Col xs={24} md={6}>
- <Card>
- <Statistic
- title="当前我的待办"
- value={mineTodo.length}
- prefix={<UserRoundCheck className="text-purple-500" />}
- />
- </Card>
- </Col>
- </Row>
- <Row gutter={16}>
- <Col xs={24} md={10}>
- <Card title="今日预警概况">
- <EChart option={todayPieOption} style={{ height: 260 }} opts={{ notMerge: true, lazyUpdate: true }} />
- <Divider />
- <Table<WarningItem>
- size="small"
- rowKey="id"
- pagination={{ pageSize: 5 }}
- dataSource={todayStats.list}
- columns={[
- { title: "名称", dataIndex: "name", render: (t, r) => <a onClick={() => setDetail(r)}>{t}</a> },
- { title: "等级", dataIndex: "level", render: (l: Level) => <Tag color={levelColor(l)}>{l}</Tag> },
- { title: "状态", dataIndex: "status", render: (s: WarningStatus) => statusTag(s) },
- ]}
- />
- </Card>
- </Col>
- <Col xs={24} md={14}>
- <Card title="历史预警概况">
- <Row gutter={12}>
- <Col span={14}>
- <EChart
- option={historyBarOption}
- style={{ height: 280 }}
- opts={{ notMerge: true, lazyUpdate: true }}
- />
- </Col>
- <Col span={10}>
- <EChart
- option={levelRingOption}
- style={{ height: 280 }}
- opts={{ notMerge: true, lazyUpdate: true }}
- />
- </Col>
- </Row>
- </Card>
- </Col>
- </Row>
- <Card title="今日未处置预警">
- <Table<WarningItem>
- size="small"
- rowKey="id"
- dataSource={todayUnhandled}
- pagination={{ pageSize: 8 }}
- columns={warnColumns}
- scroll={{ x: 1000 }}
- />
- </Card>
- </Space>
- )
- }
- function Publish() {
- return (
- <Space direction="vertical" size={16} className="w-full">
- <Card
- title={
- <Space>
- <Siren />
- <span>预警发布</span>
- </Space>
- }
- extra={
- <Button type="primary" onClick={() => setPublishOpen(true)} icon={<Siren />}>
- 新建预警
- </Button>
- }
- >
- <Form
- form={publishForm}
- layout="vertical"
- onFinish={handlePublish}
- initialValues={{
- type: "综合",
- level: "黄色",
- assignee: currentUser,
- }}
- >
- <Row gutter={12}>
- <Col xs={24} md={8}>
- <Form.Item label="预警名称" name="name" rules={[{ required: true, message: "请输入预警名称" }]}>
- <Input placeholder="如:某路段燃气压力异常预警" />
- </Form.Item>
- </Col>
- <Col xs={24} md={6}>
- <Form.Item label="预警类型" name="type" rules={[{ required: true }]}>
- <Select options={TYPES.map((t) => ({ label: t, value: t }))} />
- </Form.Item>
- </Col>
- <Col xs={24} md={6}>
- <Form.Item label="预警等级" name="level" rules={[{ required: true }]}>
- <Select options={LEVELS.map((l) => ({ label: l, value: l }))} />
- </Form.Item>
- </Col>
- <Col xs={24} md={4}>
- <Form.Item label="发布时间" name="releasedAt">
- <DatePicker showTime className="w-full" />
- </Form.Item>
- </Col>
- <Col xs={24} md={8}>
- <Form.Item label="位置" name="location">
- <Input prefix={<MapPin size={16} />} placeholder="输入预警位置" />
- </Form.Item>
- </Col>
- <Col xs={24} md={8}>
- <Form.Item label="责任人" name="assignee" rules={[{ required: true }]}>
- <Select options={USERS.map((u) => ({ label: u, value: u }))} />
- </Form.Item>
- </Col>
- <Col xs={24} md={8}>
- <Form.Item label="关联报警(示例)" name="alarms">
- <Select
- mode="multiple"
- placeholder="选择报警记录"
- options={seedAlarms.slice(0, 10).map((a) => ({
- label: `${a.device}-${a.metric}(${dayjs(a.time).format("MM-DD HH:mm")})`,
- value: a.id,
- }))}
- />
- </Form.Item>
- </Col>
- <Col span={24}>
- <Form.Item label="预警描述" name="description" rules={[{ required: true, message: "请输入描述" }]}>
- <Input.TextArea rows={4} placeholder="结合设备报警曲线、位置及周边要素分析,描述具体内容..." />
- </Form.Item>
- </Col>
- <Col span={24}>
- <Form.Item
- label="预警报告上传"
- valuePropName="fileList"
- getValueFromEvent={normFile}
- name="attachments"
- >
- <Upload.Dragger beforeUpload={() => false} multiple>
- <p className="ant-upload-drag-icon">
- <FileText />
- </p>
- <p className="ant-upload-text">点击或拖拽文件到此处上传</p>
- <p className="ant-upload-hint">支持多文件</p>
- </Upload.Dragger>
- </Form.Item>
- </Col>
- </Row>
- <Space>
- <Button type="primary" htmlType="submit">
- 发布预警
- </Button>
- <Button htmlType="reset">重置</Button>
- </Space>
- </Form>
- </Card>
- <Row gutter={16}>
- <Col xs={24} md={12}>
- <Card title="报警信息关联(统一接口示例)" extra={<Tag>仅演示</Tag>}>
- <Table<Alarm>
- size="small"
- rowKey="id"
- dataSource={seedAlarms.slice(0, 8)}
- pagination={false}
- columns={[
- { title: "设备", dataIndex: "device" },
- { title: "指标", dataIndex: "metric" },
- { title: "值/阈值", render: (_, a) => `${a.value}/${a.threshold}` },
- { title: "时间", dataIndex: "time", render: (t) => dayjs(t).format("MM-DD HH:mm") },
- ]}
- />
- </Card>
- </Col>
- <Col xs={24} md={12}>
- <Card title="发布记录(最近)">
- <Table<WarningItem>
- size="small"
- rowKey="id"
- pagination={false}
- dataSource={[...warnings].filter((w) => isToday(w.createdAt)).slice(0, 6)}
- columns={[
- { title: "名称", dataIndex: "name" },
- { title: "编码", dataIndex: "code" },
- { title: "等级", dataIndex: "level", render: (l: Level) => <Tag color={levelColor(l)}>{l}</Tag> },
- { title: "状态", dataIndex: "status", render: (s: WarningStatus) => statusTag(s) },
- ]}
- />
- </Card>
- </Col>
- </Row>
- <Modal
- title="快速发布预警"
- open={publishOpen}
- onCancel={() => setPublishOpen(false)}
- footer={null}
- destroyOnHidden
- >
- <Form
- form={publishForm}
- layout="vertical"
- onFinish={handlePublish}
- initialValues={{ type: "综合", level: "黄色", assignee: currentUser }}
- >
- <Form.Item label="预警名称" name="name" rules={[{ required: true }]}>
- <Input />
- </Form.Item>
- <Form.Item label="类型" name="type" rules={[{ required: true }]}>
- <Select options={TYPES.map((t) => ({ label: t, value: t }))} />
- </Form.Item>
- <Form.Item label="等级" name="level" rules={[{ required: true }]}>
- <Select options={LEVELS.map((l) => ({ label: l, value: l }))} />
- </Form.Item>
- <Form.Item label="责任人" name="assignee" rules={[{ required: true }]}>
- <Select options={USERS.map((u) => ({ label: u, value: u }))} />
- </Form.Item>
- <Form.Item label="描述" name="description" rules={[{ required: true }]}>
- <Input.TextArea rows={3} />
- </Form.Item>
- <Space>
- <Button htmlType="submit" type="primary">
- 发布
- </Button>
- <Button onClick={() => setPublishOpen(false)}>取消</Button>
- </Space>
- </Form>
- </Modal>
- </Space>
- )
- }
- function Handling() {
- return (
- <Space direction="vertical" size={16} className="w-full">
- <Card
- title={
- <Space>
- <SquareGanttChart />
- <span>预警处置</span>
- </Space>
- }
- >
- <Table<WarningItem>
- size="small"
- rowKey="id"
- dataSource={warnings.filter((w) =>
- ["published", "in_process", "supervising", "upgraded", "returned"].includes(w.status),
- )}
- columns={warnColumns}
- pagination={{ pageSize: 8 }}
- scroll={{ x: 1000 }}
- />
- </Card>
- <UpgradeModal />
- <SuperviseModal />
- <LeaderModal />
- <ReturnModal />
- </Space>
- )
- }
- function Todos() {
- const list = todoFilter.status === "todo" ? mineTodo : mineDone
- const data = list.filter((w) => {
- if (!todoFilter.q) return true
- const q = todoFilter.q.trim()
- if (!q) return true
- return w.name.includes(q) || w.code.includes(q)
- })
- return (
- <Space direction="vertical" size={16} className="w-full">
- <Card
- title={
- <Space>
- <UserRoundCheck />
- <span>我的预警</span>
- </Space>
- }
- extra={
- <Space>
- <Segmented
- value={todoFilter.status}
- onChange={(v) => setTodoFilter((p) => ({ ...p, status: v as any }))}
- options={[
- { label: "我的待办", value: "todo" },
- { label: "我的已办", value: "done" },
- ]}
- />
- <Input.Search
- allowClear
- placeholder="按名称/编号筛选"
- onSearch={(q) => setTodoFilter((p) => ({ ...p, q }))}
- />
- </Space>
- }
- >
- <Table<WarningItem>
- size="small"
- rowKey="id"
- dataSource={data}
- columns={warnColumns}
- pagination={{ pageSize: 8 }}
- scroll={{ x: 1000 }}
- />
- </Card>
- </Space>
- )
- }
- function Management() {
- function SearchButton() {
- return (
- <Space>
- <ChartBar size={16} />
- <span>查询</span>
- </Space>
- )
- }
- return (
- <Space direction="vertical" size={16} className="w-full">
- <Card
- title={
- <Space>
- <Settings />
- <span>预警信息管理</span>
- </Space>
- }
- >
- <Space wrap className="w-full">
- <Select
- allowClear
- placeholder="类型"
- options={TYPES.map((t) => ({ label: t, value: t }))}
- onChange={(type) => setMgmtFilters((p) => ({ ...p, type: type as WarnType | undefined }))}
- style={{ width: 140 }}
- />
- <Select
- allowClear
- placeholder="等级"
- options={LEVELS.map((l) => ({ label: l, value: l }))}
- onChange={(level) => setMgmtFilters((p) => ({ ...p, level: level as Level | undefined }))}
- style={{ width: 140 }}
- />
- <Select
- allowClear
- placeholder="行业"
- options={INDUSTRIES.map((i) => ({ label: i, value: i }))}
- onChange={(industry) => setMgmtFilters((p) => ({ ...p, industry: industry as Industry | undefined }))}
- style={{ width: 140 }}
- />
- <Select
- allowClear
- placeholder="状态"
- options={[
- "draft",
- "published",
- "in_process",
- "supervising",
- "returned",
- "upgraded",
- "resolved",
- "released",
- ].map((s) => ({ label: s, value: s }))}
- onChange={(status) => setMgmtFilters((p) => ({ ...p, status: status as WarningStatus | undefined }))}
- style={{ width: 160 }}
- />
- <DatePicker.RangePicker
- onChange={(v) => setMgmtFilters((p) => ({ ...p, date: (v as any) || undefined }))}
- />
- <Input.Search
- allowClear
- placeholder="名称/编码/位置/描述"
- onSearch={(q) => setMgmtFilters((p) => ({ ...p, q }))}
- style={{ width: 280 }}
- enterButton={<SearchButton />}
- />
- <Radio.Group
- value={todayOnly}
- onChange={(e) => setTodayOnly(e.target.value)}
- optionType="button"
- options={[
- { label: "今日", value: true },
- { label: "全部", value: false },
- ]}
- />
- </Space>
- <Divider />
- <Table<WarningItem>
- size="small"
- rowKey="id"
- dataSource={filteredMgmtList}
- columns={warnColumns}
- pagination={{ pageSize: 10 }}
- scroll={{ x: 1000 }}
- />
- </Card>
- </Space>
- )
- }
- function Statistics() {
- const statusAgg = useMemo(() => {
- const agg = new Map<WarningStatus, number>()
- warnings.forEach((w) => agg.set(w.status, (agg.get(w.status) || 0) + 1))
- return Array.from(agg.entries()).map(([name, value]) => ({ name, value }))
- }, [warnings])
- const statusPie = {
- tooltip: { trigger: "item" },
- series: [{ type: "pie", radius: ["35%", "60%"], data: statusAgg }],
- color: ["#64748b", "#60a5fa", "#34d399", "#f59e0b", "#ef4444", "#a78bfa", "#22c55e", "#93c5fd"],
- }
- const todayNonDraftResolvedReleased = warnings.filter((w) => isToday(w.createdAt) && !["draft"].includes(w.status))
- return (
- <Space direction="vertical" size={16} className="w-full">
- <Row gutter={16}>
- <Col xs={24} md={8}>
- <Card title="处置状态分析">
- <EChart option={statusPie} style={{ height: 260 }} opts={{ notMerge: true, lazyUpdate: true }} />
- </Card>
- </Col>
- <Col xs={24} md={8}>
- <Card title="今日预警统计(非草稿)">
- <Table<WarningItem>
- size="small"
- rowKey="id"
- pagination={{ pageSize: 5 }}
- dataSource={todayNonDraftResolvedReleased}
- columns={[
- { title: "名称", dataIndex: "name" },
- { title: "等级", dataIndex: "level", render: (l: Level) => <Tag color={levelColor(l)}>{l}</Tag> },
- { title: "状态", dataIndex: "status", render: (s: WarningStatus) => statusTag(s) },
- ]}
- />
- </Card>
- </Col>
- <Col xs={24} md={8}>
- <Card title="接警效率 TOP(按类型)">
- <EChart
- option={efficiencyTopOption}
- style={{ height: 260 }}
- opts={{ notMerge: true, lazyUpdate: true }}
- />
- </Card>
- </Col>
- </Row>
- <Row gutter={16}>
- <Col xs={24} md={14}>
- <Card
- title={
- <Space>
- <LineChart />
- <span>预警发展趋势(14日)</span>
- </Space>
- }
- >
- <EChart option={trendOption} style={{ height: 320 }} opts={{ notMerge: true, lazyUpdate: true }} />
- </Card>
- </Col>
- <Col xs={24} md={10}>
- <Card title="行业处置分析">
- <EChart
- option={{
- tooltip: { trigger: "axis" },
- legend: { data: ["处置中", "已处置", "已解除"] },
- xAxis: { type: "category", data: INDUSTRIES },
- yAxis: { type: "value" },
- series: [
- {
- name: "处置中",
- type: "bar",
- stack: "total",
- data: INDUSTRIES.map(
- (i) =>
- warnings.filter(
- (w) =>
- w.industry === i &&
- ["in_process", "supervising", "upgraded", "returned"].includes(w.status),
- ).length,
- ),
- },
- {
- name: "已处置",
- type: "bar",
- stack: "total",
- data: INDUSTRIES.map(
- (i) => warnings.filter((w) => w.industry === i && w.status === "resolved").length,
- ),
- itemStyle: { color: "#22c55e" },
- },
- {
- name: "已解除",
- type: "bar",
- stack: "total",
- data: INDUSTRIES.map(
- (i) => warnings.filter((w) => w.industry === i && w.status === "released").length,
- ),
- itemStyle: { color: "#10b981" },
- },
- ],
- }}
- style={{ height: 320 }}
- opts={{ notMerge: true, lazyUpdate: true }}
- />
- </Card>
- </Col>
- </Row>
- </Space>
- )
- }
- return (
- <main className="min-h-screen w-full bg-slate-50">
- <div className="mx-auto max-w-[1400px] px-4 py-6">
- <Space direction="vertical" size={16} className="w-full">
- <Space align="center" className="w-full justify-between">
- <Space>
- <Siren className="text-orange-500" />
- <h1 className="text-2xl font-semibold">生命线预警联动处置平台</h1>
- </Space>
- <Space>
- <AntdBadge status="processing" text={`当前用户:${currentUser}`} />
- </Space>
- </Space>
- <Tabs
- activeKey={activeTab}
- onChange={(k) => {
- setActiveTab(k as string)
- // 切换 Tab 时主动触发一次窗口 resize,配合组件内的 ResizeObserver,确保图表正确自适应
- if (typeof window !== "undefined") {
- setTimeout(() => window.dispatchEvent(new Event("resize")), 0)
- }
- }}
- items={[
- {
- key: "overview",
- label: (
- <Space>
- <Timeline size={16} />
- <span>预警总览</span>
- </Space>
- ),
- children: <Overview />,
- },
- {
- key: "publish",
- label: (
- <Space>
- <Siren size={16} />
- <span>预警信息发布</span>
- </Space>
- ),
- children: <Publish />,
- },
- {
- key: "todos",
- label: (
- <Space>
- <UserRoundCheck size={16} />
- <span>待办预警</span>
- </Space>
- ),
- children: <Todos />,
- },
- {
- key: "handling",
- label: (
- <Space>
- <SquareGanttChart size={16} />
- <span>预警处置</span>
- </Space>
- ),
- children: <Handling />,
- },
- {
- key: "management",
- label: (
- <Space>
- <Settings size={16} />
- <span>预警信息管理</span>
- </Space>
- ),
- children: <Management />,
- },
- {
- key: "matters",
- label: (
- <Space>
- <FileText size={16} />
- <span>预警清单管理</span>
- </Space>
- ),
- children: (
- <MatterManagement
- matters={matters}
- setMatters={setMatters}
- reasons={reasons}
- setReasons={setReasons}
- />
- ),
- },
- {
- key: "stats",
- label: (
- <Space>
- <ChartBar size={16} />
- <span>预警信息统计</span>
- </Space>
- ),
- children: <Statistics />,
- },
- ]}
- />
- <DetailDrawer />
- </Space>
- </div>
- </main>
- )
- }
|