| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461 |
- import 'dart:convert';
- import 'dart:typed_data';
- import 'package:dio/dio.dart';
- import 'package:flutter_riverpod/flutter_riverpod.dart';
- import '../../core/network/api_client.dart';
- import '../../core/network/api_response.dart';
- import '../../app.dart';
- import '../../shared/models/pagination_model.dart';
- import '../../shared/models/bill_attachment.dart';
- import '../../core/navigation/host_app_channel.dart';
- import '../../core/utils/date_utils.dart' as du;
- import 'expense_model.dart';
- import 'expense_list_controller.dart';
- final expenseApiProvider = Provider<ExpenseApi>(
- (ref) => ExpenseApi(ref.read(apiClientProvider)),
- );
- // ═══ 参考数据模型(API 返回) ═══
- class CostTypeItem {
- final String typeNo;
- final String typeName;
- final String accNo;
- final String accName;
- const CostTypeItem({required this.typeNo, required this.typeName, required this.accNo, required this.accName});
- factory CostTypeItem.fromJson(Map<String, dynamic> json) => CostTypeItem(
- typeNo: json['typeNo'] as String? ?? '',
- typeName: json['typeName'] as String? ?? '',
- accNo: json['accNo'] as String? ?? '',
- accName: json['accName'] as String? ?? '',
- );
- }
- class ProjectCodeItem {
- final String objNo;
- final String name;
- const ProjectCodeItem({required this.objNo, required this.name});
- factory ProjectCodeItem.fromJson(Map<String, dynamic> json) => ProjectCodeItem(
- objNo: json['objNo'] as String? ?? '',
- name: json['name'] as String? ?? '',
- );
- }
- class DepartmentItem {
- final String dep;
- final String name;
- const DepartmentItem({required this.dep, required this.name});
- factory DepartmentItem.fromJson(Map<String, dynamic> json) => DepartmentItem(
- dep: json['dep'] as String? ?? '',
- name: json['name'] as String? ?? '',
- );
- }
- class CustomerItem {
- final String cusNo;
- final String name;
- const CustomerItem({required this.cusNo, required this.name});
- factory CustomerItem.fromJson(Map<String, dynamic> json) => CustomerItem(
- cusNo: json['cusNo'] as String? ?? '',
- name: json['name'] as String? ?? '',
- );
- }
- class CurrencyItem {
- final String curId;
- final String name;
- const CurrencyItem({required this.curId, required this.name});
- factory CurrencyItem.fromJson(Map<String, dynamic> json) => CurrencyItem(
- curId: json['curId'] as String? ?? '',
- name: json['name'] as String? ?? '',
- );
- }
- class EmployeeItem {
- final String salNo;
- final String name;
- final String dep;
- final String tel;
- final String email;
- final String bnkNo;
- final String bnkId;
- final String accName;
- const EmployeeItem({required this.salNo, required this.name, this.dep = '', this.tel = '', this.email = '', this.bnkNo = '', this.bnkId = '', this.accName = ''});
- factory EmployeeItem.fromJson(Map<String, dynamic> json) => EmployeeItem(
- salNo: json['salNo'] as String? ?? '',
- name: json['name'] as String? ?? '',
- dep: json['dep'] as String? ?? '',
- tel: json['tel'] as String? ?? '',
- email: json['email'] as String? ?? '',
- bnkNo: json['bnkNo'] as String? ?? '',
- bnkId: json['bnkId'] as String? ?? '',
- accName: json['accName'] as String? ?? '',
- );
- }
- class ExpenseApi {
- final ApiClient _client;
- ExpenseApi(this._client);
- /// 费用报销列表(分页)
- Future<PaginatedData<ExpenseModel>> fetchList({
- String status = '',
- String keyword = '',
- String startDate = '',
- String endDate = '',
- String usr = '',
- String sortDir = 'DESC',
- int page = 1,
- int size = 20,
- }) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetExpenseReports',
- queryParameters: {
- 'status': status,
- 'keyword': keyword,
- 'startDate': startDate,
- 'endDate': endDate,
- 'usr': usr,
- 'sortDir': sortDir,
- 'page': page,
- 'size': size,
- },
- );
- return PaginatedData.fromJson(response.data!, ExpenseModel.fromJson);
- }
- /// 费用报销详情(主表+明细)
- Future<ExpenseModel> fetchDetail(String billNo) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetExpenseReportDetail',
- queryParameters: {'billNo': billNo},
- );
- return ExpenseModel.fromJson(response.data!);
- }
- /// 提交审批,返回报销单号(提取失败时返回 null,不影响主流程)
- /// BillSave 返回格式: { callok:true, resultData:{ BIL_NO:"BX20267020005", ... } }
- Future<String?> submit(Map<String, dynamic> data) async {
- final response = await _client.post<Map<String, dynamic>>('/OA/BillSave', data: {
- 'erpCategory': 'MasterService',
- 'billId': 'BX',
- 'procId': '',
- 'data': data,
- });
- final resData = response.data;
- if (resData == null) return null;
- final resultData = resData['resultData'];
- if (resultData is Map<String, dynamic>) {
- final bilNo = resultData['BIL_NO'] as String?;
- if (bilNo != null && bilNo.isNotEmpty) return bilNo;
- }
- if (resultData is String && resultData.isNotEmpty) {
- try {
- final parsed = json.decode(resultData) as Map<String, dynamic>;
- final bilNo = parsed['BIL_NO'] as String?;
- if (bilNo != null && bilNo.isNotEmpty) return bilNo;
- } catch (_) {}
- }
- final rootBilNo = resData['BIL_NO'] as String?;
- if (rootBilNo != null && rootBilNo.isNotEmpty) return rootBilNo;
- return null;
- }
- /// 下载附件文件字节
- Future<Uint8List?> downloadAttachment(String id) async {
- return await _client.downloadFile('/OA/DownloadAttachment', queryParameters: {'id': id});
- }
- /// 检测附件服务是否可用
- Future<bool> checkAttachHealth() async {
- try {
- final response = await _client.get<Map<String, dynamic>>('/OA/CheckAttachHealth');
- return response.data?['available'] as bool? ?? false;
- } catch (_) {
- return false;
- }
- }
- /// 财务核销
- Future<void> verify(Map<String, dynamic> data) async {
- await _client.post('/OA/ExpenseVerify', data: data);
- }
- /// 费用类别字典
- Future<List<CostTypeItem>> getCostTypes({String keyword = '', String accNo = ''}) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetCostTypes',
- queryParameters: {'keyword': keyword, 'accNo': accNo, 'page': 1, 'size': 100},
- );
- final list = (response.data?['list'] as List<dynamic>?) ?? [];
- return list.map((e) => CostTypeItem.fromJson(e as Map<String, dynamic>)).toList();
- }
- /// 项目代号
- Future<List<ProjectCodeItem>> getProjectCodes({String keyword = '', String billDate = ''}) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetProjectCodes',
- queryParameters: {'keyword': keyword, 'billDate': billDate, 'page': 1, 'size': 100},
- );
- final list = (response.data?['list'] as List<dynamic>?) ?? [];
- return list.map((e) => ProjectCodeItem.fromJson(e as Map<String, dynamic>)).toList();
- }
- /// 部门
- Future<List<DepartmentItem>> getDepartments({String keyword = '', bool onlyActive = true}) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetDepartments',
- queryParameters: {'keyword': keyword, 'onlyActive': onlyActive, 'page': 1, 'size': 100},
- );
- final list = (response.data?['list'] as List<dynamic>?) ?? [];
- return list.map((e) => DepartmentItem.fromJson(e as Map<String, dynamic>)).toList();
- }
- /// 客户/厂商
- Future<List<CustomerItem>> getCustomers({String keyword = ''}) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetCustomers',
- queryParameters: {'keyword': keyword, 'page': 1, 'size': 100},
- );
- final list = (response.data?['list'] as List<dynamic>?) ?? [];
- return list.map((e) => CustomerItem.fromJson(e as Map<String, dynamic>)).toList();
- }
- /// 员工查询
- Future<List<EmployeeItem>> getEmployees({String keyword = '', String salNo = ''}) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetEmployees',
- queryParameters: {'keyword': keyword, 'salNo': salNo, 'page': 1, 'size': 100},
- );
- final list = (response.data?['list'] as List<dynamic>?) ?? [];
- return list.map((e) => EmployeeItem.fromJson(e as Map<String, dynamic>)).toList();
- }
- /// 可转入报销单的费用申请明细(keyword 模糊匹配单号/事由/部门/申请人/费用类型)
- Future<Map<String, dynamic>> getImportableExpenseApplies({
- String keyword = '', String startDate = '', String endDate = '', int page = 1, int size = 20, String sortDir = 'DESC',
- }) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetImportableExpenseApplies',
- queryParameters: {'keyword': keyword, 'startDate': startDate, 'endDate': endDate, 'page': page, 'size': size, 'sortDir': sortDir},
- );
- return response.data!;
- }
- /// 币别
- Future<List<CurrencyItem>> getCurrencies({String keyword = ''}) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetCurrencies',
- queryParameters: {'keyword': keyword, 'page': 1, 'size': 50},
- );
- final list = (response.data?['list'] as List<dynamic>?) ?? [];
- return list.map((e) => CurrencyItem.fromJson(e as Map<String, dynamic>)).toList();
- }
- /// 币别查询(旧版,保留兼容)
- Future<ApiResponse<Map<String, dynamic>>> fetchCurrencies({
- String keyword = '',
- int page = 1,
- int size = 50,
- }) async {
- return await _client.get('/OA/GetCurrencies',
- queryParameters: {'keyword': keyword, 'page': page, 'size': size});
- }
- /// 审批列表
- Future<Map<String, dynamic>> fetchApprovalList({
- required String bilId,
- required String status,
- String erpCategory = 'MasterService',
- String keyword = '',
- String startDate = '',
- String endDate = '',
- int page = 1,
- int size = 20,
- }) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetApprovalList',
- queryParameters: {
- 'bilId': bilId,
- 'status': status,
- 'erpCategory': erpCategory,
- 'keyword': keyword,
- 'startDate': startDate,
- 'endDate': endDate,
- 'page': page,
- 'size': size,
- },
- );
- return response.data!;
- }
- /// 审批进度
- Future<Map<String, dynamic>> fetchApprovalTimeline(String bilId, String bilNo, {int bilItm = 0}) async {
- final response = await _client.get<Map<String, dynamic>>(
- '/OA/GetApprovalTimeline',
- queryParameters: {'bilId': bilId, 'bilNo': bilNo, 'bilItm': bilItm},
- );
- return response.data!;
- }
- /// 获取单据附件列表
- Future<List<BillAttachment>> getAttachments(String bilId, String bilNo, {int? srcItm}) async {
- final params = <String, dynamic>{
- 'bilId': bilId,
- 'bilNo': bilNo,
- };
- if (srcItm != null) params['srcItm'] = srcItm;
- final response = await _client.get<dynamic>(
- '/OA/GetAttachments',
- queryParameters: params,
- );
- final body = response.data;
- if (body is! Map) return [];
- final result = body['Result'];
- if (result is! Map) return [];
- final documents = result['documents'];
- if (documents is! List) return [];
- return documents
- .whereType<Map<String, dynamic>>()
- .map((e) => BillAttachment.fromJson(e))
- .toList();
- }
- /// 上传附件
- /// [metadata] Map: { BIL_ID, BIL_NO, SRCITM, ITM, TAG, EFF_DD, USR, FILENAME, EXT }
- Future<Map<String, dynamic>> uploadAttachment(
- String filePath,
- Map<String, dynamic> metadata,
- ) async {
- final fileName = (metadata['FILENAME'] as String?) ?? filePath.split('/').last;
- final response = await _client.uploadMultipart<Map<String, dynamic>>(
- '/OA/UploadAttachment',
- files: [await MultipartFile.fromFile(filePath, filename: fileName)],
- extraFields: {'metadata': json.encode(metadata)}, // proper JSON encoding
- );
- return response.data ?? {};
- }
- /// 审核执行(通过/驳回/反审核)
- Future<Map<String, dynamic>> executeApproval({
- required String bilId, required String bilNo, int bilItm = 0,
- required String action, String rem = '', String effDd = '',
- String reason = '', bool isPreToStart = false, int nodeIndex = -1,
- String dataBx = '',
- }) async {
- final response = await _client.post<Map<String, dynamic>>(
- '/OA/ExecuteApproval',
- data: {
- 'bilId': bilId, 'bilNo': bilNo, 'bilItm': bilItm,
- 'action': action, 'rem': rem, 'effDd': effDd,
- 'reason': reason, 'isPreToStart': isPreToStart,
- 'nodeIndex': nodeIndex, 'dataBx': dataBx,
- },
- );
- return response.data!;
- }
- }
- /// 费用报销审批列表项
- class ExpenseApprovalListItem {
- final String bilId;
- final String bilNo;
- final String bilDate;
- final String applicant;
- final String dept;
- final double amount;
- final String remark;
- final String status;
- const ExpenseApprovalListItem({
- required this.bilId,
- required this.bilNo,
- required this.bilDate,
- required this.applicant,
- required this.dept,
- required this.amount,
- required this.remark,
- required this.status,
- });
- factory ExpenseApprovalListItem.fromJson(Map<String, dynamic> json) =>
- ExpenseApprovalListItem(
- bilId: json['bilId'] as String? ?? '',
- bilNo: json['bilNo'] as String? ?? '',
- bilDate: json['bilDate'] as String? ?? '',
- applicant: json['applicant'] as String? ?? '',
- dept: json['dept'] as String? ?? '',
- amount: (json['amount'] as num?)?.toDouble() ?? 0,
- remark: json['remark'] as String? ?? '',
- status: json['status'] as String? ?? 'pending',
- );
- ExpenseModel toExpenseModel() => ExpenseModel(
- id: bilNo,
- expenseNo: bilNo,
- expenseDate:
- bilDate.isNotEmpty ? DateTime.tryParse(bilDate) : null,
- applicantId: applicant,
- applicantName: applicant,
- deptId: dept,
- deptName: dept,
- totalAmount: amount,
- purpose: remark,
- status: status == 'approved' ? 'approved' : 'pending',
- paymentStatus: 'unpaid',
- createTime: DateTime.now(),
- updateTime: DateTime.now(),
- );
- }
- final expenseApprovalListProvider =
- FutureProvider.autoDispose.family<List<ExpenseModel>, String>(
- (ref, status) async {
- ref.watch(expenseRefreshProvider);
- ref.watch(expenseDateStartProvider);
- ref.watch(expenseDateEndProvider);
- ref.watch(expenseKeywordProvider);
- final api = ref.read(expenseApiProvider);
- final dateStart = ref.read(expenseDateStartProvider);
- final dateEnd = ref.read(expenseDateEndProvider);
- final keyword = ref.read(expenseKeywordProvider);
- final response = await api.fetchApprovalList(
- bilId: 'BX',
- keyword: keyword,
- status: status.isEmpty ? '' : status,
- startDate: dateStart != null ? du.DateUtils.formatDate(dateStart) : '',
- endDate: dateEnd != null ? du.DateUtils.formatDate(dateEnd) : '',
- );
- return (response['list'] as List<dynamic>?)?.map((e) => ExpenseApprovalListItem.fromJson(e as Map<String, dynamic>).toExpenseModel()).toList() ?? [];
- },
- );
- /// "我的报销单" 列表(制单人=当前用户,非审批流)
- final expenseMyListProvider =
- FutureProvider.autoDispose.family<List<ExpenseModel>, String>(
- (ref, status) async {
- ref.watch(expenseRefreshProvider);
- ref.watch(expenseDateStartProvider);
- ref.watch(expenseDateEndProvider);
- ref.watch(expenseKeywordProvider);
- final api = ref.read(expenseApiProvider);
- final dateStart = ref.read(expenseDateStartProvider);
- final dateEnd = ref.read(expenseDateEndProvider);
- final keyword = ref.read(expenseKeywordProvider);
- final result = await api.fetchList(
- keyword: keyword,
- startDate: dateStart != null ? du.DateUtils.formatDate(dateStart) : '',
- endDate: dateEnd != null ? du.DateUtils.formatDate(dateEnd) : '',
- usr: HostAppChannel.usr,
- sortDir: ref.watch(expenseSortDirProvider),
- );
- return result.list;
- },
- );
|