站点登录

This commit is contained in:
2025-11-05 10:10:47 +08:00
parent 6faf03a331
commit ea18b71523
10 changed files with 329 additions and 99 deletions

View File

@@ -0,0 +1,37 @@
import 'package:encrypt/encrypt.dart';
class LoginUtil {
// 定义密钥
// 对于ECB模式我们只需要Key不需要IV (初始化向量)
static final _keyString = '915eae87951a448c86c47796e44c1fcf';
static final _key = Key.fromUtf8(_keyString);
// 【核心修改】创建使用 ECB 模式的加密器实例
// 1. mode: AESMode.ecb -> 使用ECB模式它不需要IV。
// 2. padding: 'PKCS7' -> 在encrypt库中PKCS5和PKCS7的填充方式是兼容的。
static final _encrypter = Encrypter(AES(_key, mode: AESMode.ecb, padding: 'PKCS7'));
/// AES 加密方法
static String encrypt(String plainText) {
if (plainText.isEmpty) {
return '';
}
// 【核心修改】调用 encrypt 方法时不再需要传递 iv
final encrypted = _encrypter.encrypt(plainText);
// 返回Base64编码的密文这是网络传输的标准做法
return encrypted.base64;
}
/// AES 解密方法 (可选,如果需要解密的话)
static String decrypt(String encryptedText) {
if (encryptedText.isEmpty) {
return '';
}
final encrypted = Encrypted.fromBase64(encryptedText);
// 【核心修改】调用 decrypt 方法时不再需要传递 iv
final decrypted = _encrypter.decrypt(encrypted);
return decrypted;
}
}

View File

@@ -1,19 +1,75 @@
class BaseModel {
bool? success;
String? type;
String? url;
/// 通用的 API 响应基础模型
/// 使用泛型 <T> 来适应 'data' 字段中任何可能的数据类型
class BaseModel<T> {
final int code; // 状态码0正常其他异常
final bool status; // 状态布尔值true正常false异常
final String message; // 消息,例如 "success"
final T? data; // 核心数据,使用泛型 T可以是任何类型
final int time; // 时间戳
final dynamic error; // 错误信息,可以是任何类型或 null
BaseModel({this.success, this.type, this.url});
BaseModel({
required this.code,
required this.status,
required this.message,
this.data, // data 可以为 null
required this.time,
this.error, // error 可以为 null
});
factory BaseModel.fromJson(Map<String, dynamic> json) => BaseModel(
success: json['success']?.toString().contains("true"),
type: json['type']?.toString(),
url: json['url']?.toString(),
);
/// fromJson 工厂构造函数(重构后)
factory BaseModel.fromJson(
Map<String, dynamic> json, {
T? Function(dynamic dataJson)? dataBuilder,
}) {
// 使用一个辅助函数来安全地转换类型,防止因类型不匹配(如 "0" vs 0而崩溃
int _parseInt(dynamic value) {
if (value is int) return value;
if (value is String) return int.tryParse(value) ?? -1;
return -1;
}
T? finalData;
// 检查 'data' 字段是否存在
if (json.containsKey('data') && json['data'] != null) {
if (dataBuilder != null) {
// 如果提供了 dataBuilder就用它来解析成具体的 T 类型对象
finalData = dataBuilder(json['data']);
} else {
// 如果没有提供 dataBuilder但 T 不是 dynamic我们假设 data 就是 T 类型
// 这在使用 BaseModel<Map<String, dynamic>> 时很有用
if (T != dynamic) {
try {
finalData = json['data'] as T?;
} catch(e) {
// 如果直接转换失败,保持为 null避免崩溃
finalData = null;
}
} else {
// 如果 T 是 dynamic直接赋值
finalData = json['data'];
}
}
}
return BaseModel<T>(
code: _parseInt(json['code']),
status: json['status'] as bool? ?? false,
message: json['message']?.toString() ?? 'No message',
time: _parseInt(json['time']),
data: finalData,
error: json['error'],
);
}
/// toJson 方法 (可选)
Map<String, dynamic> toJson() => {
if (success != null) 'success': success,
if (type != null) 'type': type,
if (url != null) 'url': url,
};
'code': code,
'status': status,
'message': message,
'time': time,
'data': data,
'error': error,
};
}

View File

@@ -7,6 +7,10 @@ class AppTheme {
static const Color themeColor = Color(0xFF0c83c3);
static const String test_service_url = "http://beta-esg.api.lnh2e.com/";
static const String release_service_url = "";
static const Color secondaryColor = Colors.orange;
static const Color darkThemeColor = Color(0xFF032896);