page.js 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. "use client";
  2. import Alert from "../ui/components/Alert";
  3. import { fetchApi } from "../utils/fetch";
  4. import ExchangeForm from "./ExchangeForm";
  5. import { ChevronLeft, X } from "lucide-react";
  6. import { useRouter } from "next/navigation";
  7. import { useState, useEffect } from "react";
  8. import InfiniteScroll from "react-infinite-scroll-component";
  9. const PAGE_SIZE = 10;
  10. export default function ExchangePage() {
  11. const [items, setItems] = useState([]);
  12. const [selectedItem, setSelectedItem] = useState(null);
  13. const [alert, setAlert] = useState(null);
  14. const [exchanges, setExchanges] = useState([]);
  15. const [page, setPage] = useState(1);
  16. const [hasMore, setHasMore] = useState(true);
  17. const [currentUser, setCurrentUser] = useState(null);
  18. const [isRulesOpen, setIsRulesOpen] = useState(false);
  19. const router = useRouter();
  20. useEffect(() => {
  21. fetchExchangeItems();
  22. const user = JSON.parse(localStorage.getItem("currentUser") || "null");
  23. setCurrentUser(user);
  24. }, []);
  25. useEffect(() => {
  26. loadExchanges();
  27. }, [currentUser?.id]);
  28. const loadExchanges = async () => {
  29. if (!currentUser) return;
  30. try {
  31. const data = await fetchApi(
  32. `/api/exchange-history?userId=${encodeURIComponent(
  33. currentUser.id
  34. )}&current=${page}&pageSize=${PAGE_SIZE}`
  35. );
  36. if (data.success) {
  37. setExchanges((prev) => [...prev, ...data.data]);
  38. setPage((prev) => prev + 1);
  39. setHasMore(data.data.length === PAGE_SIZE);
  40. } else {
  41. setAlert({ type: "error", message: data.error });
  42. console.error("Failed to fetch exchange history:", data.error);
  43. }
  44. } catch (error) {
  45. console.error("Error fetching exchange history:", error);
  46. }
  47. };
  48. const fetchExchangeItems = async () => {
  49. try {
  50. const data = await fetchApi("/api/exchange-items");
  51. if (data.success) {
  52. setItems(data.data);
  53. } else {
  54. setAlert({ type: "error", message: data.error });
  55. console.error("获取兑换项目失败:", data.message);
  56. }
  57. } catch (error) {
  58. console.error("获取兑换项目失败:", error);
  59. }
  60. };
  61. const handleExchange = (item) => {
  62. if (currentUser.points < item.points) {
  63. setAlert({ type: "error", message: "积分不足,无法兑换" });
  64. return;
  65. }
  66. setSelectedItem(item);
  67. };
  68. const Modal = ({ isOpen, onClose, children }) => {
  69. if (!isOpen) return null;
  70. return (
  71. <div className="fixed inset-0 z-50 flex items-center justify-center">
  72. <div
  73. className="absolute inset-0 bg-black/50 backdrop-blur-sm"
  74. onClick={onClose}
  75. />
  76. <div className="relative bg-white rounded-lg w-full max-w-2xl m-4 p-6 shadow-xl max-h-[70vh] overflow-y-auto">
  77. <button
  78. onClick={onClose}
  79. className="absolute right-4 top-4 text-gray-500 hover:text-gray-700"
  80. >
  81. <X size={24} />
  82. </button>
  83. {children}
  84. </div>
  85. </div>
  86. );
  87. };
  88. const PointsRules = () => {
  89. return (
  90. <div className="space-y-4 text-gray-700">
  91. <h2 className="text-xl font-bold text-gray-900">积分兑换规则</h2>
  92. <section>
  93. <h3 className="font-medium text-lg text-gray-900 mb-3">
  94. 1. 积分兑换规则
  95. </h3>
  96. <div className="space-y-2 pl-4">
  97. <p className="font-medium">兑换资格:</p>
  98. <p>用户必须拥有足够积分才能进行兑换。</p>
  99. <p className="font-medium mt-3">商品种类:</p>
  100. <p>积分可用于兑换以下几类商品:</p>
  101. <ul className="list-disc pl-6 space-y-1">
  102. <li>实物商品:如电子产品、生活用品等</li>
  103. <li>虚拟商品:如优惠券、抵用券、彩金等</li>
  104. </ul>
  105. <p className="text-gray-500 italic text-sm">
  106. (积分兑换商品会不定期更新,以显示为准)
  107. </p>
  108. </div>
  109. </section>
  110. <section>
  111. <h3 className="font-medium text-lg text-gray-900 mb-3">
  112. 2. 兑换流程
  113. </h3>
  114. <ol className="list-decimal pl-6 space-y-2">
  115. <li>点击"兑换积分"按钮</li>
  116. <li>
  117. 按需兑换商品,输入智博会员账号(输入错误会影响奖品发放)点击"确认兑换"
  118. </li>
  119. <li>兑换完成后需要等待审核,审核完成后会在3个工作日内自动派发</li>
  120. <li>实物商品类需要填写姓名,联系方式,地址,预计7个工作日到件</li>
  121. </ol>
  122. </section>
  123. <section>
  124. <h3 className="font-medium text-lg text-gray-900 mb-3">
  125. 3. 兑换商品退换规则
  126. </h3>
  127. <ul className="list-disc pl-6 space-y-2">
  128. <li>实物商品,一旦兑换成功,概不退换</li>
  129. <li>虚拟商品(如优惠券、抵用券),一经兑换,不可折现</li>
  130. </ul>
  131. </section>
  132. <section>
  133. <h3 className="font-medium text-lg text-gray-900 mb-3">4. 解释权</h3>
  134. <p className="pl-4">平台对积分获取、使用及兑换规则保留最终解释权。</p>
  135. </section>
  136. <section>
  137. <h3 className="font-medium text-lg text-gray-900 mb-3">
  138. 5. 常见问题
  139. </h3>
  140. <div className="space-y-4 pl-4">
  141. <div>
  142. <p className="font-medium">是否可以兑换多个商品?</p>
  143. <p className="text-gray-600">
  144. 可以,但需满足商品兑换所需积分条件。
  145. </p>
  146. </div>
  147. <div>
  148. <p className="font-medium">积分是否可以转让给他人?</p>
  149. <p className="text-gray-600">积分不可转让或赠送。</p>
  150. </div>
  151. <div>
  152. <p className="font-medium">积分有效期是多久?</p>
  153. <p className="text-gray-600">积分有效期为1年,逾期作废。</p>
  154. </div>
  155. </div>
  156. </section>
  157. <p className="text-sm text-gray-500 italic mt-6">
  158. 以上规则确保会员能够公平参与积分兑换活动,并提供了清晰的指引,保证平台和用户的权益。
  159. </p>
  160. </div>
  161. );
  162. };
  163. return (
  164. <div className="bg-blue-600 text-white min-h-screen">
  165. <div className="max-w-4xl mx-auto p-4">
  166. <div className="flex items-center mb-4">
  167. <ChevronLeft
  168. className="cursor-pointer hover:text-blue-200 transition-colors"
  169. onClick={() => router.back()}
  170. size={28}
  171. />
  172. <h1 className="text-xl font-bold ml-4">积分兑换</h1>
  173. <button
  174. onClick={() => setIsRulesOpen(true)}
  175. className="ml-auto px-4 py-2 font-bold bg-blue-500 hover:bg-blue-400 rounded-lg transition-colors"
  176. >
  177. 兑换规则
  178. </button>
  179. </div>
  180. <div className="grid gap-4 sm:grid-cols-2">
  181. {items.map((item) => (
  182. <div
  183. key={item._id}
  184. className="bg-white text-black rounded-lg p-4 shadow-md hover:shadow-lg transition-shadow"
  185. >
  186. <div className="flex items-center justify-between">
  187. <div className="flex items-center flex-grow">
  188. <img
  189. src={item.logo}
  190. // alt={item.title}
  191. width={50}
  192. height={50}
  193. className="mr-4 flex-shrink-0"
  194. />
  195. <div>
  196. <h2
  197. className="text-lg font-bold"
  198. dangerouslySetInnerHTML={{ __html: item.title }}
  199. ></h2>
  200. <p className="text-gray-600 text-sm">
  201. 所需积分: {item.points}
  202. </p>
  203. </div>
  204. </div>
  205. <button
  206. onClick={() => handleExchange(item)}
  207. 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"
  208. >
  209. 兑换
  210. </button>
  211. </div>
  212. </div>
  213. ))}
  214. </div>
  215. {/* 积分预测记录 */}
  216. <div className="bg-white text-black rounded-lg p-4 mb-4 mt-6">
  217. <h3 className="text-lg font-bold mb-2 text-blue-700">积分兑换记录</h3>
  218. <div
  219. id="exchangeScroll"
  220. style={{ height: "280px", overflow: "auto" }}
  221. >
  222. <InfiniteScroll
  223. dataLength={exchanges.length}
  224. next={loadExchanges}
  225. hasMore={hasMore}
  226. loader={<h4>Loading...</h4>}
  227. endMessage={
  228. <p className="text-center text-gray-500">没有更多记录了</p>
  229. }
  230. scrollableTarget="exchangeScroll"
  231. >
  232. {exchanges.map((exchange, index) => (
  233. <div key={exchange._id}>
  234. <div className="flex items-start py-3">
  235. <div className="flex-grow mr-4">
  236. <p className="text-sm text-gray-500 mb-1">
  237. {new Date(exchange.exchangeTime).toLocaleDateString()}{" "}
  238. {new Date(exchange.exchangeTime).toLocaleTimeString(
  239. [],
  240. { hour: "2-digit", minute: "2-digit" }
  241. )}
  242. </p>
  243. <p
  244. className="font-medium text-gray-800 mb-1"
  245. dangerouslySetInnerHTML={{
  246. __html: exchange.item.title,
  247. }}
  248. ></p>
  249. <p
  250. className={`text-sm font-medium ${exchange.status === "待兑换已审核"
  251. ? "text-blue-600"
  252. : exchange.status === "待兑换未审核"
  253. ? "text-gray-600"
  254. : exchange.status === "已兑换"
  255. ? "text-green-600"
  256. : "text-gray-600"
  257. }`}
  258. >
  259. {exchange.status}
  260. </p>
  261. </div>
  262. <div className="flex-shrink-0 font-bold text-red-600 self-center">
  263. -{exchange.item.points}
  264. </div>
  265. </div>
  266. {index < exchanges.length - 1 && (
  267. <div className="border-b border-gray-200"></div>
  268. )}
  269. </div>
  270. ))}
  271. </InfiniteScroll>
  272. </div>
  273. </div>
  274. {/* 积分规则 */}
  275. <Modal isOpen={isRulesOpen} onClose={() => setIsRulesOpen(false)}>
  276. <PointsRules />
  277. </Modal>
  278. {selectedItem && (
  279. <ExchangeForm
  280. currentUser={currentUser}
  281. setCurrentUser={setCurrentUser}
  282. item={selectedItem}
  283. setAlert={setAlert}
  284. onClose={() => setSelectedItem(null)}
  285. />
  286. )}
  287. {alert && (
  288. <Alert
  289. message={alert.message}
  290. type={alert.type}
  291. onClose={() => setAlert(null)}
  292. />
  293. )}
  294. </div>
  295. </div>
  296. );
  297. }