Login_20250812104443.tsx 6.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203
  1. // entrypoints/popup/pages/Login.tsx
  2. import React, { useState } from 'react';
  3. import { useNavigate } from 'react-router-dom';
  4. import { Button, Input, Message, Form, Grid } from '@alifd/next';
  5. const { Row, Col } = Grid
  6. export default function Login({ setIsLoggedIn }: { setIsLoggedIn: (loggedIn: boolean) => void }) {
  7. const [username, setUsername] = useState('');
  8. const [password, setPassword] = useState('');
  9. const [loading, setLoading] = useState(false);
  10. const [errors, setErrors] = useState<{ username?: string, password?: string }>({});
  11. const [keys, setKeys] = useState('');
  12. const [validImg, setValidImg] = useState('');
  13. const navigate = useNavigate();
  14. useEffect(() => {
  15. evVerify()
  16. }, [])
  17. const validateForm = () => {
  18. const newErrors: { username?: string, password?: string } = {};
  19. if (!username.trim()) {
  20. newErrors.username = '请输入用户名';
  21. }
  22. if (!password) {
  23. newErrors.password = '请输入密码';
  24. } else if (password.length < 6) {
  25. newErrors.password = '密码长度至少为6位';
  26. }
  27. setErrors(newErrors);
  28. return Object.keys(newErrors).length === 0;
  29. };
  30. const evVerify = () => {
  31. console.log('username')
  32. fetch('https://user.landwu.com/api/user/verify', {
  33. method: 'POST',
  34. headers: {
  35. 'Content-Type': 'application/json',
  36. }
  37. }).then(resjson => {
  38. if (!resjson.ok) {
  39. throw new Error('登录失败');
  40. }
  41. return resjson.json(); // 正确解析 Response 为 JSON 数据
  42. }).then(res => {
  43. const { data = {} } = res;
  44. const { data: info = {} } = data;
  45. const { key = "", img = "" } = info;
  46. setKeys(key)
  47. setValidImg(img)
  48. })
  49. }
  50. const handleLogin = async (e: React.FormEvent) => {
  51. e.preventDefault();
  52. if (!validateForm()) {
  53. return;
  54. }
  55. setLoading(true);
  56. try {
  57. // 发送登录请求
  58. const response = await fetch('/api/login', {
  59. method: 'POST',
  60. headers: {
  61. 'Content-Type': 'application/json',
  62. },
  63. body: JSON.stringify({
  64. username,
  65. password,
  66. code
  67. }),
  68. });
  69. const data = await response.json();
  70. if (response.ok && data.success) {
  71. // 登录成功
  72. localStorage.setItem('isLoggedIn', 'true');
  73. // 如果后端返回token,也可以存储token
  74. // localStorage.setItem('token', data.token);
  75. setIsLoggedIn(true);
  76. navigate('/');
  77. Message.success('登录成功');
  78. } else {
  79. // 登录失败
  80. Message.error(data.message || '登录失败,请检查用户名和密码');
  81. }
  82. } catch (error) {
  83. console.error('登录请求失败:', error);
  84. Message.error('网络错误,请稍后重试');
  85. } finally {
  86. setLoading(false);
  87. }
  88. // 模拟登录请求
  89. // setTimeout(() => {
  90. // // 这里添加实际的登录逻辑
  91. // if (username && password) {
  92. // // 模拟登录成功
  93. // localStorage.setItem('isLoggedIn', 'true');
  94. // setIsLoggedIn(true);
  95. // // 跳转到首页
  96. // navigate('/');
  97. // Message.success('登录成功');
  98. // } else {
  99. // Message.error('登录失败,请检查用户名和密码');
  100. // }
  101. // setLoading(false);
  102. // }, 1000);
  103. };
  104. return (
  105. <div className="bg-gray-50 p-4">
  106. <div className="text-center mb-6">
  107. <h1 className="text-2xl font-bold text-gray-800">登录</h1>
  108. </div>
  109. <Form onSubmit={handleLogin}>
  110. <Form.Item
  111. label="用户名"
  112. required
  113. help={errors.username}
  114. validateState={errors.username ? 'error' : undefined}
  115. >
  116. <Input
  117. placeholder="请输入用户名"
  118. value={username}
  119. onChange={(value) => {
  120. setUsername(value);
  121. if (errors.username) {
  122. setErrors({ ...errors, username: undefined });
  123. }
  124. }}
  125. disabled={loading}
  126. />
  127. </Form.Item>
  128. <Form.Item
  129. label="密码"
  130. required
  131. help={errors.password}
  132. validateState={errors.password ? 'error' : undefined}
  133. >
  134. <Input
  135. htmlType="password"
  136. placeholder="请输入密码"
  137. value={password}
  138. onChange={(value) => {
  139. setPassword(value);
  140. if (errors.password) {
  141. setErrors({ ...errors, password: undefined });
  142. }
  143. }}
  144. disabled={loading}
  145. />
  146. </Form.Item>
  147. <Row>
  148. <Col span={16}>
  149. <Form.Item
  150. label="验证码"
  151. name="code"
  152. required
  153. >
  154. <Input
  155. size="large"
  156. />
  157. </Form.Item>
  158. </Col>
  159. <Col span={8}>
  160. <img
  161. src={validImg}
  162. style={{ width: 70, height: 38 }}
  163. onClick={evVerify}
  164. />
  165. </Col>
  166. </Row>
  167. <Form.Item>
  168. <Button
  169. type="primary"
  170. htmlType="submit"
  171. loading={loading}
  172. className="w-full"
  173. >
  174. {loading ? '登录中...' : '登录'}
  175. </Button>
  176. </Form.Item>
  177. </Form>
  178. <div className="mt-4 text-center text-sm text-gray-500">
  179. <p>忘记密码?请联系管理员</p>
  180. </div>
  181. </div>
  182. );
  183. }