2
0

hooks.tsx 7.4 KB

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