13 changed files with 1035 additions and 460 deletions
Unified View
Diff Options
-
220lib/controller/mine/add_bankcard_controller.dart
-
68lib/controller/mine/edit_info_controller.dart
-
43lib/controller/mine/withdraw_controller.dart
-
28lib/model/mine/bank_card_data.dart
-
18lib/model/mine/bank_card_ocr_data.dart
-
8lib/model/mine/user_data.dart
-
3lib/network/api_urls.dart
-
17lib/network/user_api.dart
-
129lib/network/user_api.g.dart
-
24lib/pages/home/user_information_page.dart
-
470lib/pages/mine/add_bankcard_page.dart
-
75lib/pages/mine/edit_info_page.dart
-
392lib/pages/mine/withdraw_page.dart
@ -0,0 +1,220 @@ |
|||||
|
import 'dart:io'; |
||||
|
|
||||
|
import 'package:dating_touchme_app/model/mine/bank_card_ocr_data.dart'; |
||||
|
import 'package:dating_touchme_app/network/user_api.dart'; |
||||
|
import 'package:dating_touchme_app/oss/oss_manager.dart'; |
||||
|
import 'package:flustars/flustars.dart'; |
||||
|
import 'package:flutter/material.dart'; |
||||
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; |
||||
|
import 'package:get/get.dart'; |
||||
|
import 'package:get/get_state_manager/src/simple/get_controllers.dart'; |
||||
|
import 'package:image_picker/image_picker.dart'; |
||||
|
import 'package:permission_handler/permission_handler.dart'; |
||||
|
|
||||
|
class AddBankcardController extends GetxController { |
||||
|
|
||||
|
late UserApi _userApi; |
||||
|
|
||||
|
|
||||
|
final username = ''.obs; |
||||
|
final usernameController = TextEditingController().obs; |
||||
|
|
||||
|
final cardNum = ''.obs; |
||||
|
final cardNumController = TextEditingController().obs; |
||||
|
|
||||
|
final bankName = ''.obs; |
||||
|
final bankNameController = TextEditingController().obs; |
||||
|
|
||||
|
final openingBank = ''.obs; |
||||
|
final openingBankController = TextEditingController().obs; |
||||
|
|
||||
|
@override |
||||
|
void onInit() { |
||||
|
super.onInit(); |
||||
|
_userApi = Get.find<UserApi>(); |
||||
|
} |
||||
|
|
||||
|
// 选择头像 - 业务逻辑处理 |
||||
|
Future<void> handleCameraCapture() async { |
||||
|
try { |
||||
|
// 请求相机权限 |
||||
|
final ok = await _ensurePermission( |
||||
|
Permission.camera, |
||||
|
denyToast: '相机权限被拒绝,请在设置中允许访问相机', |
||||
|
); |
||||
|
if (!ok) return; |
||||
|
|
||||
|
// 请求麦克风权限(部分设备拍照/录像会一并请求建议预授权) |
||||
|
await _ensurePermission(Permission.microphone, denyToast: '麦克风权限被拒绝'); |
||||
|
|
||||
|
// 权限通过后拍照 |
||||
|
final ImagePicker picker = ImagePicker(); |
||||
|
final XFile? photo = await picker.pickImage(source: ImageSource.camera); |
||||
|
|
||||
|
if (photo != null) { |
||||
|
await processSelectedImage(File(photo.path)); |
||||
|
} |
||||
|
} catch (e) { |
||||
|
print('拍照失败: $e'); |
||||
|
// 更友好的错误提示 |
||||
|
if (e.toString().contains('permission') || e.toString().contains('权限')) { |
||||
|
SmartDialog.showToast('相机权限被拒绝,请在设置中允许访问相机'); |
||||
|
} else if (e.toString().contains('camera') || |
||||
|
e.toString().contains('相机')) { |
||||
|
SmartDialog.showToast('设备没有可用的相机'); |
||||
|
} else { |
||||
|
SmartDialog.showToast('拍照失败,请重试'); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
Future<void> handleGallerySelection() async { |
||||
|
try { |
||||
|
// 请求相册/照片权限 |
||||
|
// final ok = await _ensurePermission( |
||||
|
// Permission.photos, |
||||
|
// // Android 上 photos 等价于 storage/mediaLibrary,permission_handler 会映射 |
||||
|
// denyToast: '相册权限被拒绝,请在设置中允许访问相册', |
||||
|
|
||||
|
// ); |
||||
|
// if (!ok) return; |
||||
|
|
||||
|
// 从相册选择图片 |
||||
|
final ImagePicker picker = ImagePicker(); |
||||
|
final XFile? image = await picker.pickImage(source: ImageSource.gallery); |
||||
|
|
||||
|
if (image != null) { |
||||
|
await processSelectedImage(File(image.path)); |
||||
|
} |
||||
|
} catch (e) { |
||||
|
print('选择图片失败: $e'); |
||||
|
// 更友好的错误提示 |
||||
|
if (e.toString().contains('permission') || e.toString().contains('权限')) { |
||||
|
SmartDialog.showToast('相册权限被拒绝,请在设置中允许访问相册'); |
||||
|
} else { |
||||
|
SmartDialog.showToast('选择图片失败,请重试'); |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
// 通用权限申请 |
||||
|
Future<bool> _ensurePermission(Permission permission, {String? denyToast}) async { |
||||
|
var status = await permission.status; |
||||
|
if (status.isGranted) return true; |
||||
|
|
||||
|
if (status.isDenied || status.isRestricted || status.isLimited) { |
||||
|
status = await permission.request(); |
||||
|
if (status.isGranted) return true; |
||||
|
if (denyToast != null) SmartDialog.showToast(denyToast); |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
if (status.isPermanentlyDenied) { |
||||
|
if (denyToast != null) SmartDialog.showToast('$denyToast,前往系统设置开启'); |
||||
|
// 延迟弹设置,避免与弹窗冲突 |
||||
|
Future.delayed(const Duration(milliseconds: 300), openAppSettings); |
||||
|
return false; |
||||
|
} |
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
|
||||
|
// 处理选中的图片 |
||||
|
Future<void> processSelectedImage(File imageFile) async { |
||||
|
try { |
||||
|
// 显示加载提示 |
||||
|
SmartDialog.showLoading(msg: '识别银行卡中...'); |
||||
|
String objectName = '${DateUtil.getNowDateMs()}.${imageFile.path.split('.').last}'; |
||||
|
String imageUrl = await OSSManager.instance.uploadFile(imageFile.readAsBytesSync(), objectName); |
||||
|
print('上传成功,图片URL: $imageUrl'); |
||||
|
ocrBankCard(imageUrl); |
||||
|
// SmartDialog.dismiss(); |
||||
|
// SmartDialog.showToast('银行卡上传成功'); |
||||
|
} catch (e) { |
||||
|
SmartDialog.dismiss(); |
||||
|
print('处理图片失败: $e'); |
||||
|
SmartDialog.showToast('银行卡上传失败,请重试'); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
ocrBankCard(String url) async { |
||||
|
try { |
||||
|
final response = await _userApi.recognizeBankCard({ |
||||
|
"url":url |
||||
|
}); |
||||
|
if (response.data.isSuccess) { |
||||
|
|
||||
|
final BankCardOcrData data = response.data.data ?? BankCardOcrData(); |
||||
|
bankName.value = data.bankName ?? ""; |
||||
|
bankNameController.value.value = TextEditingValue( |
||||
|
text: bankName.value, |
||||
|
selection: TextSelection.fromPosition(TextPosition(offset: bankName.value.length)), |
||||
|
); |
||||
|
cardNum.value = data.cardNum ?? ""; |
||||
|
cardNumController.value.value = TextEditingValue( |
||||
|
text: cardNum.value, |
||||
|
selection: TextSelection.fromPosition(TextPosition(offset: cardNum.value.length)), |
||||
|
); |
||||
|
SmartDialog.dismiss(); |
||||
|
SmartDialog.showToast('银行卡识别成功'); |
||||
|
} |
||||
|
else{ |
||||
|
|
||||
|
SmartDialog.showToast(response.data.message); |
||||
|
|
||||
|
return; |
||||
|
} |
||||
|
} catch (e){ |
||||
|
print('银行卡识别失败: $e'); |
||||
|
SmartDialog.showToast('银行卡识别失败,请重试'); |
||||
|
return; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
saveCard() async { |
||||
|
|
||||
|
if(username.value == ""){ |
||||
|
SmartDialog.showToast('请输入持卡人姓名'); |
||||
|
return; |
||||
|
} |
||||
|
if(cardNum.value == ""){ |
||||
|
SmartDialog.showToast('请输入卡号'); |
||||
|
return; |
||||
|
} |
||||
|
if(bankName.value == ""){ |
||||
|
SmartDialog.showToast('请输入所属银行'); |
||||
|
return; |
||||
|
} |
||||
|
if(openingBank.value == ""){ |
||||
|
SmartDialog.showToast('请输入开户行'); |
||||
|
return; |
||||
|
} |
||||
|
|
||||
|
final payload = { |
||||
|
"ownerName": username.value, |
||||
|
"cardNum": cardNum.value, |
||||
|
"bankName": bankName.value, |
||||
|
"openingBank": openingBank.value, |
||||
|
}; |
||||
|
|
||||
|
try { |
||||
|
final response = await _userApi.createBankCardByIndividual(payload); |
||||
|
if (response.data.isSuccess) { |
||||
|
|
||||
|
SmartDialog.showToast('保存成功'); |
||||
|
Get.back(); |
||||
|
} |
||||
|
else{ |
||||
|
|
||||
|
SmartDialog.showToast(response.data.message); |
||||
|
|
||||
|
return; |
||||
|
} |
||||
|
} catch (e){ |
||||
|
print('保存失败: $e'); |
||||
|
SmartDialog.showToast('保存失败,请重试'); |
||||
|
return; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
} |
||||
@ -0,0 +1,43 @@ |
|||||
|
import 'package:dating_touchme_app/model/mine/bank_card_data.dart'; |
||||
|
import 'package:dating_touchme_app/network/user_api.dart'; |
||||
|
import 'package:flutter_smart_dialog/flutter_smart_dialog.dart'; |
||||
|
import 'package:get/get.dart'; |
||||
|
import 'package:get/get_state_manager/src/simple/get_controllers.dart'; |
||||
|
|
||||
|
class WithdrawController extends GetxController { |
||||
|
|
||||
|
late UserApi _userApi; |
||||
|
|
||||
|
final money = 3880.obs; |
||||
|
|
||||
|
final bankCardList = <BankCardData>[].obs; |
||||
|
|
||||
|
final nowBankCard = BankCardData().obs; |
||||
|
|
||||
|
@override |
||||
|
void onInit() { |
||||
|
super.onInit(); |
||||
|
_userApi = Get.find<UserApi>(); |
||||
|
|
||||
|
getBankCard(); |
||||
|
} |
||||
|
|
||||
|
getBankCard() async { |
||||
|
try{ |
||||
|
final response = await _userApi.listBankCardByIndividual({}); |
||||
|
if (response.data.isSuccess && response.data.data != null) { |
||||
|
final data = response.data.data; |
||||
|
bankCardList.clear(); |
||||
|
bankCardList.addAll(data?.toList() ?? []); |
||||
|
} else { |
||||
|
|
||||
|
// 响应失败,抛出异常 |
||||
|
throw Exception(response.data.message ?? '获取数据失败'); |
||||
|
} |
||||
|
} catch (e) { |
||||
|
print('银行卡列表获取失败: $e'); |
||||
|
SmartDialog.showToast('银行卡列表获取失败'); |
||||
|
rethrow; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,28 @@ |
|||||
|
class BankCardData { |
||||
|
String? id; |
||||
|
String? ownerName; |
||||
|
String? bankName; |
||||
|
String? cardNum; |
||||
|
String? openingBank; |
||||
|
|
||||
|
BankCardData( |
||||
|
{this.id, this.ownerName, this.bankName, this.cardNum, this.openingBank}); |
||||
|
|
||||
|
BankCardData.fromJson(Map<String, dynamic> json) { |
||||
|
id = json['id']; |
||||
|
ownerName = json['ownerName']; |
||||
|
bankName = json['bankName']; |
||||
|
cardNum = json['cardNum']; |
||||
|
openingBank = json['openingBank']; |
||||
|
} |
||||
|
|
||||
|
Map<String, dynamic> toJson() { |
||||
|
final Map<String, dynamic> data = new Map<String, dynamic>(); |
||||
|
data['id'] = this.id; |
||||
|
data['ownerName'] = this.ownerName; |
||||
|
data['bankName'] = this.bankName; |
||||
|
data['cardNum'] = this.cardNum; |
||||
|
data['openingBank'] = this.openingBank; |
||||
|
return data; |
||||
|
} |
||||
|
} |
||||
@ -0,0 +1,18 @@ |
|||||
|
class BankCardOcrData { |
||||
|
String? bankName; |
||||
|
String? cardNum; |
||||
|
|
||||
|
BankCardOcrData({this.bankName, this.cardNum}); |
||||
|
|
||||
|
BankCardOcrData.fromJson(Map<String, dynamic> json) { |
||||
|
bankName = json['bankName']; |
||||
|
cardNum = json['cardNum']; |
||||
|
} |
||||
|
|
||||
|
Map<String, dynamic> toJson() { |
||||
|
final Map<String, dynamic> data = new Map<String, dynamic>(); |
||||
|
data['bankName'] = this.bankName; |
||||
|
data['cardNum'] = this.cardNum; |
||||
|
return data; |
||||
|
} |
||||
|
} |
||||
Write
Preview
Loading…
Cancel
Save