feat: add firearm page for display firearms

This commit is contained in:
2026-04-06 20:40:42 +08:00
parent 38b25099de
commit 452b807b62
2 changed files with 64 additions and 163 deletions
+3 -1
View File
@@ -1,4 +1,6 @@
@import "tailwindcss"; @layer theme, base, antd, components, utilities;
@import 'tailwindcss';
html, body { html, body {
margin: 0; margin: 0;
+61 -162
View File
@@ -1,9 +1,7 @@
import { useEffect, useMemo, useState } from "react" import { useEffect, useState } from "react"
import { Link } from "react-router-dom"
import { FirearmApi } from "@/api" import { FirearmApi } from "@/api"
import { Firearm, FirearmType } from "@/types" import { Firearm, FirearmType } from "@/types"
import { setFirearmsPage } from "@/store/firearms-slice" import { Card, Col, Pagination, Row, Tag, Typography } from "antd"
import { useAppDispatch, useAppSelector } from "@/store"
const firearmTypeText: Record<FirearmType, string> = { const firearmTypeText: Record<FirearmType, string> = {
RIFLE: "步枪", RIFLE: "步枪",
@@ -17,171 +15,72 @@ const firearmTypeText: Record<FirearmType, string> = {
} }
export default function FirearmsPage() { export default function FirearmsPage() {
const pageSize = 12 const [page, setPage] = useState<number>(1)
const dispatch = useAppDispatch() const [firearms, setFirearms] = useState<Firearm[]>([])
const firearmsState = useAppSelector((state) => state.firearms) const [total, setTotal] = useState<number>(0)
const firearms = firearmsState.items
const [isLoading, setIsLoading] = useState<boolean>(true)
const [isRefreshing, setIsRefreshing] = useState<boolean>(false)
const [loadError, setLoadError] = useState<string | null>(null)
const [keyword, setKeyword] = useState<string>("")
const [activeType, setActiveType] = useState<"全部" | FirearmType>("全部")
const [currentPage, setCurrentPage] = useState<number>(firearmsState.page)
const fetchFirearms = (page: number, forceRefresh = false) => {
if (!forceRefresh && firearms.length > 0 && page === firearmsState.page) {
setIsLoading(false)
setLoadError(null)
return
}
setIsLoading(true)
setLoadError(null)
if (forceRefresh) {
setIsRefreshing(true)
}
useEffect(() => {
FirearmApi.getFirearms({ FirearmApi.getFirearms({
page, page: page - 1,
size: pageSize, size: 12,
sortBy: "name", sortBy: "id",
direction: "ASC", direction: "ASC",
}).then((pagedData) => {
setFirearms(pagedData.items)
setTotal(pagedData.totalElements)
}) })
.then((page) => { }, [page])
dispatch(setFirearmsPage(page))
})
.catch(() => {
setLoadError("武器列表加载失败,请确认后端服务是否已启动。")
})
.finally(() => {
setIsLoading(false)
setIsRefreshing(false)
})
}
useEffect(() => {
fetchFirearms(currentPage, false)
}, [currentPage, dispatch])
useEffect(() => {
if (firearmsState.page !== currentPage) {
setCurrentPage(firearmsState.page)
}
}, [currentPage, firearmsState.page])
const filteredFirearms = useMemo(() => {
const trimmed = keyword.trim().toLowerCase()
return firearms.filter((item) => {
const matchKeyword = !trimmed || item.name.toLowerCase().includes(trimmed)
const matchType = activeType === "全部" || item.type === activeType
return matchKeyword && matchType
})
}, [activeType, firearms, keyword])
return ( return (
<section className="space-y-6"> <>
<div className="space-y-4 px-1"> <div className="mb-6">
<div className="flex flex-wrap items-center justify-between gap-3"> <Row gutter={[16, 16]}>
<p className="text-sm text-gray-600"></p> {firearms.map((firearm) => (
<button <Col key={firearm.id} xs={24} md={12} lg={8}>
type="button" <Card
onClick={() => fetchFirearms(currentPage, true)} title={firearm.name}
disabled={isRefreshing} variant="outlined"
className="rounded-lg border border-gray-300 px-4 py-2 text-sm text-gray-700 hover:bg-gray-50 disabled:cursor-not-allowed disabled:opacity-60" styles={{
> header: { minHeight: 56 },
{isRefreshing ? "刷新中..." : "强制刷新"} }}
</button> >
</div> <div className="flex flex-col gap-3">
{isLoading ? <p className="text-sm text-gray-500">...</p> : null} <div className="flex items-center justify-between">
{loadError ? <p className="text-sm text-red-600">{loadError}</p> : null} <Tag color="blue">{firearmTypeText[firearm.type]}</Tag>
<Typography.Text type="secondary">ID: {firearm.id}</Typography.Text>
<div className="grid gap-3 sm:grid-cols-2 max-w-2xl"> </div>
<label className="block"> <Typography.Text>
<span className="block text-sm font-medium text-gray-700 mb-1"></span> <strong></strong>
<input {firearm.level}
value={keyword} </Typography.Text>
onChange={(event) => setKeyword(event.target.value)} <Typography.Paragraph className="!mb-0" type="secondary" ellipsis={{ rows: 3 }}>
placeholder="例如:M4A1" {firearm.review || "暂无描述"}
className="w-full rounded-lg border border-gray-300 px-3 py-2 text-sm outline-none focus:border-blue-500 focus:ring-2 focus:ring-blue-100 bg-white" </Typography.Paragraph>
/> </div>
</label> </Card>
</Col>
<label className="block"> ))}
<span className="block text-sm font-medium text-gray-700 mb-1"></span> {firearms.length === 0 && (
<select <Col span={24}>
value={activeType} <Card>
onChange={(event) => setActiveType(event.target.value as "全部" | FirearmType)} <Typography.Text type="secondary"></Typography.Text>
className="w-full rounded-lg border border-gray-300 px-3 py-2 text-sm outline-none focus:border-blue-500 focus:ring-2 focus:ring-blue-100 bg-white" </Card>
> </Col>
<option value="全部"></option> )}
{Object.entries(firearmTypeText).map(([type, text]) => ( </Row>
<option key={type} value={type}>{text}</option>
))}
</select>
</label>
</div>
</div> </div>
<div className="flex justify-end">
<div className="grid grid-cols-1 sm:grid-cols-2 lg:grid-cols-3 gap-4"> <Pagination
{filteredFirearms.map((item) => ( align="end"
<article key={item.id} className="bg-white border rounded-xl p-4 shadow-sm space-y-3"> current={page}
<div className="flex items-start justify-between gap-3"> pageSize={12}
<h2 className="text-lg font-semibold text-gray-900">{item.name}</h2> total={total}
<span className="text-xs rounded-full bg-gray-100 text-gray-700 px-2 py-1"> onChange={(nextPage) => {
{firearmTypeText[item.type]} setPage(nextPage)
</span> }}
</div> showSizeChanger={false}
/>
<div className="flex flex-wrap gap-2 text-xs text-gray-600">
{item.level ? (
<span className="rounded-full bg-gray-100 px-2 py-1">{item.level}</span>
) : null}
</div>
{item.review ? (
<p className="text-sm text-gray-600 whitespace-pre-line">{item.review}</p>
) : null}
<Link
to={`/mod-codes?firearmId=${encodeURIComponent(item.id)}`}
className="inline-flex items-center justify-center rounded-lg border border-blue-300 bg-blue-50 px-3 py-2 text-sm font-medium text-blue-700 hover:bg-blue-100">
</Link>
</article>
))}
</div> </div>
</>
{!isLoading && filteredFirearms.length === 0 ? (
<div className="bg-white border rounded-xl p-6 text-center text-gray-600">
</div>
) : null}
<div className="flex flex-wrap items-center justify-between gap-3 rounded-lg border border-gray-200 bg-white px-3 py-2">
<p className="text-sm text-gray-600">
{firearmsState.page + 1} / {Math.max(firearmsState.totalPages, 1)} {firearmsState.totalElements}
</p>
<div className="flex items-center gap-2">
<button
type="button"
disabled={isLoading || currentPage <= 0}
onClick={() => setCurrentPage((page) => Math.max(page - 1, 0))}
className="rounded-lg border border-gray-300 px-3 py-1.5 text-sm text-gray-700 hover:bg-gray-50 disabled:cursor-not-allowed disabled:opacity-60"
>
</button>
<button
type="button"
disabled={isLoading || firearmsState.totalPages === 0 || currentPage >= firearmsState.totalPages - 1}
onClick={() => setCurrentPage((page) => page + 1)}
className="rounded-lg border border-gray-300 px-3 py-1.5 text-sm text-gray-700 hover:bg-gray-50 disabled:cursor-not-allowed disabled:opacity-60"
>
</button>
</div>
</div>
</section>
) )
} }