sa2kit 2.0.1 → 2.0.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (278) hide show
  1. package/README.md +1 -1
  2. package/dist/CollisionBalls-BpHufX3H.d.mts +41 -0
  3. package/dist/CollisionBalls-BpHufX3H.d.ts +41 -0
  4. package/dist/ConfigService-BxK06xP6.d.mts +262 -0
  5. package/dist/ConfigService-BxK06xP6.d.ts +262 -0
  6. package/dist/UniversalFileService-BpvbZitV.d.mts +139 -0
  7. package/dist/UniversalFileService-GsP6D3Rc.d.ts +139 -0
  8. package/dist/audioDetection/index.d.mts +449 -0
  9. package/dist/audioDetection/index.d.ts +449 -0
  10. package/dist/audioDetection/index.js +1244 -0
  11. package/dist/audioDetection/index.js.map +1 -0
  12. package/dist/audioDetection/index.mjs +1227 -0
  13. package/dist/audioDetection/index.mjs.map +1 -0
  14. package/dist/auth/legacy/core/index.d.mts +42 -0
  15. package/dist/auth/legacy/core/index.d.ts +42 -0
  16. package/dist/auth/legacy/core/index.js +242 -0
  17. package/dist/auth/legacy/core/index.js.map +1 -0
  18. package/dist/auth/legacy/core/index.mjs +226 -0
  19. package/dist/auth/legacy/core/index.mjs.map +1 -0
  20. package/dist/auth/legacy/db/index.d.mts +5 -0
  21. package/dist/auth/legacy/db/index.d.ts +5 -0
  22. package/dist/auth/legacy/db/index.js +261 -0
  23. package/dist/auth/legacy/db/index.js.map +1 -0
  24. package/dist/auth/legacy/db/index.mjs +250 -0
  25. package/dist/auth/legacy/db/index.mjs.map +1 -0
  26. package/dist/auth/legacy/index.d.mts +5 -0
  27. package/dist/auth/legacy/index.d.ts +5 -0
  28. package/dist/auth/legacy/index.js +1107 -0
  29. package/dist/auth/legacy/index.js.map +1 -0
  30. package/dist/auth/legacy/index.mjs +1086 -0
  31. package/dist/auth/legacy/index.mjs.map +1 -0
  32. package/dist/auth/legacy/logic/index.d.mts +9 -0
  33. package/dist/auth/legacy/logic/index.d.ts +9 -0
  34. package/dist/auth/legacy/logic/index.js +194 -0
  35. package/dist/auth/legacy/logic/index.js.map +1 -0
  36. package/dist/auth/legacy/logic/index.mjs +187 -0
  37. package/dist/auth/legacy/logic/index.mjs.map +1 -0
  38. package/dist/auth/legacy/miniapp/index.d.mts +5 -0
  39. package/dist/auth/legacy/miniapp/index.d.ts +5 -0
  40. package/dist/auth/legacy/miniapp/index.js +506 -0
  41. package/dist/auth/legacy/miniapp/index.js.map +1 -0
  42. package/dist/auth/legacy/miniapp/index.mjs +487 -0
  43. package/dist/auth/legacy/miniapp/index.mjs.map +1 -0
  44. package/dist/auth/legacy/routes/index.d.mts +53 -0
  45. package/dist/auth/legacy/routes/index.d.ts +53 -0
  46. package/dist/auth/legacy/routes/index.js +278 -0
  47. package/dist/auth/legacy/routes/index.js.map +1 -0
  48. package/dist/auth/legacy/routes/index.mjs +271 -0
  49. package/dist/auth/legacy/routes/index.mjs.map +1 -0
  50. package/dist/auth/legacy/schema/index.d.mts +401 -0
  51. package/dist/auth/legacy/schema/index.d.ts +401 -0
  52. package/dist/auth/legacy/schema/index.js +50 -0
  53. package/dist/auth/legacy/schema/index.js.map +1 -0
  54. package/dist/auth/legacy/schema/index.mjs +44 -0
  55. package/dist/auth/legacy/schema/index.mjs.map +1 -0
  56. package/dist/auth/legacy/server/index.d.mts +13 -0
  57. package/dist/auth/legacy/server/index.d.ts +13 -0
  58. package/dist/auth/legacy/server/index.js +21 -0
  59. package/dist/auth/legacy/server/index.js.map +1 -0
  60. package/dist/auth/legacy/server/index.mjs +19 -0
  61. package/dist/auth/legacy/server/index.mjs.map +1 -0
  62. package/dist/auth/legacy/services/index.d.mts +40 -0
  63. package/dist/auth/legacy/services/index.d.ts +40 -0
  64. package/dist/auth/legacy/services/index.js +258 -0
  65. package/dist/auth/legacy/services/index.js.map +1 -0
  66. package/dist/auth/legacy/services/index.mjs +252 -0
  67. package/dist/auth/legacy/services/index.mjs.map +1 -0
  68. package/dist/auth/legacy/ui/miniapp/index.d.mts +10 -0
  69. package/dist/auth/legacy/ui/miniapp/index.d.ts +10 -0
  70. package/dist/auth/legacy/ui/miniapp/index.js +298 -0
  71. package/dist/auth/legacy/ui/miniapp/index.js.map +1 -0
  72. package/dist/auth/legacy/ui/miniapp/index.mjs +290 -0
  73. package/dist/auth/legacy/ui/miniapp/index.mjs.map +1 -0
  74. package/dist/auth/legacy/ui/web/index.d.mts +22 -0
  75. package/dist/auth/legacy/ui/web/index.d.ts +22 -0
  76. package/dist/auth/legacy/ui/web/index.js +899 -0
  77. package/dist/auth/legacy/ui/web/index.js.map +1 -0
  78. package/dist/auth/legacy/ui/web/index.mjs +889 -0
  79. package/dist/auth/legacy/ui/web/index.mjs.map +1 -0
  80. package/dist/auth/legacy/web/index.d.mts +5 -0
  81. package/dist/auth/legacy/web/index.d.ts +5 -0
  82. package/dist/auth/legacy/web/index.js +1107 -0
  83. package/dist/auth/legacy/web/index.js.map +1 -0
  84. package/dist/auth/legacy/web/index.mjs +1086 -0
  85. package/dist/auth/legacy/web/index.mjs.map +1 -0
  86. package/dist/auth/rn/index.d.mts +64 -0
  87. package/dist/auth/rn/index.d.ts +64 -0
  88. package/dist/auth/rn/index.js +765 -0
  89. package/dist/auth/rn/index.js.map +1 -0
  90. package/dist/auth/rn/index.mjs +754 -0
  91. package/dist/auth/rn/index.mjs.map +1 -0
  92. package/dist/base-api-client-ACKKt13v.d.mts +277 -0
  93. package/dist/base-api-client-ACKKt13v.d.ts +277 -0
  94. package/dist/boothVaultService-Cn4WPhjg.d.mts +83 -0
  95. package/dist/boothVaultService-Cn4WPhjg.d.ts +83 -0
  96. package/dist/business/index.d.mts +6 -0
  97. package/dist/business/index.d.ts +6 -0
  98. package/dist/business/index.js +1682 -0
  99. package/dist/business/index.js.map +1 -0
  100. package/dist/business/index.mjs +1675 -0
  101. package/dist/business/index.mjs.map +1 -0
  102. package/dist/calendar/index.d.mts +1325 -0
  103. package/dist/calendar/index.d.ts +1325 -0
  104. package/dist/calendar/index.js +5964 -0
  105. package/dist/calendar/index.js.map +1 -0
  106. package/dist/calendar/index.mjs +5878 -0
  107. package/dist/calendar/index.mjs.map +1 -0
  108. package/dist/calendar/routes/index.d.mts +118 -0
  109. package/dist/calendar/routes/index.d.ts +118 -0
  110. package/dist/calendar/routes/index.js +755 -0
  111. package/dist/calendar/routes/index.js.map +1 -0
  112. package/dist/calendar/routes/index.mjs +747 -0
  113. package/dist/calendar/routes/index.mjs.map +1 -0
  114. package/dist/components/index.d.mts +405 -0
  115. package/dist/components/index.d.ts +405 -0
  116. package/dist/components/index.js +2516 -0
  117. package/dist/components/index.js.map +1 -0
  118. package/dist/components/index.mjs +2396 -0
  119. package/dist/components/index.mjs.map +1 -0
  120. package/dist/drizzle-schema-BNhqj2AZ.d.mts +1114 -0
  121. package/dist/drizzle-schema-BNhqj2AZ.d.ts +1114 -0
  122. package/dist/festivalCard/index.d.mts +76 -0
  123. package/dist/festivalCard/index.d.ts +76 -0
  124. package/dist/festivalCard/index.js +1492 -0
  125. package/dist/festivalCard/index.js.map +1 -0
  126. package/dist/festivalCard/index.mjs +1475 -0
  127. package/dist/festivalCard/index.mjs.map +1 -0
  128. package/dist/festivalCard/routes/index.d.mts +42 -0
  129. package/dist/festivalCard/routes/index.d.ts +42 -0
  130. package/dist/festivalCard/routes/index.js +361 -0
  131. package/dist/festivalCard/routes/index.js.map +1 -0
  132. package/dist/festivalCard/routes/index.mjs +356 -0
  133. package/dist/festivalCard/routes/index.mjs.map +1 -0
  134. package/dist/festivalCard/server/index.d.mts +120 -0
  135. package/dist/festivalCard/server/index.d.ts +120 -0
  136. package/dist/festivalCard/server/index.js +272 -0
  137. package/dist/festivalCard/server/index.js.map +1 -0
  138. package/dist/festivalCard/server/index.mjs +265 -0
  139. package/dist/festivalCard/server/index.mjs.map +1 -0
  140. package/dist/festivalCardService-BFCRhJrq.d.ts +13 -0
  141. package/dist/festivalCardService-GriR2VMc.d.mts +13 -0
  142. package/dist/index-1Ag7IBXN.d.ts +144 -0
  143. package/dist/index-DNKZ7-R_.d.mts +184 -0
  144. package/dist/index-DNKZ7-R_.d.ts +184 -0
  145. package/dist/index-DSel44Ke.d.mts +93 -0
  146. package/dist/index-DSel44Ke.d.ts +93 -0
  147. package/dist/index-DdeZSeTJ.d.mts +144 -0
  148. package/dist/index-DrPcMJPc.d.mts +250 -0
  149. package/dist/index-DrPcMJPc.d.ts +250 -0
  150. package/dist/index.d.mts +5334 -0
  151. package/dist/index.d.ts +5334 -0
  152. package/dist/index.js +18809 -0
  153. package/dist/index.js.map +1 -0
  154. package/dist/index.mjs +18533 -0
  155. package/dist/index.mjs.map +1 -0
  156. package/dist/mikuContest/ui/web/index.d.mts +2 -0
  157. package/dist/mikuContest/ui/web/index.d.ts +2 -0
  158. package/dist/mikuContest/ui/web/index.js +353 -0
  159. package/dist/mikuContest/ui/web/index.js.map +1 -0
  160. package/dist/mikuContest/ui/web/index.mjs +343 -0
  161. package/dist/mikuContest/ui/web/index.mjs.map +1 -0
  162. package/dist/mikuFireworks3D/index.d.mts +268 -0
  163. package/dist/mikuFireworks3D/index.d.ts +268 -0
  164. package/dist/mikuFireworks3D/index.js +1267 -0
  165. package/dist/mikuFireworks3D/index.js.map +1 -0
  166. package/dist/mikuFireworks3D/index.mjs +1228 -0
  167. package/dist/mikuFireworks3D/index.mjs.map +1 -0
  168. package/dist/mikuFusionGame/index.d.mts +117 -0
  169. package/dist/mikuFusionGame/index.d.ts +117 -0
  170. package/dist/mikuFusionGame/index.js +1208 -0
  171. package/dist/mikuFusionGame/index.js.map +1 -0
  172. package/dist/mikuFusionGame/index.mjs +1195 -0
  173. package/dist/mikuFusionGame/index.mjs.map +1 -0
  174. package/dist/mmd/admin/index.d.mts +487 -0
  175. package/dist/mmd/admin/index.d.ts +487 -0
  176. package/dist/mmd/admin/index.js +1058 -0
  177. package/dist/mmd/admin/index.js.map +1 -0
  178. package/dist/mmd/admin/index.mjs +1027 -0
  179. package/dist/mmd/admin/index.mjs.map +1 -0
  180. package/dist/mmd/index.d.mts +2467 -0
  181. package/dist/mmd/index.d.ts +2467 -0
  182. package/dist/mmd/index.js +10119 -0
  183. package/dist/mmd/index.js.map +1 -0
  184. package/dist/mmd/index.mjs +10028 -0
  185. package/dist/mmd/index.mjs.map +1 -0
  186. package/dist/mmd/server/index.d.mts +139 -0
  187. package/dist/mmd/server/index.d.ts +139 -0
  188. package/dist/mmd/server/index.js +424 -0
  189. package/dist/mmd/server/index.js.map +1 -0
  190. package/dist/mmd/server/index.mjs +404 -0
  191. package/dist/mmd/server/index.mjs.map +1 -0
  192. package/dist/music/index.d.mts +74 -0
  193. package/dist/music/index.d.ts +74 -0
  194. package/dist/music/index.js +830 -0
  195. package/dist/music/index.js.map +1 -0
  196. package/dist/music/index.mjs +809 -0
  197. package/dist/music/index.mjs.map +1 -0
  198. package/dist/music/server/index.d.mts +1 -0
  199. package/dist/music/server/index.d.ts +1 -0
  200. package/dist/music/server/index.js +194 -0
  201. package/dist/music/server/index.js.map +1 -0
  202. package/dist/music/server/index.mjs +182 -0
  203. package/dist/music/server/index.mjs.map +1 -0
  204. package/dist/navigation/index.d.mts +93 -0
  205. package/dist/navigation/index.d.ts +93 -0
  206. package/dist/navigation/index.js +453 -0
  207. package/dist/navigation/index.js.map +1 -0
  208. package/dist/navigation/index.mjs +443 -0
  209. package/dist/navigation/index.mjs.map +1 -0
  210. package/dist/portfolio/index.d.mts +66 -0
  211. package/dist/portfolio/index.d.ts +66 -0
  212. package/dist/portfolio/index.js +736 -0
  213. package/dist/portfolio/index.js.map +1 -0
  214. package/dist/portfolio/index.mjs +724 -0
  215. package/dist/portfolio/index.mjs.map +1 -0
  216. package/dist/qqbot/server/index.d.mts +216 -0
  217. package/dist/qqbot/server/index.d.ts +216 -0
  218. package/dist/qqbot/server/index.js +394 -0
  219. package/dist/qqbot/server/index.js.map +1 -0
  220. package/dist/qqbot/server/index.mjs +385 -0
  221. package/dist/qqbot/server/index.mjs.map +1 -0
  222. package/dist/qqbot/ui/web/index.d.mts +10 -0
  223. package/dist/qqbot/ui/web/index.d.ts +10 -0
  224. package/dist/qqbot/ui/web/index.js +105 -0
  225. package/dist/qqbot/ui/web/index.js.map +1 -0
  226. package/dist/qqbot/ui/web/index.mjs +99 -0
  227. package/dist/qqbot/ui/web/index.mjs.map +1 -0
  228. package/dist/screenReceiver/index.d.mts +86 -0
  229. package/dist/screenReceiver/index.d.ts +86 -0
  230. package/dist/screenReceiver/index.js +281 -0
  231. package/dist/screenReceiver/index.js.map +1 -0
  232. package/dist/screenReceiver/index.mjs +273 -0
  233. package/dist/screenReceiver/index.mjs.map +1 -0
  234. package/dist/testYourself/admin/index.d.mts +58 -0
  235. package/dist/testYourself/admin/index.d.ts +58 -0
  236. package/dist/testYourself/admin/index.js +1009 -0
  237. package/dist/testYourself/admin/index.js.map +1 -0
  238. package/dist/testYourself/admin/index.mjs +1002 -0
  239. package/dist/testYourself/admin/index.mjs.map +1 -0
  240. package/dist/testYourself/index.d.mts +53 -0
  241. package/dist/testYourself/index.d.ts +53 -0
  242. package/dist/testYourself/index.js +2551 -0
  243. package/dist/testYourself/index.js.map +1 -0
  244. package/dist/testYourself/index.mjs +2531 -0
  245. package/dist/testYourself/index.mjs.map +1 -0
  246. package/dist/testYourself/server/index.d.mts +1029 -0
  247. package/dist/testYourself/server/index.d.ts +1029 -0
  248. package/dist/testYourself/server/index.js +825 -0
  249. package/dist/testYourself/server/index.js.map +1 -0
  250. package/dist/testYourself/server/index.mjs +816 -0
  251. package/dist/testYourself/server/index.mjs.map +1 -0
  252. package/dist/types-BTiaMsBz.d.mts +292 -0
  253. package/dist/types-DyG3ZV9V.d.mts +270 -0
  254. package/dist/types-DyG3ZV9V.d.ts +270 -0
  255. package/dist/types-ERmJyjx8.d.ts +292 -0
  256. package/dist/types-HorDyIRv.d.mts +303 -0
  257. package/dist/types-HorDyIRv.d.ts +303 -0
  258. package/dist/types-tQfupO6d.d.mts +70 -0
  259. package/dist/types-tQfupO6d.d.ts +70 -0
  260. package/dist/vocaloidBooth/index.d.mts +64 -0
  261. package/dist/vocaloidBooth/index.d.ts +64 -0
  262. package/dist/vocaloidBooth/index.js +376 -0
  263. package/dist/vocaloidBooth/index.js.map +1 -0
  264. package/dist/vocaloidBooth/index.mjs +362 -0
  265. package/dist/vocaloidBooth/index.mjs.map +1 -0
  266. package/dist/vocaloidBooth/server/index.d.mts +111 -0
  267. package/dist/vocaloidBooth/server/index.d.ts +111 -0
  268. package/dist/vocaloidBooth/server/index.js +247 -0
  269. package/dist/vocaloidBooth/server/index.js.map +1 -0
  270. package/dist/vocaloidBooth/server/index.mjs +237 -0
  271. package/dist/vocaloidBooth/server/index.mjs.map +1 -0
  272. package/dist/vocaloidBooth/web/index.d.mts +3 -0
  273. package/dist/vocaloidBooth/web/index.d.ts +3 -0
  274. package/dist/vocaloidBooth/web/index.js +376 -0
  275. package/dist/vocaloidBooth/web/index.js.map +1 -0
  276. package/dist/vocaloidBooth/web/index.mjs +362 -0
  277. package/dist/vocaloidBooth/web/index.mjs.map +1 -0
  278. package/package.json +11 -1
