hooks.tsx 7.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258
  1. import React, { createContext, useContext, useState, useEffect } from 'react';
  2. import axios from 'axios';
  3. import { getLocalStorageWithExpiry, setLocalStorageWithExpiry } from './utils.ts';
  4. import type { User, AuthContextType, ThemeContextType, ThemeSettings } from '../share/types.ts';
  5. import { ThemeMode, FontSize, CompactMode } from '../share/types.ts';
  6. import { AuthAPI, ThemeAPI } from './api.ts';
  7. // 创建axios实例
  8. const api = axios.create({
  9. baseURL: window.CONFIG?.API_BASE_URL || '/api',
  10. timeout: 10000,
  11. headers: {
  12. 'Content-Type': 'application/json',
  13. }
  14. });
  15. // 请求拦截器添加token
  16. api.interceptors.request.use(
  17. (config) => {
  18. const token = getLocalStorageWithExpiry('token');
  19. if (token) {
  20. config.headers['Authorization'] = `Bearer ${token}`;
  21. }
  22. return config;
  23. },
  24. (error) => {
  25. return Promise.reject(error);
  26. }
  27. );
  28. // 响应拦截器处理错误
  29. api.interceptors.response.use(
  30. (response) => {
  31. return response;
  32. },
  33. (error) => {
  34. if (error.response && error.response.status === 401) {
  35. // 清除本地存储并刷新页面
  36. localStorage.removeItem('token');
  37. localStorage.removeItem('user');
  38. window.location.href = '/mobile/login';
  39. }
  40. return Promise.reject(error);
  41. }
  42. );
  43. // 默认主题设置
  44. const defaultThemeSettings: ThemeSettings = {
  45. user_id: 0,
  46. theme_mode: ThemeMode.LIGHT,
  47. primary_color: '#3B82F6', // 蓝色
  48. background_color: '#F9FAFB',
  49. text_color: '#111827',
  50. border_radius: 8,
  51. font_size: FontSize.MEDIUM,
  52. is_compact: CompactMode.NORMAL
  53. };
  54. // 创建认证上下文
  55. const AuthContext = createContext<AuthContextType | null>(null);
  56. // 创建主题上下文
  57. const ThemeContext = createContext<ThemeContextType | null>(null);
  58. // 认证提供者组件
  59. export const AuthProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {
  60. const [user, setUser] = useState<User | null>(null);
  61. const [token, setToken] = useState<string | null>(null);
  62. const [isLoading, setIsLoading] = useState(true);
  63. // 初始化时从本地存储获取用户信息和令牌
  64. useEffect(() => {
  65. const storedToken = getLocalStorageWithExpiry('token');
  66. const storedUser = getLocalStorageWithExpiry('user');
  67. if (storedToken && storedUser) {
  68. setToken(storedToken);
  69. setUser(storedUser);
  70. }
  71. setIsLoading(false);
  72. }, []);
  73. // 登录函数
  74. const login = async (username: string, password: string) => {
  75. try {
  76. const response = await AuthAPI.login(username, password);
  77. const { token, user } = response;
  78. // 保存到状态和本地存储
  79. setToken(token);
  80. setUser(user);
  81. setLocalStorageWithExpiry('token', token, 24); // 24小时过期
  82. setLocalStorageWithExpiry('user', user, 24);
  83. } catch (error) {
  84. console.error('登录失败:', error);
  85. throw error;
  86. }
  87. };
  88. // 登出函数
  89. const logout = async () => {
  90. try {
  91. // 调用登出API
  92. await AuthAPI.logout();
  93. } catch (error) {
  94. console.error('登出API调用失败:', error);
  95. } finally {
  96. // 无论API调用成功与否,都清除本地状态
  97. setToken(null);
  98. setUser(null);
  99. localStorage.removeItem('token');
  100. localStorage.removeItem('user');
  101. }
  102. };
  103. return (
  104. <AuthContext.Provider
  105. value={{
  106. user,
  107. token,
  108. login,
  109. logout,
  110. isAuthenticated: !!token,
  111. isLoading
  112. }}
  113. >
  114. {children}
  115. </AuthContext.Provider>
  116. );
  117. };
  118. // 主题提供者组件
  119. export const ThemeProvider: React.FC<{ children: React.ReactNode }> = ({ children }) => {
  120. const [currentTheme, setCurrentTheme] = useState<ThemeSettings>(() => {
  121. const storedTheme = localStorage.getItem('theme');
  122. return storedTheme ? JSON.parse(storedTheme) : defaultThemeSettings;
  123. });
  124. const isDark = currentTheme.theme_mode === ThemeMode.DARK;
  125. // 更新主题(实时预览)
  126. const updateTheme = (theme: Partial<ThemeSettings>) => {
  127. setCurrentTheme(prev => {
  128. const updatedTheme = { ...prev, ...theme };
  129. localStorage.setItem('theme', JSON.stringify(updatedTheme));
  130. return updatedTheme;
  131. });
  132. };
  133. // 保存主题到后端
  134. const saveTheme = async (theme: Partial<ThemeSettings>): Promise<ThemeSettings> => {
  135. try {
  136. const updatedTheme = { ...currentTheme, ...theme };
  137. const data = await ThemeAPI.updateThemeSettings(updatedTheme);
  138. setCurrentTheme(data);
  139. localStorage.setItem('theme', JSON.stringify(data));
  140. return data;
  141. } catch (error) {
  142. console.error('保存主题失败:', error);
  143. throw error;
  144. }
  145. };
  146. // 重置主题
  147. const resetTheme = async (): Promise<ThemeSettings> => {
  148. try {
  149. const data = await ThemeAPI.resetThemeSettings();
  150. setCurrentTheme(data);
  151. localStorage.setItem('theme', JSON.stringify(data));
  152. return data;
  153. } catch (error) {
  154. console.error('重置主题失败:', error);
  155. // 如果API失败,至少重置到默认主题
  156. setCurrentTheme(defaultThemeSettings);
  157. localStorage.setItem('theme', JSON.stringify(defaultThemeSettings));
  158. return defaultThemeSettings;
  159. }
  160. };
  161. // 切换主题模式(亮色/暗色)
  162. const toggleTheme = () => {
  163. const newMode = isDark ? ThemeMode.LIGHT : ThemeMode.DARK;
  164. const updatedTheme = {
  165. ...currentTheme,
  166. theme_mode: newMode,
  167. // 暗色和亮色模式下自动调整背景色和文字颜色
  168. background_color: newMode === ThemeMode.DARK ? '#121212' : '#F9FAFB',
  169. text_color: newMode === ThemeMode.DARK ? '#E5E7EB' : '#111827'
  170. };
  171. setCurrentTheme(updatedTheme);
  172. localStorage.setItem('theme', JSON.stringify(updatedTheme));
  173. };
  174. // 主题变化时应用CSS变量
  175. useEffect(() => {
  176. document.documentElement.style.setProperty('--primary-color', currentTheme.primary_color);
  177. document.documentElement.style.setProperty('--background-color', currentTheme.background_color || '#F9FAFB');
  178. document.documentElement.style.setProperty('--text-color', currentTheme.text_color || '#111827');
  179. document.documentElement.style.setProperty('--border-radius', `${currentTheme.border_radius || 8}px`);
  180. // 设置字体大小
  181. let rootFontSize = '16px'; // 默认中等字体
  182. if (currentTheme.font_size === FontSize.SMALL) {
  183. rootFontSize = '14px';
  184. } else if (currentTheme.font_size === FontSize.LARGE) {
  185. rootFontSize = '18px';
  186. }
  187. document.documentElement.style.setProperty('--font-size', rootFontSize);
  188. // 设置暗色模式类
  189. if (isDark) {
  190. document.documentElement.classList.add('dark');
  191. } else {
  192. document.documentElement.classList.remove('dark');
  193. }
  194. }, [currentTheme, isDark]);
  195. return (
  196. <ThemeContext.Provider
  197. value={{
  198. isDark,
  199. currentTheme,
  200. updateTheme,
  201. saveTheme,
  202. resetTheme,
  203. toggleTheme
  204. }}
  205. >
  206. {children}
  207. </ThemeContext.Provider>
  208. );
  209. };
  210. // 使用上下文的钩子
  211. export const useAuth = () => {
  212. const context = useContext(AuthContext);
  213. if (!context) {
  214. throw new Error('useAuth必须在AuthProvider内部使用');
  215. }
  216. return context;
  217. };
  218. export const useTheme = () => {
  219. const context = useContext(ThemeContext);
  220. if (!context) {
  221. throw new Error('useTheme必须在ThemeProvider内部使用');
  222. }
  223. return context;
  224. };