You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
153 lines
5.0 KiB
153 lines
5.0 KiB
// lib/modules/auth/controllers/auth_controller.dart |
|
import 'package:flutter/material.dart'; // 导入此库以使用 TextEditingController |
|
import 'package:get/get.dart'; |
|
import 'package:dio/dio.dart'; |
|
import 'package:get_storage/get_storage.dart'; |
|
import 'package:connectivity_plus/connectivity_plus.dart'; |
|
import 'package:problem_check_system/data/providers/auth_provider.dart'; |
|
import 'package:problem_check_system/data/models/login_model.dart'; |
|
import 'package:problem_check_system/app/routes/app_routes.dart'; |
|
|
|
class AuthController extends GetxController { |
|
final AuthProvider _authProvider; |
|
final GetStorage _storage; |
|
|
|
// 将 TextEditingController 移动到控制器中,由控制器管理它们的生命周期 |
|
late final TextEditingController usernameController; |
|
late final TextEditingController passwordController; |
|
|
|
final isLoading = false.obs; |
|
final rememberPassword = false.obs; |
|
|
|
static const _usernameKey = 'username'; |
|
static const _passwordKey = 'password'; |
|
|
|
AuthController({ |
|
required AuthProvider authProvider, |
|
required GetStorage storage, |
|
}) : _authProvider = authProvider, |
|
_storage = storage; |
|
|
|
@override |
|
void onInit() { |
|
super.onInit(); |
|
// 在 onInit 中初始化控制器 |
|
usernameController = TextEditingController(); |
|
passwordController = TextEditingController(); |
|
_loadSavedCredentials(); |
|
} |
|
|
|
@override |
|
void onClose() { |
|
// 在 onClose 中销毁控制器,防止内存泄漏 |
|
usernameController.dispose(); |
|
passwordController.dispose(); |
|
super.onClose(); |
|
} |
|
|
|
void _loadSavedCredentials() { |
|
final savedUsername = _storage.read(_usernameKey); |
|
final savedPassword = _storage.read(_passwordKey); |
|
|
|
if (savedUsername != null && savedPassword != null) { |
|
// 从本地存储加载值,并设置给控制器 |
|
usernameController.text = savedUsername; |
|
passwordController.text = savedPassword; |
|
rememberPassword.value = true; |
|
} |
|
} |
|
|
|
String getToken() { |
|
return _storage.read('token') ?? ''; |
|
} |
|
|
|
// 移除了 updateUsername 和 updatePassword 方法,因为不再需要它们来同步值。 |
|
// TextEditingController 本身就是值的来源。 |
|
|
|
Future<void> login() async { |
|
// 直接从控制器获取文本内容 |
|
final username = usernameController.text; |
|
final password = passwordController.text; |
|
|
|
if (username.isEmpty || password.isEmpty) { |
|
Get.snackbar('输入错误', '用户名和密码不能为空'); |
|
return; |
|
} |
|
|
|
isLoading.value = true; |
|
|
|
var connectivityResult = await (Connectivity().checkConnectivity()); |
|
bool isConnected = |
|
connectivityResult.contains(ConnectivityResult.mobile) || |
|
connectivityResult.contains(ConnectivityResult.wifi) || |
|
connectivityResult.contains(ConnectivityResult.ethernet); |
|
|
|
if (isConnected) { |
|
await _onlineLogin(username, password); |
|
} else { |
|
await _offlineLogin(username, password); |
|
} |
|
|
|
isLoading.value = false; |
|
} |
|
|
|
Future<void> _onlineLogin(String username, String password) async { |
|
try { |
|
final loginData = LoginModel(username: username, password: password); |
|
|
|
final response = await _authProvider.signIn(loginData); |
|
|
|
if (response.statusCode == 200 && response.data['token'] != null) { |
|
final token = response.data['token']; |
|
final refreshToken = response.data['refreshToken']; |
|
|
|
_storage.write('token', token); |
|
_storage.write('refreshToken', refreshToken); |
|
|
|
// 如果记住密码,保存用户名和密码 |
|
if (rememberPassword.value) { |
|
_storage.write(_usernameKey, username); |
|
_storage.write(_passwordKey, password); |
|
} else { |
|
// 否则清除所有本地凭据 |
|
_storage.remove(_usernameKey); |
|
_storage.remove(_passwordKey); |
|
} |
|
|
|
Get.offAllNamed(AppRoutes.home); |
|
} else { |
|
Get.snackbar('登录失败', '服务器返回了异常数据'); |
|
} |
|
} on DioException catch (e) { |
|
if (e.response?.data != null && e.response?.data['detail'] != null) { |
|
final serverMessage = e.response!.data['detail']; |
|
Get.snackbar('登录失败', serverMessage); |
|
} else { |
|
Get.snackbar('网络错误', '登录失败,请检查您的网络连接'); |
|
} |
|
} catch (e) { |
|
Get.snackbar('错误', '发生未知错误: ${e.toString()}'); |
|
} |
|
} |
|
|
|
Future<void> _offlineLogin(String username, String password) async { |
|
final storedUsername = _storage.read(_usernameKey); |
|
final storedPassword = _storage.read(_passwordKey); |
|
|
|
if (storedUsername != null && |
|
storedPassword != null && |
|
storedUsername == username && |
|
storedPassword == password) { |
|
Get.offAllNamed(AppRoutes.home); |
|
Get.snackbar('离线登录成功', '您已离线登录到系统'); |
|
} else { |
|
Get.snackbar('离线登录失败', '无网络连接,且无法验证本地凭证'); |
|
} |
|
} |
|
|
|
Future<void> logout() async { |
|
_storage.remove('token'); |
|
_storage.remove('refreshToken'); |
|
Get.offAllNamed(AppRoutes.login); |
|
} |
|
}
|
|
|