@@ -0,0 +1,506 @@
1
+ 'use strict';
2
+
3
+ var React4 = require('react');
4
+ var components = require('@tarojs/components');
5
+
6
+ function _interopDefault (e) { return e && e.__esModule ? e : { default: e }; }
7
+
8
+ var React4__default = /*#__PURE__*/_interopDefault(React4);
9
+
10
+ // src/business/auth-legacy/ui/miniapp/components/LoginModal.tsx
11
+ var AuthContext = React4.createContext(void 0);
12
+ function extractUser(data) {
13
+ return data?.user ?? data?.data?.user ?? null;
14
+ }
15
+ function extractValid(data) {
16
+ if (typeof data?.valid === "boolean") return data.valid;
17
+ if (typeof data?.data?.valid === "boolean") return data.data.valid;
18
+ return false;
19
+ }
20
+ function AuthProvider({ children }) {
21
+ const [user, setUser] = React4.useState(null);
22
+ const [loading, setLoading] = React4.useState(true);
23
+ const [isAuthenticated, setIsAuthenticated] = React4.useState(false);
24
+ const isMountedRef = React4.useRef(true);
25
+ const safeSetState = React4.useCallback((updater) => {
26
+ if (isMountedRef.current) {
27
+ updater();
28
+ }
29
+ }, []);
30
+ const validateSession = React4.useCallback(async () => {
31
+ console.log("\u{1F50D} [AuthContext] \u5F00\u59CB\u9A8C\u8BC1\u4F1A\u8BDD...");
32
+ try {
33
+ const response = await fetch("/api/auth/validate");
34
+ console.log("\u{1F4E1} [AuthContext] \u4F1A\u8BDD\u9A8C\u8BC1\u54CD\u5E94\u72B6\u6001:", response.status);
35
+ const data = await response.json();
36
+ console.log("\u{1F4C4} [AuthContext] \u4F1A\u8BDD\u9A8C\u8BC1\u54CD\u5E94\u6570\u636E:", data);
37
+ const resolvedUser = extractUser(data);
38
+ const resolvedValid = extractValid(data);
39
+ safeSetState(() => {
40
+ if (resolvedValid && resolvedUser) {
41
+ console.log("\u2705 [AuthContext] \u4F1A\u8BDD\u9A8C\u8BC1\u6210\u529F, \u7528\u6237:", resolvedUser);
42
+ setUser(resolvedUser);
43
+ setIsAuthenticated(true);
44
+ } else {
45
+ console.log("\u274C [AuthContext] \u4F1A\u8BDD\u9A8C\u8BC1\u5931\u8D25:", data.message);
46
+ setUser(null);
47
+ setIsAuthenticated(false);
48
+ }
49
+ setLoading(false);
50
+ });
51
+ } catch (error) {
52
+ console.error("\u{1F4A5} [AuthContext] \u4F1A\u8BDD\u9A8C\u8BC1\u5F02\u5E38:", error);
53
+ safeSetState(() => {
54
+ setUser(null);
55
+ setIsAuthenticated(false);
56
+ setLoading(false);
57
+ });
58
+ }
59
+ }, [safeSetState]);
60
+ const login = React4.useCallback(async (credentials) => {
61
+ console.log("\u{1F511} [AuthContext] \u5F00\u59CB\u767B\u5F55...");
62
+ console.log("\u{1F4DD} [AuthContext] \u767B\u5F55\u51ED\u636E:", { phone: credentials.phone, password: "***" });
63
+ try {
64
+ console.log("\u{1F4E4} [AuthContext] \u53D1\u9001\u767B\u5F55\u8BF7\u6C42\u5230 /api/auth/login");
65
+ const response = await fetch("/api/auth/login", {
66
+ method: "POST",
67
+ headers: { "Content-Type": "application/json" },
68
+ body: JSON.stringify(credentials)
69
+ });
70
+ console.log("\u{1F4E1} [AuthContext] \u6536\u5230\u54CD\u5E94\uFF0C\u72B6\u6001\u7801:", response.status);
71
+ const data = await response.json();
72
+ console.log("\u{1F4C4} [AuthContext] \u54CD\u5E94\u6570\u636E:", data);
73
+ const resolvedUser = extractUser(data);
74
+ if (data.success && resolvedUser) {
75
+ console.log("\u2705 [AuthContext] \u767B\u5F55\u6210\u529F, \u5F00\u59CB\u66F4\u65B0\u5168\u5C40\u72B6\u6001");
76
+ console.log("\u{1F464} [AuthContext] \u7528\u6237\u6570\u636E:", resolvedUser);
77
+ console.log("\u{1F4CA} [AuthContext] \u66F4\u65B0\u524D\u72B6\u6001:", {
78
+ currentUser: user ? `${user.name || "\u672A\u8BBE\u7F6E"} (${user.phone})` : null,
79
+ currentIsAuthenticated: isAuthenticated,
80
+ currentLoading: loading
81
+ });
82
+ console.log("\u{1F504} [AuthContext] \u6267\u884C\u5168\u5C40\u72B6\u6001\u66F4\u65B0...");
83
+ safeSetState(() => {
84
+ console.log("\u{1F504} [AuthContext] \u6B63\u5728\u8BBE\u7F6E\u7528\u6237:", resolvedUser);
85
+ setUser(resolvedUser);
86
+ console.log("\u{1F504} [AuthContext] \u6B63\u5728\u8BBE\u7F6E\u8BA4\u8BC1\u72B6\u6001: true");
87
+ setIsAuthenticated(true);
88
+ console.log("\u{1F504} [AuthContext] \u6B63\u5728\u8BBE\u7F6E\u52A0\u8F7D\u72B6\u6001: false");
89
+ setLoading(false);
90
+ console.log("\u2705 [AuthContext] \u5168\u5C40\u72B6\u6001\u66F4\u65B0\u5B8C\u6210");
91
+ });
92
+ setTimeout(() => {
93
+ console.log("\u{1F389} [AuthContext] \u5EF6\u8FDF\u786E\u8BA4 - \u5168\u5C40\u767B\u5F55\u72B6\u6001\u5E94\u8BE5\u5DF2\u66F4\u65B0:", {
94
+ user: resolvedUser,
95
+ isAuthenticated: true
96
+ });
97
+ }, 0);
98
+ console.log("\u{1F680} [AuthContext] \u8FD4\u56DE\u6210\u529F\u7ED3\u679C");
99
+ return { success: true, user: resolvedUser };
100
+ } else {
101
+ console.log("\u274C [AuthContext] \u767B\u5F55\u5931\u8D25:", data.message);
102
+ return { success: false, message: data.message };
103
+ }
104
+ } catch (error) {
105
+ console.error("\u{1F4A5} [AuthContext] \u767B\u5F55\u5F02\u5E38:", error);
106
+ return { success: false, message: "\u767B\u5F55\u5931\u8D25\uFF0C\u8BF7\u7A0D\u540E\u91CD\u8BD5" };
107
+ }
108
+ }, [safeSetState, user, isAuthenticated, loading]);
109
+ const register = React4.useCallback(async (userData) => {
110
+ console.log("\u{1F4DD} [AuthContext] \u5F00\u59CB\u6CE8\u518C...");
111
+ try {
112
+ const response = await fetch("/api/auth/register", {
113
+ method: "POST",
114
+ headers: { "Content-Type": "application/json" },
115
+ body: JSON.stringify(userData)
116
+ });
117
+ const data = await response.json();
118
+ console.log("\u{1F4E1} [AuthContext] \u6CE8\u518C\u54CD\u5E94:", data);
119
+ const resolvedUser = extractUser(data);
120
+ if (data.success && resolvedUser) {
121
+ console.log("\u2705 [AuthContext] \u6CE8\u518C\u6210\u529F, \u7ACB\u5373\u66F4\u65B0\u5168\u5C40\u72B6\u6001");
122
+ safeSetState(() => {
123
+ setUser(resolvedUser);
124
+ setIsAuthenticated(true);
125
+ setLoading(false);
126
+ });
127
+ console.log("\u{1F680} [AuthContext] \u8FD4\u56DE\u6CE8\u518C\u6210\u529F\u7ED3\u679C");
128
+ return { success: true, user: resolvedUser };
129
+ } else {
130
+ console.log("\u274C [AuthContext] \u6CE8\u518C\u5931\u8D25:", data.message);
131
+ return { success: false, message: data.message };
132
+ }
133
+ } catch (error) {
134
+ console.error("\u{1F4A5} [AuthContext] \u6CE8\u518C\u5F02\u5E38:", error);
135
+ return { success: false, message: "\u6CE8\u518C\u5931\u8D25\uFF0C\u8BF7\u7A0D\u540E\u91CD\u8BD5" };
136
+ }
137
+ }, [safeSetState]);
138
+ const logout = React4.useCallback(async () => {
139
+ console.log("\u{1F6AA} [AuthContext] \u5F00\u59CB\u767B\u51FA...");
140
+ try {
141
+ await fetch("/api/auth/logout", { method: "POST" });
142
+ safeSetState(() => {
143
+ setUser(null);
144
+ setIsAuthenticated(false);
145
+ });
146
+ console.log("\u2705 [AuthContext] \u767B\u51FA\u6210\u529F, \u5168\u5C40\u72B6\u6001\u5DF2\u6E05\u9664");
147
+ } catch (error) {
148
+ console.error("\u{1F4A5} [AuthContext] \u767B\u51FA\u5931\u8D25:", error);
149
+ }
150
+ }, [safeSetState]);
151
+ const refreshUser = React4.useCallback(() => {
152
+ console.log("\u{1F504} [AuthContext] \u5237\u65B0\u7528\u6237\u4FE1\u606F...");
153
+ setLoading(true);
154
+ validateSession();
155
+ }, [validateSession]);
156
+ React4.useEffect(() => {
157
+ isMountedRef.current = true;
158
+ return () => {
159
+ isMountedRef.current = false;
160
+ };
161
+ }, []);
162
+ React4.useEffect(() => {
163
+ console.log("\u{1F680} [AuthContext] \u521D\u59CB\u5316, \u5F00\u59CB\u9A8C\u8BC1\u4F1A\u8BDD");
164
+ validateSession();
165
+ }, [validateSession]);
166
+ React4.useEffect(() => {
167
+ console.log("\u{1F4CA} [AuthContext] \u5168\u5C40\u72B6\u6001\u53D8\u5316:", {
168
+ isAuthenticated,
169
+ user: user ? `${user.name || "\u672A\u8BBE\u7F6E"} (${user.phone})` : null,
170
+ loading
171
+ });
172
+ }, [isAuthenticated, user, loading]);
173
+ const value = {
174
+ user,
175
+ loading,
176
+ isAuthenticated,
177
+ login,
178
+ register,
179
+ logout,
180
+ refreshUser
181
+ };
182
+ return /* @__PURE__ */ React4__default.default.createElement(AuthContext.Provider, { value }, children);
183
+ }
184
+ function useAuth() {
185
+ const context = React4.useContext(AuthContext);
186
+ if (context === void 0) {
187
+ throw new Error("useAuth must be used within an AuthProvider");
188
+ }
189
+ return context;
190
+ }
191
+
192
+ // src/business/auth-legacy/utils/authUtils.ts
193
+ function validatePhoneNumber(phone) {
194
+ return /^1[3-9]\d{9}$/.test(phone);
195
+ }
196
+ function validatePassword(password) {
197
+ if (!password) {
198
+ return { valid: false, message: "\u5BC6\u7801\u4E0D\u80FD\u4E3A\u7A7A" };
199
+ }
200
+ if (password.length < 6) {
201
+ return { valid: false, message: "\u5BC6\u7801\u957F\u5EA6\u81F3\u5C116\u4F4D" };
202
+ }
203
+ return { valid: true };
204
+ }
205
+ function generateSessionToken() {
206
+ return Math.random().toString(36).substring(2) + Date.now().toString(36) + Math.random().toString(36).substring(2);
207
+ }
208
+ function isAdmin(user) {
209
+ return user?.role === "admin";
210
+ }
211
+ function isActiveUser(user) {
212
+ return user?.isActive === true;
213
+ }
214
+ function getUserDisplayName(user) {
215
+ return user.name || user.phone || "\u672A\u77E5\u7528\u6237";
216
+ }
217
+ function calculateSessionExpiry(days = 30) {
218
+ return new Date(Date.now() + days * 24 * 60 * 60 * 1e3);
219
+ }
220
+ function isSessionExpired(expiresAt) {
221
+ return /* @__PURE__ */ new Date() > new Date(expiresAt);
222
+ }
223
+
224
+ // src/business/auth-legacy/ui/miniapp/components/LoginModal.tsx
225
+ function LoginModal({ isOpen, onClose, onSuccess, onSwitchToRegister }) {
226
+ const { login } = useAuth();
227
+ const [formData, setFormData] = React4.useState({ phone: "", password: "" });
228
+ const [error, setError] = React4.useState("");
229
+ const [loading, setLoading] = React4.useState(false);
230
+ if (!isOpen) return null;
231
+ const handleSubmit = async () => {
232
+ setError("");
233
+ setLoading(true);
234
+ if (!formData.phone || !formData.password) {
235
+ setError("\u8BF7\u586B\u5199\u5B8C\u6574\u4FE1\u606F");
236
+ setLoading(false);
237
+ return;
238
+ }
239
+ if (!validatePhoneNumber(formData.phone)) {
240
+ setError("\u8BF7\u8F93\u5165\u6B63\u786E\u7684\u624B\u673A\u53F7");
241
+ setLoading(false);
242
+ return;
243
+ }
244
+ const result = await login(formData);
245
+ if (result.success) {
246
+ onSuccess();
247
+ } else {
248
+ setError(result.message || "\u767B\u5F55\u5931\u8D25");
249
+ }
250
+ setLoading(false);
251
+ };
252
+ return /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-modal" }, /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-card" }, /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-header" }, /* @__PURE__ */ React4__default.default.createElement(components.Text, { className: "auth-title" }, "\u7528\u6237\u767B\u5F55"), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-close", onClick: onClose }, "\u5173\u95ED")), /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-form" }, /* @__PURE__ */ React4__default.default.createElement(
253
+ components.Input,
254
+ {
255
+ name: "phone",
256
+ value: formData.phone,
257
+ onInput: (e) => setFormData((prev) => ({ ...prev, phone: e.detail.value })),
258
+ placeholder: "\u8BF7\u8F93\u5165\u624B\u673A\u53F7",
259
+ type: "text"
260
+ }
261
+ ), /* @__PURE__ */ React4__default.default.createElement(
262
+ components.Input,
263
+ {
264
+ name: "password",
265
+ value: formData.password,
266
+ onInput: (e) => setFormData((prev) => ({ ...prev, password: e.detail.value })),
267
+ placeholder: "\u8BF7\u8F93\u5165\u5BC6\u7801",
268
+ type: "password"
269
+ }
270
+ ), error && /* @__PURE__ */ React4__default.default.createElement(components.Text, { className: "auth-error" }, error), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-submit", loading, onClick: handleSubmit }, loading ? "\u767B\u5F55\u4E2D..." : "\u767B\u5F55"), onSwitchToRegister && /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-switch", onClick: onSwitchToRegister }, "\u53BB\u6CE8\u518C"))));
271
+ }
272
+ function RegisterModal({ isOpen, onClose, onSuccess, onSwitchToLogin }) {
273
+ const { register } = useAuth();
274
+ const [formData, setFormData] = React4.useState({
275
+ phone: "",
276
+ name: "",
277
+ password: "",
278
+ confirmPassword: ""
279
+ });
280
+ const [error, setError] = React4.useState("");
281
+ const [loading, setLoading] = React4.useState(false);
282
+ if (!isOpen) return null;
283
+ const handleSubmit = async () => {
284
+ setError("");
285
+ setLoading(true);
286
+ if (!formData.phone || !formData.password || !formData.confirmPassword) {
287
+ setError("\u8BF7\u586B\u5199\u5B8C\u6574\u4FE1\u606F");
288
+ setLoading(false);
289
+ return;
290
+ }
291
+ if (!validatePhoneNumber(formData.phone)) {
292
+ setError("\u8BF7\u8F93\u5165\u6B63\u786E\u7684\u624B\u673A\u53F7");
293
+ setLoading(false);
294
+ return;
295
+ }
296
+ const passwordValidation = validatePassword(formData.password);
297
+ if (!passwordValidation.valid) {
298
+ setError(passwordValidation.message || "\u5BC6\u7801\u683C\u5F0F\u9519\u8BEF");
299
+ setLoading(false);
300
+ return;
301
+ }
302
+ if (formData.password !== formData.confirmPassword) {
303
+ setError("\u4E24\u6B21\u8F93\u5165\u7684\u5BC6\u7801\u4E0D\u4E00\u81F4");
304
+ setLoading(false);
305
+ return;
306
+ }
307
+ const result = await register({
308
+ phone: formData.phone,
309
+ password: formData.password,
310
+ name: formData.name || void 0
311
+ });
312
+ if (result.success) {
313
+ onSuccess();
314
+ } else {
315
+ setError(result.message || "\u6CE8\u518C\u5931\u8D25");
316
+ }
317
+ setLoading(false);
318
+ };
319
+ return /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-modal" }, /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-card" }, /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-header" }, /* @__PURE__ */ React4__default.default.createElement(components.Text, { className: "auth-title" }, "\u7528\u6237\u6CE8\u518C"), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-close", onClick: onClose }, "\u5173\u95ED")), /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-form" }, /* @__PURE__ */ React4__default.default.createElement(
320
+ components.Input,
321
+ {
322
+ name: "phone",
323
+ value: formData.phone,
324
+ onInput: (e) => setFormData((prev) => ({ ...prev, phone: e.detail.value })),
325
+ placeholder: "\u8BF7\u8F93\u5165\u624B\u673A\u53F7",
326
+ type: "text"
327
+ }
328
+ ), /* @__PURE__ */ React4__default.default.createElement(
329
+ components.Input,
330
+ {
331
+ name: "name",
332
+ value: formData.name,
333
+ onInput: (e) => setFormData((prev) => ({ ...prev, name: e.detail.value })),
334
+ placeholder: "\u8BF7\u8F93\u5165\u6635\u79F0",
335
+ type: "text"
336
+ }
337
+ ), /* @__PURE__ */ React4__default.default.createElement(
338
+ components.Input,
339
+ {
340
+ name: "password",
341
+ value: formData.password,
342
+ onInput: (e) => setFormData((prev) => ({ ...prev, password: e.detail.value })),
343
+ placeholder: "\u8BF7\u8F93\u5165\u5BC6\u7801",
344
+ type: "password"
345
+ }
346
+ ), /* @__PURE__ */ React4__default.default.createElement(
347
+ components.Input,
348
+ {
349
+ name: "confirmPassword",
350
+ value: formData.confirmPassword,
351
+ onInput: (e) => setFormData((prev) => ({ ...prev, confirmPassword: e.detail.value })),
352
+ placeholder: "\u8BF7\u786E\u8BA4\u5BC6\u7801",
353
+ type: "password"
354
+ }
355
+ ), error && /* @__PURE__ */ React4__default.default.createElement(components.Text, { className: "auth-error" }, error), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-submit", loading, onClick: handleSubmit }, loading ? "\u6CE8\u518C\u4E2D..." : "\u6CE8\u518C"), onSwitchToLogin && /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-switch", onClick: onSwitchToLogin }, "\u53BB\u767B\u5F55"))));
356
+ }
357
+ function ForgotPasswordModal({ isOpen, onClose, onSuccess }) {
358
+ const [step, setStep] = React4.useState("phone");
359
+ const [formData, setFormData] = React4.useState({
360
+ phone: "",
361
+ verificationCode: "",
362
+ newPassword: "",
363
+ confirmPassword: ""
364
+ });
365
+ const [error, setError] = React4.useState("");
366
+ const [loading, setLoading] = React4.useState(false);
367
+ if (!isOpen) return null;
368
+ const handleSendCode = async () => {
369
+ setError("");
370
+ setLoading(true);
371
+ if (!validatePhoneNumber(formData.phone)) {
372
+ setError("\u8BF7\u8F93\u5165\u6B63\u786E\u7684\u624B\u673A\u53F7");
373
+ setLoading(false);
374
+ return;
375
+ }
376
+ try {
377
+ const response = await fetch("/api/auth/send-verification-code", {
378
+ method: "POST",
379
+ headers: { "Content-Type": "application/json" },
380
+ body: JSON.stringify({ phone: formData.phone })
381
+ });
382
+ const data = await response.json();
383
+ if (data.success) {
384
+ setStep("verify");
385
+ } else {
386
+ setError(data.message || "\u53D1\u9001\u5931\u8D25");
387
+ }
388
+ } catch (error2) {
389
+ console.error("\u{1F4A5} [ForgotPasswordModal] \u53D1\u9001\u9A8C\u8BC1\u7801\u5F02\u5E38:", error2);
390
+ setError("\u53D1\u9001\u5931\u8D25\uFF0C\u8BF7\u7A0D\u540E\u91CD\u8BD5");
391
+ } finally {
392
+ setLoading(false);
393
+ }
394
+ };
395
+ const handleVerifyCode = () => {
396
+ if (!formData.verificationCode) {
397
+ setError("\u8BF7\u8F93\u5165\u9A8C\u8BC1\u7801");
398
+ return;
399
+ }
400
+ setStep("reset");
401
+ };
402
+ const handleResetPassword = async () => {
403
+ const passwordValidation = validatePassword(formData.newPassword);
404
+ if (!passwordValidation.valid) {
405
+ setError(passwordValidation.message || "\u5BC6\u7801\u683C\u5F0F\u9519\u8BEF");
406
+ return;
407
+ }
408
+ if (formData.newPassword !== formData.confirmPassword) {
409
+ setError("\u4E24\u6B21\u8F93\u5165\u7684\u5BC6\u7801\u4E0D\u4E00\u81F4");
410
+ return;
411
+ }
412
+ setLoading(true);
413
+ try {
414
+ const response = await fetch("/api/auth/reset-password", {
415
+ method: "POST",
416
+ headers: { "Content-Type": "application/json" },
417
+ body: JSON.stringify({
418
+ phone: formData.phone,
419
+ verificationCode: formData.verificationCode,
420
+ newPassword: formData.newPassword
421
+ })
422
+ });
423
+ const data = await response.json();
424
+ if (data.success) {
425
+ onSuccess();
426
+ } else {
427
+ setError(data.message || "\u91CD\u7F6E\u5931\u8D25");
428
+ }
429
+ } catch (error2) {
430
+ console.error("\u{1F4A5} [ForgotPasswordModal] \u91CD\u7F6E\u5BC6\u7801\u5F02\u5E38:", error2);
431
+ setError("\u91CD\u7F6E\u5931\u8D25\uFF0C\u8BF7\u7A0D\u540E\u91CD\u8BD5");
432
+ } finally {
433
+ setLoading(false);
434
+ }
435
+ };
436
+ const renderContent = () => {
437
+ if (step === "phone") {
438
+ return /* @__PURE__ */ React4__default.default.createElement(React4__default.default.Fragment, null, /* @__PURE__ */ React4__default.default.createElement(
439
+ components.Input,
440
+ {
441
+ name: "phone",
442
+ value: formData.phone,
443
+ onInput: (e) => setFormData((prev) => ({ ...prev, phone: e.detail.value })),
444
+ placeholder: "\u8BF7\u8F93\u5165\u624B\u673A\u53F7",
445
+ type: "text"
446
+ }
447
+ ), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-submit", loading, onClick: handleSendCode }, loading ? "\u53D1\u9001\u4E2D..." : "\u53D1\u9001\u9A8C\u8BC1\u7801"));
448
+ }
449
+ if (step === "verify") {
450
+ return /* @__PURE__ */ React4__default.default.createElement(React4__default.default.Fragment, null, /* @__PURE__ */ React4__default.default.createElement(
451
+ components.Input,
452
+ {
453
+ name: "verificationCode",
454
+ value: formData.verificationCode,
455
+ onInput: (e) => setFormData((prev) => ({ ...prev, verificationCode: e.detail.value })),
456
+ placeholder: "\u8BF7\u8F93\u5165\u9A8C\u8BC1\u7801",
457
+ type: "text"
458
+ }
459
+ ), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-submit", onClick: handleVerifyCode }, "\u4E0B\u4E00\u6B65"));
460
+ }
461
+ return /* @__PURE__ */ React4__default.default.createElement(React4__default.default.Fragment, null, /* @__PURE__ */ React4__default.default.createElement(
462
+ components.Input,
463
+ {
464
+ name: "newPassword",
465
+ value: formData.newPassword,
466
+ onInput: (e) => setFormData((prev) => ({ ...prev, newPassword: e.detail.value })),
467
+ placeholder: "\u8BF7\u8F93\u5165\u65B0\u5BC6\u7801",
468
+ type: "password"
469
+ }
470
+ ), /* @__PURE__ */ React4__default.default.createElement(
471
+ components.Input,
472
+ {
473
+ name: "confirmPassword",
474
+ value: formData.confirmPassword,
475
+ onInput: (e) => setFormData((prev) => ({ ...prev, confirmPassword: e.detail.value })),
476
+ placeholder: "\u8BF7\u786E\u8BA4\u65B0\u5BC6\u7801",
477
+ type: "password"
478
+ }
479
+ ), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-submit", loading, onClick: handleResetPassword }, loading ? "\u63D0\u4EA4\u4E2D..." : "\u91CD\u7F6E\u5BC6\u7801"));
480
+ };
481
+ return /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-modal" }, /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-card" }, /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-header" }, /* @__PURE__ */ React4__default.default.createElement(components.Text, { className: "auth-title" }, "\u627E\u56DE\u5BC6\u7801"), /* @__PURE__ */ React4__default.default.createElement(components.Button, { className: "auth-close", onClick: onClose }, "\u5173\u95ED")), /* @__PURE__ */ React4__default.default.createElement(components.View, { className: "auth-form" }, renderContent(), error && /* @__PURE__ */ React4__default.default.createElement(components.Text, { className: "auth-error" }, error))));
482
+ }
483
+
484
+ // src/business/auth-legacy/types/index.ts
485
+ var UserRole = /* @__PURE__ */ ((UserRole2) => {
486
+ UserRole2["USER"] = "user";
487
+ UserRole2["ADMIN"] = "admin";
488
+ return UserRole2;
489
+ })(UserRole || {});
490
+
491
+ exports.AuthProvider = AuthProvider;
492
+ exports.ForgotPasswordModal = ForgotPasswordModal;
493
+ exports.LoginModal = LoginModal;
494
+ exports.RegisterModal = RegisterModal;
495
+ exports.UserRole = UserRole;
496
+ exports.calculateSessionExpiry = calculateSessionExpiry;
497
+ exports.generateSessionToken = generateSessionToken;
498
+ exports.getUserDisplayName = getUserDisplayName;
499
+ exports.isActiveUser = isActiveUser;
500
+ exports.isAdmin = isAdmin;
501
+ exports.isSessionExpired = isSessionExpired;
502
+ exports.useAuth = useAuth;
503
+ exports.validatePassword = validatePassword;
504
+ exports.validatePhoneNumber = validatePhoneNumber;
505
+ //# sourceMappingURL=index.js.map
506
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../../src/business/auth-legacy/contexts/AuthContext.tsx","../../../../src/business/auth-legacy/utils/authUtils.ts","../../../../src/business/auth-legacy/ui/miniapp/components/LoginModal.tsx","../../../../src/business/auth-legacy/ui/miniapp/components/RegisterModal.tsx","../../../../src/business/auth-legacy/ui/miniapp/components/ForgotPasswordModal.tsx","../../../../src/business/auth-legacy/types/index.ts"],"names":["createContext","useState","useRef","useCallback","useEffect","React","useContext","View","Text","Button","Input","error","UserRole"],"mappings":"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","file":"index.js","sourcesContent":["'use client';\n\nimport React, { createContext, useContext, useState, useEffect, useCallback, useRef } from 'react';\nimport type { User, UseAuthReturn, LoginRequest, RegisterRequest } from '../types';\n\ninterface AuthContextType extends UseAuthReturn {}\n\nconst AuthContext = createContext<AuthContextType | undefined>(undefined);\n\nfunction extractUser(data: any): User | null {\n return data?.user ?? data?.data?.user ?? null;\n}\n\nfunction extractValid(data: any): boolean {\n if (typeof data?.valid === 'boolean') return data.valid;\n if (typeof data?.data?.valid === 'boolean') return data.data.valid;\n return false;\n}\n\nexport function AuthProvider({ children }: { children: React.ReactNode }) {\n const [user, setUser] = useState<User | null>(null);\n const [loading, setLoading] = useState(true);\n const [isAuthenticated, setIsAuthenticated] = useState(false);\n const isMountedRef = useRef(true);\n\n // 安全的状态更新函数\n const safeSetState = useCallback((updater: () => void) => {\n if (isMountedRef.current) {\n updater();\n }\n }, []);\n\n // 验证会话\n const validateSession = useCallback(async () => {\n console.log('🔍 [AuthContext] 开始验证会话...');\n try {\n const response = await fetch('/api/auth/validate');\n console.log('📡 [AuthContext] 会话验证响应状态:', response.status);\n \n const data = await response.json();\n console.log('📄 [AuthContext] 会话验证响应数据:', data);\n const resolvedUser = extractUser(data);\n const resolvedValid = extractValid(data);\n \n safeSetState(() => {\n if (resolvedValid && resolvedUser) {\n console.log('✅ [AuthContext] 会话验证成功, 用户:', resolvedUser);\n setUser(resolvedUser);\n setIsAuthenticated(true);\n } else {\n console.log('❌ [AuthContext] 会话验证失败:', data.message);\n setUser(null);\n setIsAuthenticated(false);\n }\n setLoading(false);\n });\n } catch (error) {\n console.error('💥 [AuthContext] 会话验证异常:', error);\n safeSetState(() => {\n setUser(null);\n setIsAuthenticated(false);\n setLoading(false);\n });\n }\n }, [safeSetState]);\n\n // 登录\n const login = useCallback(async (credentials: LoginRequest) => {\n console.log('🔑 [AuthContext] 开始登录...');\n console.log('📝 [AuthContext] 登录凭据:', { phone: credentials.phone, password: '***' });\n \n try {\n console.log('📤 [AuthContext] 发送登录请求到 /api/auth/login');\n const response = await fetch('/api/auth/login', {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(credentials),\n });\n\n console.log('📡 [AuthContext] 收到响应,状态码:', response.status);\n const data = await response.json();\n console.log('📄 [AuthContext] 响应数据:', data);\n const resolvedUser = extractUser(data);\n\n if (data.success && resolvedUser) {\n console.log('✅ [AuthContext] 登录成功, 开始更新全局状态');\n console.log('👤 [AuthContext] 用户数据:', resolvedUser);\n \n // 记录当前状态\n console.log('📊 [AuthContext] 更新前状态:', {\n currentUser: user ? `${user.name || '未设置'} (${user.phone})` : null,\n currentIsAuthenticated: isAuthenticated,\n currentLoading: loading\n });\n \n // 使用同步的状态更新确保立即生效\n console.log('🔄 [AuthContext] 执行全局状态更新...');\n safeSetState(() => {\n console.log('🔄 [AuthContext] 正在设置用户:', resolvedUser);\n setUser(resolvedUser);\n console.log('🔄 [AuthContext] 正在设置认证状态: true');\n setIsAuthenticated(true);\n console.log('🔄 [AuthContext] 正在设置加载状态: false');\n setLoading(false);\n console.log('✅ [AuthContext] 全局状态更新完成');\n });\n \n // 等待下一个事件循环后输出确认日志\n setTimeout(() => {\n console.log('🎉 [AuthContext] 延迟确认 - 全局登录状态应该已更新:', {\n user: resolvedUser,\n isAuthenticated: true\n });\n }, 0);\n \n console.log('🚀 [AuthContext] 返回成功结果');\n return { success: true, user: resolvedUser };\n } else {\n console.log('❌ [AuthContext] 登录失败:', data.message);\n return { success: false, message: data.message };\n }\n } catch (error) {\n console.error('💥 [AuthContext] 登录异常:', error);\n return { success: false, message: '登录失败,请稍后重试' };\n }\n }, [safeSetState, user, isAuthenticated, loading]);\n\n // 注册\n const register = useCallback(async (userData: RegisterRequest) => {\n console.log('📝 [AuthContext] 开始注册...');\n try {\n const response = await fetch('/api/auth/register', {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify(userData),\n });\n\n const data = await response.json();\n console.log('📡 [AuthContext] 注册响应:', data);\n const resolvedUser = extractUser(data);\n\n if (data.success && resolvedUser) {\n console.log('✅ [AuthContext] 注册成功, 立即更新全局状态');\n \n // 使用同步的状态更新确保立即生效\n safeSetState(() => {\n setUser(resolvedUser);\n setIsAuthenticated(true);\n setLoading(false);\n });\n \n console.log('🚀 [AuthContext] 返回注册成功结果');\n return { success: true, user: resolvedUser };\n } else {\n console.log('❌ [AuthContext] 注册失败:', data.message);\n return { success: false, message: data.message };\n }\n } catch (error) {\n console.error('💥 [AuthContext] 注册异常:', error);\n return { success: false, message: '注册失败,请稍后重试' };\n }\n }, [safeSetState]);\n\n // 登出\n const logout = useCallback(async () => {\n console.log('🚪 [AuthContext] 开始登出...');\n try {\n await fetch('/api/auth/logout', { method: 'POST' });\n safeSetState(() => {\n setUser(null);\n setIsAuthenticated(false);\n });\n console.log('✅ [AuthContext] 登出成功, 全局状态已清除');\n } catch (error) {\n console.error('💥 [AuthContext] 登出失败:', error);\n }\n }, [safeSetState]);\n\n // 刷新用户信息\n const refreshUser = useCallback(() => {\n console.log('🔄 [AuthContext] 刷新用户信息...');\n setLoading(true);\n validateSession();\n }, [validateSession]);\n\n // 组件挂载/卸载管理\n useEffect(() => {\n isMountedRef.current = true;\n return () => {\n isMountedRef.current = false;\n };\n }, []);\n\n // 初始化时验证会话\n useEffect(() => {\n console.log('🚀 [AuthContext] 初始化, 开始验证会话');\n validateSession();\n }, [validateSession]);\n\n // 状态变化监控(用于调试)\n useEffect(() => {\n console.log('📊 [AuthContext] 全局状态变化:', {\n isAuthenticated,\n user: user ? `${user.name || '未设置'} (${user.phone})` : null,\n loading\n });\n }, [isAuthenticated, user, loading]);\n\n const value: AuthContextType = {\n user,\n loading,\n isAuthenticated,\n login,\n register,\n logout,\n refreshUser,\n };\n\n return (\n <AuthContext.Provider value={value}>\n {children}\n </AuthContext.Provider>\n );\n}\n\nexport function useAuth(): UseAuthReturn {\n const context = useContext(AuthContext);\n if (context === undefined) {\n throw new Error('useAuth must be used within an AuthProvider');\n }\n return context;\n} \n","import type { User } from '../types';\n\n/**\n * 验证手机号格式\n */\nexport function validatePhoneNumber(phone: string): boolean {\n return /^1[3-9]\\d{9}$/.test(phone);\n}\n\n/**\n * 验证密码强度(可扩展)\n */\nexport function validatePassword(password: string): { valid: boolean; message?: string } {\n if (!password) {\n return { valid: false, message: '密码不能为空' };\n }\n \n if (password.length < 6) {\n return { valid: false, message: '密码长度至少6位' };\n }\n \n return { valid: true };\n}\n\n/**\n * 生成安全的会话令牌\n */\nexport function generateSessionToken(): string {\n // 在实际环境中,应该使用更安全的随机数生成\n return Math.random().toString(36).substring(2) + \n Date.now().toString(36) + \n Math.random().toString(36).substring(2);\n}\n\n/**\n * 检查用户是否为管理员\n */\nexport function isAdmin(user: User | null): boolean {\n return user?.role === 'admin';\n}\n\n/**\n * 检查用户是否处于活跃状态\n */\nexport function isActiveUser(user: User | null): boolean {\n return user?.isActive === true;\n}\n\n/**\n * 格式化用户显示名称\n */\nexport function getUserDisplayName(user: User): string {\n return user.name || user.phone || '未知用户';\n}\n\n/**\n * 计算会话过期时间\n */\nexport function calculateSessionExpiry(days: number = 30): Date {\n return new Date(Date.now() + days * 24 * 60 * 60 * 1000);\n}\n\n/**\n * 检查会话是否过期\n */\nexport function isSessionExpired(expiresAt: Date): boolean {\n return new Date() > new Date(expiresAt);\n} ","'use client';\n\nimport React, { useState } from 'react';\nimport { View, Text, Input, Button } from '@tarojs/components';\nimport { useAuth } from '../../../contexts/AuthContext';\nimport { validatePhoneNumber } from '../../../utils/authUtils';\nimport type { LoginModalProps } from '../../../types';\n\ntype MiniappInputEvent = {\n detail: {\n value: string;\n };\n};\n\nexport default function LoginModal({ isOpen, onClose, onSuccess, onSwitchToRegister }: LoginModalProps) {\n const { login } = useAuth();\n const [formData, setFormData] = useState({ phone: '', password: '' });\n const [error, setError] = useState('');\n const [loading, setLoading] = useState(false);\n\n if (!isOpen) return null;\n\n const handleSubmit = async () => {\n setError('');\n setLoading(true);\n\n if (!formData.phone || !formData.password) {\n setError('请填写完整信息');\n setLoading(false);\n return;\n }\n\n if (!validatePhoneNumber(formData.phone)) {\n setError('请输入正确的手机号');\n setLoading(false);\n return;\n }\n\n const result = await login(formData);\n if (result.success) {\n onSuccess();\n } else {\n setError(result.message || '登录失败');\n }\n setLoading(false);\n };\n\n return (\n <View className=\"auth-modal\">\n <View className=\"auth-card\">\n <View className=\"auth-header\">\n <Text className=\"auth-title\">用户登录</Text>\n <Button className=\"auth-close\" onClick={onClose}>\n 关闭\n </Button>\n </View>\n <View className=\"auth-form\">\n <Input\n name=\"phone\"\n value={formData.phone}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, phone: e.detail.value }))}\n placeholder=\"请输入手机号\"\n type=\"text\"\n />\n <Input\n name=\"password\"\n value={formData.password}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, password: e.detail.value }))}\n placeholder=\"请输入密码\"\n type=\"password\"\n />\n {error && <Text className=\"auth-error\">{error}</Text>}\n <Button className=\"auth-submit\" loading={loading} onClick={handleSubmit}>\n {loading ? '登录中...' : '登录'}\n </Button>\n {onSwitchToRegister && (\n <Button className=\"auth-switch\" onClick={onSwitchToRegister}>\n 去注册\n </Button>\n )}\n </View>\n </View>\n </View>\n );\n}\n","'use client';\n\nimport React, { useState } from 'react';\nimport { View, Text, Input, Button } from '@tarojs/components';\nimport { useAuth } from '../../../contexts/AuthContext';\nimport { validatePhoneNumber, validatePassword } from '../../../utils/authUtils';\nimport type { RegisterModalProps } from '../../../types';\n\ntype MiniappInputEvent = {\n detail: {\n value: string;\n };\n};\n\nexport default function RegisterModal({ isOpen, onClose, onSuccess, onSwitchToLogin }: RegisterModalProps) {\n const { register } = useAuth();\n const [formData, setFormData] = useState({\n phone: '',\n name: '',\n password: '',\n confirmPassword: '',\n });\n const [error, setError] = useState('');\n const [loading, setLoading] = useState(false);\n\n if (!isOpen) return null;\n\n const handleSubmit = async () => {\n setError('');\n setLoading(true);\n\n if (!formData.phone || !formData.password || !formData.confirmPassword) {\n setError('请填写完整信息');\n setLoading(false);\n return;\n }\n\n if (!validatePhoneNumber(formData.phone)) {\n setError('请输入正确的手机号');\n setLoading(false);\n return;\n }\n\n const passwordValidation = validatePassword(formData.password);\n if (!passwordValidation.valid) {\n setError(passwordValidation.message || '密码格式错误');\n setLoading(false);\n return;\n }\n\n if (formData.password !== formData.confirmPassword) {\n setError('两次输入的密码不一致');\n setLoading(false);\n return;\n }\n\n const result = await register({\n phone: formData.phone,\n password: formData.password,\n name: formData.name || undefined,\n });\n\n if (result.success) {\n onSuccess();\n } else {\n setError(result.message || '注册失败');\n }\n\n setLoading(false);\n };\n\n return (\n <View className=\"auth-modal\">\n <View className=\"auth-card\">\n <View className=\"auth-header\">\n <Text className=\"auth-title\">用户注册</Text>\n <Button className=\"auth-close\" onClick={onClose}>\n 关闭\n </Button>\n </View>\n <View className=\"auth-form\">\n <Input\n name=\"phone\"\n value={formData.phone}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, phone: e.detail.value }))}\n placeholder=\"请输入手机号\"\n type=\"text\"\n />\n <Input\n name=\"name\"\n value={formData.name}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, name: e.detail.value }))}\n placeholder=\"请输入昵称\"\n type=\"text\"\n />\n <Input\n name=\"password\"\n value={formData.password}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, password: e.detail.value }))}\n placeholder=\"请输入密码\"\n type=\"password\"\n />\n <Input\n name=\"confirmPassword\"\n value={formData.confirmPassword}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, confirmPassword: e.detail.value }))}\n placeholder=\"请确认密码\"\n type=\"password\"\n />\n {error && <Text className=\"auth-error\">{error}</Text>}\n <Button className=\"auth-submit\" loading={loading} onClick={handleSubmit}>\n {loading ? '注册中...' : '注册'}\n </Button>\n {onSwitchToLogin && (\n <Button className=\"auth-switch\" onClick={onSwitchToLogin}>\n 去登录\n </Button>\n )}\n </View>\n </View>\n </View>\n );\n}\n","'use client';\n\nimport React, { useState } from 'react';\nimport { View, Text, Input, Button } from '@tarojs/components';\nimport { validatePhoneNumber, validatePassword } from '../../../utils/authUtils';\nimport type { ForgotPasswordModalProps } from '../../../types';\n\ntype MiniappInputEvent = {\n detail: {\n value: string;\n };\n};\n\nexport default function ForgotPasswordModal({ isOpen, onClose, onSuccess }: ForgotPasswordModalProps) {\n const [step, setStep] = useState<'phone' | 'verify' | 'reset'>('phone');\n const [formData, setFormData] = useState({\n phone: '',\n verificationCode: '',\n newPassword: '',\n confirmPassword: '',\n });\n const [error, setError] = useState('');\n const [loading, setLoading] = useState(false);\n\n if (!isOpen) return null;\n\n const handleSendCode = async () => {\n setError('');\n setLoading(true);\n\n if (!validatePhoneNumber(formData.phone)) {\n setError('请输入正确的手机号');\n setLoading(false);\n return;\n }\n\n try {\n const response = await fetch('/api/auth/send-verification-code', {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({ phone: formData.phone }),\n });\n const data = await response.json();\n if (data.success) {\n setStep('verify');\n } else {\n setError(data.message || '发送失败');\n }\n } catch (error) {\n console.error('💥 [ForgotPasswordModal] 发送验证码异常:', error);\n setError('发送失败,请稍后重试');\n } finally {\n setLoading(false);\n }\n };\n\n const handleVerifyCode = () => {\n if (!formData.verificationCode) {\n setError('请输入验证码');\n return;\n }\n setStep('reset');\n };\n\n const handleResetPassword = async () => {\n const passwordValidation = validatePassword(formData.newPassword);\n if (!passwordValidation.valid) {\n setError(passwordValidation.message || '密码格式错误');\n return;\n }\n if (formData.newPassword !== formData.confirmPassword) {\n setError('两次输入的密码不一致');\n return;\n }\n\n setLoading(true);\n try {\n const response = await fetch('/api/auth/reset-password', {\n method: 'POST',\n headers: { 'Content-Type': 'application/json' },\n body: JSON.stringify({\n phone: formData.phone,\n verificationCode: formData.verificationCode,\n newPassword: formData.newPassword,\n }),\n });\n const data = await response.json();\n if (data.success) {\n onSuccess();\n } else {\n setError(data.message || '重置失败');\n }\n } catch (error) {\n console.error('💥 [ForgotPasswordModal] 重置密码异常:', error);\n setError('重置失败,请稍后重试');\n } finally {\n setLoading(false);\n }\n };\n\n const renderContent = () => {\n if (step === 'phone') {\n return (\n <>\n <Input\n name=\"phone\"\n value={formData.phone}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, phone: e.detail.value }))}\n placeholder=\"请输入手机号\"\n type=\"text\"\n />\n <Button className=\"auth-submit\" loading={loading} onClick={handleSendCode}>\n {loading ? '发送中...' : '发送验证码'}\n </Button>\n </>\n );\n }\n\n if (step === 'verify') {\n return (\n <>\n <Input\n name=\"verificationCode\"\n value={formData.verificationCode}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, verificationCode: e.detail.value }))}\n placeholder=\"请输入验证码\"\n type=\"text\"\n />\n <Button className=\"auth-submit\" onClick={handleVerifyCode}>\n 下一步\n </Button>\n </>\n );\n }\n\n return (\n <>\n <Input\n name=\"newPassword\"\n value={formData.newPassword}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, newPassword: e.detail.value }))}\n placeholder=\"请输入新密码\"\n type=\"password\"\n />\n <Input\n name=\"confirmPassword\"\n value={formData.confirmPassword}\n onInput={(e: MiniappInputEvent) => setFormData(prev => ({ ...prev, confirmPassword: e.detail.value }))}\n placeholder=\"请确认新密码\"\n type=\"password\"\n />\n <Button className=\"auth-submit\" loading={loading} onClick={handleResetPassword}>\n {loading ? '提交中...' : '重置密码'}\n </Button>\n </>\n );\n };\n\n return (\n <View className=\"auth-modal\">\n <View className=\"auth-card\">\n <View className=\"auth-header\">\n <Text className=\"auth-title\">找回密码</Text>\n <Button className=\"auth-close\" onClick={onClose}>\n 关闭\n </Button>\n </View>\n <View className=\"auth-form\">\n {renderContent()}\n {error && <Text className=\"auth-error\">{error}</Text>}\n </View>\n </View>\n </View>\n );\n}\n","// ===== 用户相关类型 =====\n\n/**\n * 用户信息接口\n */\nexport interface User {\n id: number;\n phone: string;\n name?: string | null;\n email?: string | null;\n role: string;\n isActive: boolean;\n lastLoginAt?: Date | null;\n createdAt: Date;\n updatedAt: Date;\n}\n\n/**\n * 用户会话接口\n */\nexport interface UserSession {\n id: number;\n userId: number;\n sessionToken: string;\n expiresAt: Date;\n createdAt: Date;\n}\n\n// ===== 请求/响应类型 =====\n\n/**\n * 登录请求接口\n */\nexport interface LoginRequest {\n phone: string;\n password: string;\n}\n\n/**\n * 注册请求接口\n */\nexport interface RegisterRequest {\n phone: string;\n password: string;\n name?: string;\n}\n\n/**\n * 登录响应接口\n */\nexport interface LoginResponse {\n success: boolean;\n message: string;\n user?: User;\n sessionToken?: string;\n}\n\n/**\n * 注册响应接口\n */\nexport interface RegisterResponse {\n success: boolean;\n message: string;\n user?: User;\n sessionToken?: string;\n}\n\n/**\n * 会话验证响应接口\n */\nexport interface SessionValidationResponse {\n valid: boolean;\n user?: User;\n message?: string;\n}\n\n/**\n * 会话验证结果接口\n */\nexport interface SessionValidation {\n valid: boolean;\n user?: User;\n}\n\n// ===== 组件Props类型 =====\n\n/**\n * 登录模态框Props\n */\nexport interface LoginModalProps {\n isOpen: boolean;\n onClose: () => void;\n onSuccess: () => void;\n onSwitchToRegister?: () => void;\n}\n\n/**\n * 注册模态框Props\n */\nexport interface RegisterModalProps {\n isOpen: boolean;\n onClose: () => void;\n onSuccess: () => void;\n onSwitchToLogin?: () => void;\n}\n\n/**\n * 认证守卫Props\n */\nexport interface AuthGuardProps {\n children: React.ReactNode;\n fallback?: React.ReactNode;\n requireAuth?: boolean;\n}\n\n/**\n * 自定义菜单项接口\n */\nexport interface CustomMenuItem {\n id: string;\n label: string;\n icon?: React.ComponentType<any>;\n onClick: () => void;\n requireAuth?: boolean; // 是否需要登录才显示\n}\n\n/**\n * 用户菜单Props\n */\nexport interface UserMenuProps {\n customMenuItems?: CustomMenuItem[]; // 自定义菜单项\n className?: string; // 自定义样式类名\n}\n\n/**\n * 忘记密码模态框Props\n */\nexport interface ForgotPasswordModalProps {\n isOpen: boolean;\n onClose: () => void;\n onSuccess: () => void;\n}\n\n// ===== Hook返回类型 =====\n\n/**\n * useAuth Hook返回值类型\n */\nexport interface UseAuthReturn {\n user: User | null;\n loading: boolean;\n isAuthenticated: boolean;\n login: (credentials: LoginRequest) => Promise<{ success: boolean; user?: User; message?: string }>;\n register: (userData: RegisterRequest) => Promise<{ success: boolean; user?: User; message?: string }>;\n logout: () => Promise<void>;\n refreshUser: () => void;\n}\n\n// ===== 服务相关类型 =====\n\n/**\n * 认证服务接口\n */\nexport interface AuthService {\n verifyPassword(phone: string, password: string): Promise<User | null>;\n createUser(phone: string, password: string, name?: string): Promise<User>;\n createSession(userId: number): Promise<UserSession>;\n validateSession(sessionToken: string): Promise<SessionValidation>;\n deleteSession(sessionToken: string): Promise<void>;\n deleteUserSessions(userId: number): Promise<void>;\n updateLastLogin(userId: number): Promise<void>;\n cleanupExpiredSessions(): Promise<void>;\n getUserByPhone(phone: string): Promise<User | null>;\n sendVerificationCode(phone: string): Promise<string>;\n verifyCode(phone: string, code: string): Promise<boolean>;\n resetPassword(phone: string, newPassword: string): Promise<void>;\n cleanupExpiredVerificationCodes(): Promise<void>;\n}\n\n// ===== 工具函数类型 =====\n\n/**\n * API权限验证函数类型\n */\nexport type ValidateApiAuth = (request: Request) => Promise<User | null>;\n\n// ===== 常量类型 =====\n\n/**\n * 用户角色枚举\n */\nexport enum UserRole {\n USER = 'user',\n ADMIN = 'admin'\n}\n\n/**\n * 会话配置\n */\nexport interface SessionConfig {\n maxAge: number; // 会话最大存活时间(秒)\n cookieName: string; // Cookie名称\n secure: boolean; // 是否使用安全Cookie\n httpOnly: boolean; // 是否仅HTTP访问\n sameSite: 'strict' | 'lax' | 'none'; // SameSite策略\n} "]}