|
@@ -0,0 +1,199 @@
|
|
|
+"use client";
|
|
|
+
|
|
|
+import Alert from "../ui/components/Alert";
|
|
|
+import ExchangeForm from "./ExchangeForm";
|
|
|
+import { ChevronLeft } from "lucide-react";
|
|
|
+import Image from "next/image";
|
|
|
+import { useRouter } from "next/navigation";
|
|
|
+import { useState, useEffect } from "react";
|
|
|
+import InfiniteScroll from "react-infinite-scroll-component";
|
|
|
+
|
|
|
+const PAGE_SIZE = 10;
|
|
|
+
|
|
|
+export default function ExchangePage() {
|
|
|
+ const [items, setItems] = useState([]);
|
|
|
+ const [selectedItem, setSelectedItem] = useState(null);
|
|
|
+ const [alert, setAlert] = useState(null);
|
|
|
+
|
|
|
+ const [exchanges, setExchanges] = useState([]);
|
|
|
+ const [page, setPage] = useState(1);
|
|
|
+ const [hasMore, setHasMore] = useState(true);
|
|
|
+ const [currentUser, setCurrentUser] = useState(null);
|
|
|
+
|
|
|
+ const router = useRouter();
|
|
|
+
|
|
|
+ useEffect(() => {
|
|
|
+ fetchExchangeItems();
|
|
|
+ const user = JSON.parse(localStorage.getItem("currentUser") || "null");
|
|
|
+ setCurrentUser(user);
|
|
|
+ }, []);
|
|
|
+
|
|
|
+ useEffect(() => {
|
|
|
+ loadExchanges();
|
|
|
+ }, [currentUser]);
|
|
|
+
|
|
|
+ const loadExchanges = async () => {
|
|
|
+ if (!currentUser) return;
|
|
|
+ try {
|
|
|
+ const response = await fetch(
|
|
|
+ `/api/exchange-history?userId=${encodeURIComponent(
|
|
|
+ currentUser.id
|
|
|
+ )}¤t=${page}&pageSize=${PAGE_SIZE}`
|
|
|
+ );
|
|
|
+ const data = await response.json();
|
|
|
+ if (data.success) {
|
|
|
+ setExchanges((prev) => [...prev, ...data.data]);
|
|
|
+ setPage((prev) => prev + 1);
|
|
|
+ setHasMore(data.data.length === PAGE_SIZE);
|
|
|
+ } else {
|
|
|
+ console.error("Failed to fetch exchange history:", data.error);
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error("Error fetching exchange history:", error);
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ const fetchExchangeItems = async () => {
|
|
|
+ try {
|
|
|
+ const response = await fetch("/api/exchange-items");
|
|
|
+ const res = await response.json();
|
|
|
+ if (res.success) {
|
|
|
+ setItems(res.data);
|
|
|
+ } else {
|
|
|
+ console.error("获取兑换项目失败:", res.message);
|
|
|
+ }
|
|
|
+ } catch (error) {
|
|
|
+ console.error("获取兑换项目失败:", error);
|
|
|
+ }
|
|
|
+ };
|
|
|
+
|
|
|
+ const handleExchange = (item) => {
|
|
|
+ setSelectedItem(item);
|
|
|
+ };
|
|
|
+
|
|
|
+ return (
|
|
|
+ <div className="bg-blue-600 text-white min-h-screen">
|
|
|
+ <div className="max-w-4xl mx-auto px-4 py-6">
|
|
|
+ <div className="flex items-center mb-6">
|
|
|
+ <ChevronLeft
|
|
|
+ className="cursor-pointer hover:text-blue-200 transition-colors"
|
|
|
+ onClick={() => router.back()}
|
|
|
+ size={28}
|
|
|
+ />
|
|
|
+ <h1 className="text-xl font-bold ml-4">积分兑换</h1>
|
|
|
+ </div>
|
|
|
+
|
|
|
+ <div className="grid gap-4 sm:grid-cols-2">
|
|
|
+ {items.map((item) => (
|
|
|
+ <div
|
|
|
+ key={item._id}
|
|
|
+ className="bg-white text-black rounded-lg p-4 shadow-md hover:shadow-lg transition-shadow"
|
|
|
+ >
|
|
|
+ <div className="flex items-center justify-between">
|
|
|
+ <div className="flex items-center flex-grow">
|
|
|
+ <Image
|
|
|
+ src={item.logo}
|
|
|
+ alt={item.title}
|
|
|
+ width={50}
|
|
|
+ height={50}
|
|
|
+ className="rounded-full mr-4 flex-shrink-0"
|
|
|
+ />
|
|
|
+ <div>
|
|
|
+ <h2
|
|
|
+ className="text-lg font-bold"
|
|
|
+ dangerouslySetInnerHTML={{ __html: item.title }}
|
|
|
+ ></h2>
|
|
|
+ <p className="text-gray-600 text-sm">
|
|
|
+ 所需积分: {item.points}
|
|
|
+ </p>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ <button
|
|
|
+ onClick={() => handleExchange(item)}
|
|
|
+ className="bg-blue-500 hover:bg-blue-600 text-white font-bold py-2 px-4 rounded transition duration-300 ml-4 flex-shrink-0"
|
|
|
+ >
|
|
|
+ 兑换
|
|
|
+ </button>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ ))}
|
|
|
+ </div>
|
|
|
+ {/* 积分预测记录 */}
|
|
|
+ <div className="bg-white text-black rounded-lg p-4 mb-4 mt-6">
|
|
|
+ <h3 className="text-lg font-bold mb-2 text-blue-700">积分兑换记录</h3>
|
|
|
+ <div
|
|
|
+ id="exchangeScroll"
|
|
|
+ style={{ height: "280px", overflow: "auto" }}
|
|
|
+ >
|
|
|
+ <InfiniteScroll
|
|
|
+ dataLength={exchanges.length}
|
|
|
+ next={loadExchanges}
|
|
|
+ hasMore={hasMore}
|
|
|
+ loader={<h4>Loading...</h4>}
|
|
|
+ endMessage={
|
|
|
+ <p className="text-center text-gray-500">没有更多记录了</p>
|
|
|
+ }
|
|
|
+ scrollableTarget="exchangeScroll"
|
|
|
+ >
|
|
|
+ {exchanges.map((exchange, index) => (
|
|
|
+ <div key={exchange._id}>
|
|
|
+ <div className="flex items-start py-3">
|
|
|
+ <div className="flex-grow mr-4">
|
|
|
+ <p className="text-sm text-gray-500 mb-1">
|
|
|
+ {new Date(exchange.exchangeTime).toLocaleDateString()}{" "}
|
|
|
+ {new Date(exchange.exchangeTime).toLocaleTimeString(
|
|
|
+ [],
|
|
|
+ { hour: "2-digit", minute: "2-digit" }
|
|
|
+ )}
|
|
|
+ </p>
|
|
|
+ <p
|
|
|
+ className="font-medium text-gray-800 mb-1"
|
|
|
+ dangerouslySetInnerHTML={{
|
|
|
+ __html: exchange.item.title,
|
|
|
+ }}
|
|
|
+ ></p>
|
|
|
+ <p
|
|
|
+ className={`text-sm font-medium ${
|
|
|
+ exchange.status === "待兑换已审核"
|
|
|
+ ? "text-blue-600"
|
|
|
+ : exchange.status === "待兑换未审核"
|
|
|
+ ? "text-gray-600"
|
|
|
+ : exchange.status === "已兑换"
|
|
|
+ ? "text-green-600"
|
|
|
+ : "text-gray-600"
|
|
|
+ }`}
|
|
|
+ >
|
|
|
+ {exchange.status}
|
|
|
+ </p>
|
|
|
+ </div>
|
|
|
+ <div className="flex-shrink-0 font-bold text-red-600 self-center">
|
|
|
+ -{exchange.item.points}
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ {index < exchanges.length - 1 && (
|
|
|
+ <div className="border-b border-gray-200"></div>
|
|
|
+ )}
|
|
|
+ </div>
|
|
|
+ ))}
|
|
|
+ </InfiniteScroll>
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+
|
|
|
+ {selectedItem && (
|
|
|
+ <ExchangeForm
|
|
|
+ item={selectedItem}
|
|
|
+ setAlert={setAlert}
|
|
|
+ onClose={() => setSelectedItem(null)}
|
|
|
+ />
|
|
|
+ )}
|
|
|
+ {alert && (
|
|
|
+ <Alert
|
|
|
+ message={alert.message}
|
|
|
+ type={alert.type}
|
|
|
+ onClose={() => setAlert(null)}
|
|
|
+ />
|
|
|
+ )}
|
|
|
+ </div>
|
|
|
+ </div>
|
|
|
+ );
|
|
|
+}
|