diff --git a/lib/customWidget/ItemWidgetFactory.dart b/lib/customWidget/ItemWidgetFactory.dart index 110e556..2540b59 100644 --- a/lib/customWidget/ItemWidgetFactory.dart +++ b/lib/customWidget/ItemWidgetFactory.dart @@ -35,7 +35,8 @@ class ListItemFactory { mainAxisAlignment: MainAxisAlignment.center, children: [ Text( - rightText, + _truncateText(rightText,14), + // rightText, style: TextStyle(fontSize: 15, color: Colors.grey), ), SizedBox(width: 2,), @@ -47,7 +48,14 @@ class ListItemFactory { ], ), ); + } + // 辅助函数:截断文本 + static String _truncateText(String text, int maxLength) { + if (text.length <= maxLength) return text; + return text.substring(0, maxLength) + '...'; + } + ///类型2:上下布局两个文本(自适应高度) static Widget createColumnTextItem({ diff --git a/lib/customWidget/bottom_picker_two.dart b/lib/customWidget/bottom_picker_two.dart new file mode 100644 index 0000000..199bccc --- /dev/null +++ b/lib/customWidget/bottom_picker_two.dart @@ -0,0 +1,83 @@ +import 'package:flutter/material.dart'; +import 'package:flutter/cupertino.dart'; + +/// 通用底部弹窗选择器 +/// Example: +/// ```dart +/// final choice = await BottomPickerTwo.show( +/// context, +/// items: ['选项1', '选项2', '选项3'], +/// itemBuilder: (item) => Text(item, textAlign: TextAlign.center), +/// initialIndex: 1, +/// ); +/// if (choice != null) { +/// // 用户点击确定并选择了 choice +/// } +/// ``` +class BottomPickerTwo { + /// 显示底部选择器弹窗 + /// + /// [items]: 选项列表 + /// [itemBuilder]: 每个选项的展示 Widget + /// [initialIndex]: 初始选中索引 + /// [itemExtent]: 列表行高 + /// [height]: 弹窗总高度 + static Future show( + BuildContext context, { + required List items, + required Widget Function(dynamic item) itemBuilder, + int initialIndex = 0, + double itemExtent = 40.0, + double height = 250, + }) { + // 当前选中项 + T selected = items[initialIndex]; + + return showModalBottomSheet( + context: context, + backgroundColor: Colors.white, + shape: const RoundedRectangleBorder( + borderRadius: BorderRadius.vertical(top: Radius.circular(12)), + ), + builder: (ctx) { + return SizedBox( + height: height, + child: Column( + children: [ + // 按钮行 + Padding( + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + child: Row( + mainAxisAlignment: MainAxisAlignment.spaceBetween, + children: [ + TextButton( + onPressed: () => Navigator.of(ctx).pop(), + child: const Text('取消'), + ), + TextButton( + onPressed: () => Navigator.of(ctx).pop(selected), + child: const Text('确定'), + ), + ], + ), + ), + const Divider(height: 1), + // 滚动选择器 + Expanded( + child: CupertinoPicker( + scrollController: + FixedExtentScrollController(initialItem: initialIndex), + itemExtent: 30, + onSelectedItemChanged: (index) { + selected = items[index]; + }, + children: items.map(itemBuilder).toList(), + ), + ), + ], + ), + ); + }, + ); + } +} diff --git a/lib/customWidget/department_picker_hidden_type.dart b/lib/customWidget/department_picker_hidden_type.dart new file mode 100644 index 0000000..2b82562 --- /dev/null +++ b/lib/customWidget/department_picker_hidden_type.dart @@ -0,0 +1,230 @@ +import 'dart:convert'; +import 'package:flutter/material.dart'; +import 'package:qhd_prevention/customWidget/search_bar_widget.dart'; +import 'package:qhd_prevention/http/ApiService.dart'; +import '../tools/tools.dart'; // 包含 SessionService + +// 数据模型 +class Category { + final String id; + final String name; + final List children; + + Category({ + required this.id, + required this.name, + this.children = const [], + }); + + factory Category.fromJson(Map json) { + return Category( + id: json['id'] as String, + name: json['name'] as String, + children: (json['nodes'] as List) + .map((e) => Category.fromJson(e as Map)) + .toList(), + ); + } +} + +/// 弹窗回调签名:返回选中项的 id 和 name +typedef DeptSelectHiddenTypeCallback = void Function(String id, String name); + +class DepartmentPickerHiddenType extends StatefulWidget { + /// 回调,返回选中部门 id 与 name + final DeptSelectHiddenTypeCallback onSelected; + + const DepartmentPickerHiddenType({Key? key, required this.onSelected}) : super(key: key); + + @override + _DepartmentPickerHiddenTypeState createState() => _DepartmentPickerHiddenTypeState(); +} + +class _DepartmentPickerHiddenTypeState extends State { + String selectedId = ''; + String selectedName = ''; + Set expandedSet = {}; + + List original = []; + List filtered = []; + bool loading = true; + + final TextEditingController _searchController = TextEditingController(); + + @override + void initState() { + super.initState(); + // 初始均为空 + selectedId = ''; + selectedName = ''; + expandedSet = {}; + _searchController.addListener(_onSearchChanged); + _loadData(); + } + + @override + void dispose() { + _searchController.removeListener(_onSearchChanged); + _searchController.dispose(); + super.dispose(); + } + + Future _loadData() async { + try { + List raw; + if (SessionService.instance.departmentHiddenTypeJsonStr?.isNotEmpty ?? false) { + raw = json.decode(SessionService.instance.departmentHiddenTypeJsonStr!) as List; + } else { + final resultLevel = await ApiService. getHiddenLevelsListTwo(); + List levelList = resultLevel['list'] as List; + String parentId=""; + for(int i=0;i; + } + setState(() { + original = raw.map((e) => Category.fromJson(e as Map)).toList(); + filtered = original; + loading = false; + }); + } catch (e) { + setState(() => loading = false); + } + } + + void _onSearchChanged() { + final query = _searchController.text.toLowerCase().trim(); + setState(() { + filtered = query.isEmpty ? original : _filterCategories(original, query); + }); + } + + List _filterCategories(List list, String query) { + List result = []; + for (var cat in list) { + final children = _filterCategories(cat.children, query); + if (cat.name.toLowerCase().contains(query) || children.isNotEmpty) { + result.add(Category(id: cat.id, name: cat.name, children: children)); + } + } + return result; + } + + Widget _buildRow(Category cat, int indent) { + final hasChildren = cat.children.isNotEmpty; + final isExpanded = expandedSet.contains(cat.id); + final isSelected = cat.id == selectedId; + return Column( + children: [ + InkWell( + onTap: () { + setState(() { + if (hasChildren) { + isExpanded ? expandedSet.remove(cat.id) : expandedSet.add(cat.id); + } + selectedId = cat.id; + selectedName = cat.name; + }); + }, + child: Container( + color: Colors.white, + child: Row( + children: [ + SizedBox(width: 16.0 * indent), + SizedBox( + width: 24, + child: hasChildren + ? Icon(isExpanded ? Icons.arrow_drop_down_rounded : Icons.arrow_right_rounded, + size: 35, color: Colors.grey[600]) + : const SizedBox.shrink(), + ), + const SizedBox(width: 5), + Expanded( + child: Padding( + padding: const EdgeInsets.symmetric(vertical: 12), + child: Text(cat.name), + ), + ), + Padding( + padding: const EdgeInsets.symmetric(horizontal: 16), + child: Icon( + isSelected ? Icons.radio_button_checked : Icons.radio_button_unchecked, + color: Colors.green, + ), + ), + ], + ), + ), + ), + if (hasChildren && isExpanded) + ...cat.children.map((c) => _buildRow(c, indent + 1)), + ], + ); + } + + @override + Widget build(BuildContext context) { + return Container( + width: MediaQuery.of(context).size.width, + height: MediaQuery.of(context).size.height * 0.7, + color: Colors.white, + child: Column( + children: [ + Container( + color: Colors.white, + padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + child: Row( + children: [ + GestureDetector( + onTap: () => Navigator.of(context).pop(), + child: const Text('取消', style: TextStyle(fontSize: 16)), + ), + Expanded( + child: Padding( + padding: const EdgeInsets.symmetric(horizontal: 12), + child: SearchBarWidget( + controller: _searchController, + isShowSearchButton: false, + onSearch: (keyboard) { + }, + ), + ), + ), + GestureDetector( + onTap: () { + Navigator.of(context).pop(); + widget.onSelected(selectedId, selectedName); + }, + child: const Text('确定', style: TextStyle(fontSize: 16, color: Colors.green)), + ), + ], + ), + ), + Divider(), + Expanded( + child: loading + ? const Center(child: CircularProgressIndicator()) + : Container( + color: Colors.white, + child: ListView.builder( + itemCount: filtered.length, + itemBuilder: (ctx, idx) => _buildRow(filtered[idx], 0), + ), + ), + ), + ], + ), + ); + } +} diff --git a/lib/http/ApiService.dart b/lib/http/ApiService.dart index bf7c5fe..1b8d4b3 100644 --- a/lib/http/ApiService.dart +++ b/lib/http/ApiService.dart @@ -797,6 +797,30 @@ U6Hzm1ninpWeE+awIDAQAB ); } + /// 隐患整改图片上传 + static Future> addImgFiles(String imagePath,String type,String id) async { + final file = File(imagePath); + if (!await file.exists()) { + throw ApiException('file_not_found', '图片不存在:$imagePath'); + } + final fileName = file.path.split(Platform.pathSeparator).last; + return HttpManager().uploadFaceImage( + baseUrl: basePath, + path: '/app/feedback/upload', + fromData: { + 'FOREIGN_KEY': id, + 'TYPE': type, + "corpUserId":"", + 'CORPINFO_ID': SessionService.instance.corpinfoId, + 'USER_ID': SessionService.instance.loginUserId, + 'FFILE': await MultipartFile.fromFile( + file.path, + filename: fileName + ), + } + ); + } + /// 修改密码 static Future> changePassWord(String oldPwd,String confirmPwd) { return HttpManager().request( @@ -1071,6 +1095,35 @@ U6Hzm1ninpWeE+awIDAQAB }, ); } + + /// 获取隐患等级列表2 + static Future> getHiddenLevelsListTwo() { + return HttpManager().request( + basePath, + '/dictionaries/getLevels', + method: Method.post, + data: { + "DICTIONARIES_ID": '3babc15144444bdc8d763d0af2bdfff8', + "CORPINFO_ID": SessionService.instance.corpinfoId, + "USER_ID": SessionService.instance.loginUserId, + }, + ); + } + + /// 获取隐患类型列表 + static Future> getHiddenTypeList(String parentId) { + return HttpManager().request( + basePath, + '/dictionaries/listAllDictToParId', + method: Method.post, + data: { + "parentId": parentId, + "CORPINFO_ID": SessionService.instance.corpinfoId, + "USER_ID": SessionService.instance.loginUserId, + }, + ); + } + /// 获取部门负责人列表 static Future> getListTreePersonList(String DEPARTMENT_ID) { return HttpManager().request( @@ -1101,6 +1154,20 @@ U6Hzm1ninpWeE+awIDAQAB ); } + /// 获取隐患记录详情第二个接口,不知道为什么 + static Future> getDangerDetailTwo(String id) { + return HttpManager().request( + basePath, + '/app/customHidden/goEdit', + method: Method.post, + data: { + "HIDDEN_ID": id, + "CORPINFO_ID": SessionService.instance.corpinfoId, + "USER_ID": SessionService.instance.loginUserId, + }, + ); + } + /// 获取个人信息 static Future> getUserInfo() { return HttpManager().request( @@ -1242,6 +1309,95 @@ U6Hzm1ninpWeE+awIDAQAB ); } + /// 获取法律法规类表第二层列表 + static Future> getLowListByParentId(String id) { + return HttpManager().request( + basePath, + '/app/mfolderStipulate/listByParentId', + method: Method.post, + data: { + "PARENT_ID":id, + "TYPE":2, + "CORPINFO_ID":SessionService.instance.corpinfoId, + "USER_ID":SessionService.instance.loginUserId, + }, + ); + } + + + /// 提交隐患整改 + static Future> rectificationSubmission(String id,String buMenId,String renYuanId) { + return HttpManager().request( + basePath, + '/app/hidden/editRec', + method: Method.post, + data: { + "RECTIFICATIONDEPT": buMenId, + "RECTIFICATIONOR": renYuanId, + "HIDDEN_ID": id, + "CORPINFO_ID": SessionService.instance.corpinfoId, + "USER_ID": SessionService.instance.loginUserId, + }, + ); + } + + + /// 提交隐患整改 + static Future> normalRectificationSubmission( + String dataTime,String miaoShu,String acceptedPrepareType,String acceptedPlanType, + String id,String caertTime,String listName, + String standard,String method,String fund,String person, + String workTime,String time,String work,String other,String json) { + + // print("============>"+ dataTime+ miaoShu+ acceptedPrepareType+ acceptedPlanType+ + // id+ caertTime+ listName+ standard+ method+ fund+ person+ + // workTime+ time+ work+ other+ json); + + return HttpManager().request( + basePath, + '/app/hidden/rectify', + method: Method.post, + data: { + + "RECTIFICATIONTIME": dataTime, + "RECTIFYDESCR": miaoShu, + "HAVESCHEME": acceptedPrepareType, + "HAVEPLAN": acceptedPlanType, + "HIDDEN_ID": id, + + "SCREENINGDATE": caertTime.isNotEmpty ? caertTime : '', + "LISTNAME": listName.isNotEmpty ? caertTime : '', + "GOVERNSTANDARDS": standard, + "GOVERNMETHOD": method, + "EXPENDITURE": fund, + "PRINCIPAL": person, + "PROGRAMMING": workTime, + "TIMELIMITFOR": time, + "JOBREQUIREMENT": work, + "OTHERBUSINESS": other, + "OTHER": json, + + "CORPINFO_ID": SessionService.instance.corpinfoId, + "USER_ID": SessionService.instance.loginUserId, + }, + ); + } + + + /// 获取隐患级别 + static Future> getHazardLevel() { + return HttpManager().request( + basePath, + '/dictionaries/getLevels?tm=${DateTime.now().millisecondsSinceEpoch}', + method: Method.post, + data: { + "DICTIONARIES_ID": '5e7cf8620ba54ad89719d0be62133c7a', + "CORPINFO_ID": SessionService.instance.corpinfoId, + "USER_ID": SessionService.instance.loginUserId, + }, + ); + } + } diff --git a/lib/pages/app/Danger_paicha/quick_report_page.dart b/lib/pages/app/Danger_paicha/quick_report_page.dart index 9734cf9..f080671 100644 --- a/lib/pages/app/Danger_paicha/quick_report_page.dart +++ b/lib/pages/app/Danger_paicha/quick_report_page.dart @@ -1,9 +1,16 @@ import 'dart:io'; import 'package:flutter/material.dart'; +import 'package:intl/intl.dart'; import 'package:qhd_prevention/customWidget/ItemWidgetFactory.dart'; +import 'package:qhd_prevention/customWidget/bottom_picker.dart'; +import 'package:qhd_prevention/customWidget/bottom_picker_two.dart'; import 'package:qhd_prevention/customWidget/custom_button.dart'; +import 'package:qhd_prevention/customWidget/date_picker_dialog.dart'; +import 'package:qhd_prevention/customWidget/department_picker.dart'; +import 'package:qhd_prevention/customWidget/department_picker_hidden_type.dart'; import 'package:qhd_prevention/pages/my_appbar.dart'; import '../../../customWidget/photo_picker_row.dart'; +import '../../../http/ApiService.dart'; class QuickReportPage extends StatefulWidget { const QuickReportPage({super.key}); @@ -17,13 +24,43 @@ class _QuickReportPageState extends State { final _partController = TextEditingController(); final _dangerDetailController = TextEditingController(); - final String _repairLevel = "请选择"; - final String _repairType = "请选择"; - final String _dangerOrganize = "请选择"; - final String _dangerTime = "请选择"; + String _repairLevel = ""; + late bool _isDanger = false; + late List _hazardLeveLlist = []; //隐患级别 + + + dynamic _hazardLeve; + String yinHuanId = ""; + String yinHuanName = ""; + String buMenId = ""; + String buMenName = ""; + String dataTime = ""; + + @override + void initState() { + // TODO: implement initState + super.initState(); + + _getHazardLevel(); + } + + Future _getHazardLevel() async { + try { + final result = await ApiService.getHazardLevel(); + if (result['result'] == 'success') { + final List newList = result['varList'] ?? []; + setState(() { + _hazardLeveLlist.addAll(newList); + }); + } + } catch (e) { + print('Error fetching data: $e'); + } + } + @override void dispose() { _standardController.dispose(); @@ -41,14 +78,14 @@ class _QuickReportPageState extends State { // 详情滚动区域 _pageDetail(), // 底部警示文字,固定在页面底部 - Container( - padding: const EdgeInsets.all(15), - color: Colors.white, - child: Text( - ' 严禁在本互联网非涉密平台处理、传输国家秘密和工作秘密,请确认扫描、传输的文件资料不涉及国家秘密和工作秘密', - style: TextStyle(fontSize: 14, color: Colors.red), - ), - ), + // Container( + // padding: const EdgeInsets.all(15), + // color: Colors.white, + // child: Text( + // ' 严禁在本互联网非涉密平台处理、传输国家秘密和工作秘密,请确认扫描、传输的文件资料不涉及国家秘密和工作秘密', + // style: TextStyle(fontSize: 14, color: Colors.red), + // ), + // ), ], ), ); @@ -111,18 +148,53 @@ class _QuickReportPageState extends State { _partController, ), ), - _buildSectionContainer( + GestureDetector( + onTap: () async { + final choice = await BottomPickerTwo.show( + context, + items: _hazardLeveLlist, + itemBuilder: (item) => Text(item, textAlign: TextAlign.center), + initialIndex: 0, + ); + if (choice != null) { + setState(() { + _hazardLeve = choice; + _repairLevel=_hazardLeve[""]; + }); + } + }, + child: _buildSectionContainer( child: ListItemFactory.createRowSpaceBetweenItem( leftText: "隐患级别", - rightText: _repairLevel, + rightText: _repairLevel.isNotEmpty?_repairLevel:"请选择", isRight: true, ), ), - _buildSectionContainer( - child: ListItemFactory.createRowSpaceBetweenItem( - leftText: "隐患类型", - rightText: _repairType, - isRight: true, + ), + GestureDetector( + onTap: () { + showModalBottomSheet( + context: context, + isScrollControlled: true, + barrierColor: Colors.black54, + backgroundColor: Colors.transparent, + builder: + (ctx) => DepartmentPickerHiddenType( + onSelected: (id, name) async { + setState(() { + yinHuanId = id; + yinHuanName = name; + }); + }, + ), + ); + }, + child: _buildSectionContainer( + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "隐患类型", + rightText: yinHuanName.isNotEmpty?yinHuanName:"请选择", + isRight: true, + ), ), ), _buildSectionContainer( @@ -169,25 +241,68 @@ class _QuickReportPageState extends State { if (!_isDanger) Column( children: [ - _buildSectionContainer( - child: ListItemFactory.createRowSpaceBetweenItem( - leftText: "整改责任部门", - rightText: _dangerOrganize, - isRight: true, + GestureDetector( + onTap: () { + showModalBottomSheet( + context: context, + isScrollControlled: true, + barrierColor: Colors.black54, + backgroundColor: Colors.transparent, + builder: + (ctx) => DepartmentPicker( + onSelected: (id, name) async { + setState(() { + buMenId = id; + buMenName = name; + }); + }, + ), + ); + }, + child: _buildSectionContainer( + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "整改责任部门", + rightText: buMenName.isNotEmpty ? buMenName : "请选择", + isRight: true, + ), ), ), - _buildSectionContainer( - child: ListItemFactory.createRowSpaceBetweenItem( - leftText: "整改期限", - rightText: _dangerTime, - isRight: true, + GestureDetector( + onTap: () { + showDialog( + context: context, + builder: + (_) => HDatePickerDialog( + initialDate: DateTime.now(), + onCancel: () => Navigator.of(context).pop(), + onConfirm: (selected) { + Navigator.of(context).pop(); + setState(() { + dataTime = DateFormat( + 'yyyy-MM-dd', + ).format(selected); + }); + }, + ), + ); + }, + child: _buildSectionContainer( + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "整改期限", + rightText: dataTime.isNotEmpty ? dataTime : "请选择", + isRight: true, + ), ), ), ], ), SizedBox(height: 30), - CustomButton(text: "提交", backgroundColor: Colors.blue), + CustomButton( + onPressed: () {}, + text: "提交", + backgroundColor: Colors.blue, + ), ], ), ), diff --git a/lib/pages/home/work/ai_alarm_page.dart b/lib/pages/app/ai_alarm_page.dart similarity index 99% rename from lib/pages/home/work/ai_alarm_page.dart rename to lib/pages/app/ai_alarm_page.dart index f77e0d5..2d4315f 100644 --- a/lib/pages/home/work/ai_alarm_page.dart +++ b/lib/pages/app/ai_alarm_page.dart @@ -10,7 +10,7 @@ import 'package:qhd_prevention/pages/my_appbar.dart'; import 'package:qhd_prevention/tools/SmallWidget.dart'; import 'package:qhd_prevention/tools/tools.dart'; -import '../../../http/ApiService.dart'; +import '../../http/ApiService.dart'; class AiAlarmPage extends StatefulWidget { const AiAlarmPage({super.key}); diff --git a/lib/pages/app/application_page.dart b/lib/pages/app/application_page.dart index 8d235e8..4ee8b69 100644 --- a/lib/pages/app/application_page.dart +++ b/lib/pages/app/application_page.dart @@ -1,12 +1,12 @@ import 'package:flutter/material.dart'; import 'package:qhd_prevention/pages/app/Danger_paicha/check_record_page.dart'; import 'package:qhd_prevention/pages/app/Danger_paicha/quick_report_page.dart'; -import 'package:qhd_prevention/pages/home/work/danger_wait_list_page.dart'; +import 'package:qhd_prevention/pages/app/danger_wait_list_page.dart'; import 'package:qhd_prevention/pages/home/work/risk_list_page.dart'; import '../../tools/tools.dart'; import '../home/userInfo_page.dart'; -import '../home/work/ai_alarm_page.dart'; +import 'ai_alarm_page.dart'; import '../home/work/danger_page.dart'; import '../home/work/danger_repair_page.dart'; @@ -93,26 +93,27 @@ class ApplicationPage extends StatelessWidget { // 使用枚举定义项目 final List> buttonInfos = [ { - 'title': '隐患排查', + 'title': '隐患快报', 'list': [ - {'item': AppItem.riskInspection, 'icon': 'assets/icon-apps/icon-zl-6.png', 'title': '隐患排查', 'num': 0}, {'item': AppItem.quickReport, 'icon': 'assets/icon-apps/icon-pc-1.png', 'title': '隐患快报', 'num': 2}, - {'item': AppItem.checkRecord, 'icon': 'assets/icon-apps/icon-zl-2.png', 'title': '检查记录', 'num': 0}, ], }, { - 'title': '专项检查', + 'title': '隐患排查', 'list': [ - {'item': AppItem.specialRectification, 'icon': 'assets/icon-apps/icon-pc-1.png', 'title': '隐患整改', 'num': 5}, - {'item': AppItem.specialRecord, 'icon': 'assets/icon-apps/icon-zl-2.png', 'title': '隐患记录', 'num': 0}, + {'item': AppItem.riskInspection, 'icon': 'assets/icon-apps/icon-zl-6.png', 'title': '隐患排查', 'num': 0}, + {'item': AppItem.checkRecord, 'icon': 'assets/icon-apps/icon-zl-2.png', 'title': '检查记录', 'num': 0}, + + // {'item': AppItem.specialRectification, 'icon': 'assets/icon-apps/icon-pc-1.png', 'title': '隐患整改', 'num': 5}, + // {'item': AppItem.specialRecord, 'icon': 'assets/icon-apps/icon-zl-2.png', 'title': '隐患记录', 'num': 0}, ], }, { 'title': '监管帮扶', 'list': [ - {'item': AppItem.supervisionRectification, 'icon': 'assets/icon-apps/icon-pc-1.png', 'title': '隐患整改', 'num': 2}, - {'item': AppItem.supervisionRecord, 'icon': 'assets/icon-apps/icon-zl-2.png', 'title': '隐患记录', 'num': 0}, + {'item': AppItem.supervisionRectification, 'icon': 'assets/icon-apps/icon-zl-6.png', 'title': '标准排查', 'num': 2}, + {'item': AppItem.supervisionRecord, 'icon': 'assets/icon-apps/icon-zl-2.png', 'title': '检查记录', 'num': 0}, ], }, @@ -131,7 +132,7 @@ class ApplicationPage extends StatelessWidget { 'title': '监测预警', 'list': [ {'item': AppItem.aiAlarm, 'icon': 'assets/icon-apps/icon-pc-1.png', 'title': 'AI报警', 'num': 0}, - // {'item': AppItem.supervisionRecord, 'icon': 'assets/icon-apps/icon-zl-2.png', 'title': '隐患记录', 'num': 0}, + ], }, ]; diff --git a/lib/pages/home/work/danger_wait_list_page.dart b/lib/pages/app/danger_wait_list_page.dart similarity index 91% rename from lib/pages/home/work/danger_wait_list_page.dart rename to lib/pages/app/danger_wait_list_page.dart index 4566221..26624bc 100644 --- a/lib/pages/home/work/danger_wait_list_page.dart +++ b/lib/pages/app/danger_wait_list_page.dart @@ -1,14 +1,15 @@ import 'package:flutter/material.dart'; +import 'package:qhd_prevention/pages/app/pending_rectification_detail_page.dart'; import 'package:qhd_prevention/pages/home/work/custom_driver_drawer.dart'; import 'package:qhd_prevention/pages/home/risk/risk_detail_page.dart'; import 'package:qhd_prevention/pages/home/work/danger_repair_page.dart'; import 'package:qhd_prevention/pages/my_appbar.dart'; import 'package:qhd_prevention/tools/h_colors.dart'; import 'package:qhd_prevention/tools/tools.dart'; -import '../../../http/ApiService.dart'; -import '../../app/application_page.dart'; +import '../../http/ApiService.dart'; +import 'application_page.dart'; import '/customWidget/search_bar_widget.dart'; -import 'danger_wait_deawer.dart'; +import '../home/work/danger_wait_deawer.dart'; import 'hidden_record_detail_page.dart'; enum DangerType { @@ -16,9 +17,9 @@ enum DangerType { ristRecord("隐患记录", "隐患记录-详情"), wait("待整改隐患", "隐患整改"), expired("超期未整改", "超期未整改-详情"), - waitAcceptance("隐患验收", "隐患验收"), + waitAcceptance("隐患验收", "隐患验收-详情"), acceptance("已验收隐患", "已验收隐患"), - acceptanced("已验收隐患", "隐患记录-详情"); + acceptanced("已验收隐患", "已验收隐患-详情"); final String displayName; final String detailTitle; @@ -103,6 +104,7 @@ class _DangerWaitListPageState extends State { appBar: MyAppbar( title: widget.dangerType.displayName, actions: [ + if(1==widget.appItem) TextButton( onPressed: () { // 查询 右侧弹窗页面 @@ -207,15 +209,32 @@ class _DangerWaitListPageState extends State { break; case 2://待整改隐患 // _getDangerRecord(2,_page,"","","","","","","","",keyWord,addList); + // pushPage(PendingRectificationDetailPage(widget.dangerType,item), context); + Navigator.push( + context, + MaterialPageRoute( + builder: (context) => PendingRectificationDetailPage( + widget.dangerType,item, + onClose: (result) { + print('详情页面已关闭,返回结果: $result'); + getListData(widget.appItem,false,""); + SessionService.instance.setDangerWaitInfo(""); + // reRefreshData(); + }, + ), + ), + ); break; case 3://超期未整改 // _getDangerRecord(5,_page,"","","","","-1","","","",keyWord,addList); + pushPage(HiddenRecordDetailPage(widget.dangerType,item), context); + break; case 4://隐患验收 - // _getDangerRecord(3,_page,"","","","","","3","","",keyWord,addList); + pushPage(HiddenRecordDetailPage(widget.dangerType,item), context); break; case 5://已验收隐患 - // _getDangerRecord(4,_page,"","","","","","4","","1",keyWord,addList); + pushPage(HiddenRecordDetailPage(widget.dangerType,item), context); break; } diff --git a/lib/pages/app/danner_repair.dart b/lib/pages/app/danner_repair.dart new file mode 100644 index 0000000..d6c5e4d --- /dev/null +++ b/lib/pages/app/danner_repair.dart @@ -0,0 +1,489 @@ +import 'dart:io'; +import 'package:flutter/material.dart'; +import 'package:intl/intl.dart'; +import 'package:qhd_prevention/customWidget/department_person_picker.dart'; +import 'package:qhd_prevention/customWidget/department_picker.dart'; +import 'package:qhd_prevention/customWidget/toast_util.dart'; +import 'package:qhd_prevention/http/ApiService.dart'; +import '../../customWidget/ItemWidgetFactory.dart'; +import '../../customWidget/custom_button.dart'; +import '../../customWidget/date_picker_dialog.dart'; +import '../../customWidget/photo_picker_row.dart'; +import '../../tools/h_colors.dart'; +import '../../tools/tools.dart'; + + +class DepartmentEntry { + String department; + String responsible; + + String index; + String departmentId; + String responsibleId; + + DepartmentEntry({ + required this.department, + required this.responsible, + + required this.index, + required this.departmentId, + required this.responsibleId, + + + }); + + // 将对象转换为 Map + Map toJson() { + return { + 'DEPARTMENT_ID': departmentId, + 'USER_ID': responsibleId, + 'DEPARTMENT_NAME': department, + 'USER_NAME': responsible, + 'index': index, + }; + } +} + +/// 隐患整改 +class DannerRepair extends StatefulWidget { + DannerRepair(this.pd, {super.key}); + + final Map pd; + + @override + State createState() => DannerRepairState(); +} + +class DannerRepairState extends State { + + // 是否有整改方案 + bool acceptedPrepare = false; + + // 是否有整改计划 + bool acceptedPlan = false; + + final standardController = TextEditingController(); + final methodController = TextEditingController(); + final fundController = TextEditingController(); + final personController = TextEditingController(); + final workTimeController = TextEditingController(); + final timeController = TextEditingController(); + final workController = TextEditingController(); + final otherController = TextEditingController(); + final TextEditingController miaoShuController = TextEditingController(); + late var _selectData = DateTime.now(); + + + + String dataTime=""; + // 整改后图片 + List gaiHouImages = []; + //方案图片 + List fangAnImages = []; + //计划图片 + List jiHuaImages = []; + int yanShouAdd=1; + + + + final List departments = [ + DepartmentEntry(department: '请选择', responsible: '请选择',index:'',departmentId: '',responsibleId:''), + ]; + + void _addDepartment() { + setState(() { + departments.add(DepartmentEntry(department: '请选择', responsible: '请选择',index:'',departmentId: '',responsibleId:'')); + }); + } + + void _removeDepartment(int index) { + if (index == 0) return; // 防止删除第一行 + setState(() { + departments.removeAt(index); + }); + } + + @override + void initState() { + // TODO: implement initState + super.initState(); + // _yanShouFuZeItem.add(_departmentItem(0)); + } + + @override + void dispose() { + // 释放资源 + standardController.dispose(); + methodController.dispose(); + fundController.dispose(); + personController.dispose(); + workTimeController.dispose(); + timeController.dispose(); + workController.dispose(); + otherController.dispose(); + miaoShuController.dispose(); + + super.dispose(); + } + + @override + Widget build(BuildContext context) { + return Container( + padding: EdgeInsets.only(bottom: 10), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(5), + ), + child: Column( + children: [ + ListItemFactory.createBuildSimpleSection("隐患整改"), + Divider(height: 1), + Container( + height: 130, + padding: EdgeInsets.all(15), + child: Column( + children: [ + Row( + children: [HhTextStyleUtils.mainTitle("隐患描述", fontSize: 15)], + ), + TextField( + controller: miaoShuController, + keyboardType: TextInputType.multiline, + maxLines: null, // 不限制行数,输入多少文字就撑开多少行 + style: TextStyle(fontSize: 15), + decoration: InputDecoration( + hintText: '请对隐患进行详细描述(必填项)', + border: InputBorder.none, + ), + ), + ], + ), + ), + Divider(height: 1), + GestureDetector( + onTap: () { + showDialog( + context: context, + builder: + (_) => HDatePickerDialog( + initialDate: DateTime.now(), + onCancel: () => Navigator.of(context).pop(), + onConfirm: (selected) { + Navigator.of(context).pop(); + setState(() { + _selectData = selected; + dataTime= DateFormat('yyyy-MM-dd').format(selected); + + }); + }, + ), + ); + }, + child: Padding( + padding: EdgeInsets.symmetric(horizontal: 15), + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "整改日期", + rightText: dataTime.isEmpty?"请选择":dataTime, + isRight: true, + ), + ), + ), + Divider(), + RepairedPhotoSection( + title: "整改后照片", + maxCount: 4, + mediaType: MediaType.image, + onChanged: (files) { + // 上传 files 到服务器 + gaiHouImages.clear(); + for(int i=0;i _departmentItem( + departments[index], + index, + showLabel: index == 0, + ), + ), + ), + + + // for(int m=0;m<_yanShouFuZeItem.length;m++) + // _yanShouFuZeItem[m], + // _departmentItem(m), + // _departmentItem(2), + Divider(), + ListItemFactory.createYesNoSection( + title: "是否有整改方案", + yesLabel: "是", + noLabel: "否", + groupValue: acceptedPrepare, + onChanged: (val) { + setState(() { + acceptedPrepare = val; + }); + }, + ), + acceptedPrepare ? _acceptPrepare() : SizedBox(height: 1), + + + Divider(), + // 图片上传 + // const SizedBox(height: 16), + if(acceptedPrepare) + RepairedPhotoSection( + horizontalPadding: 0, + title: "方案图片", + maxCount: 4, + mediaType: MediaType.image, + onChanged: (files) { + // 上传 files 到服务器 + fangAnImages.clear(); + for(int i=0;i const Divider(height: 1, color: Colors.black12), + itemBuilder: + (_, index) => Padding( + padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 8), + child: fields[index], + ), + ); + } + + Widget _buildReadOnlyRow(String left, String right) { + return ListItemFactory.createRowSpaceBetweenItem( + leftText: left, + rightText: right, + ); + } + + + /// 验收部门和负责人选择的item + Widget _departmentItem( + DepartmentEntry entry, + int index, { + required bool showLabel, + }) { + return Padding( + padding: const EdgeInsets.all(10), + child: Stack( + clipBehavior: Clip.none, + children: [ + Container( + decoration: BoxDecoration( + border: Border.all(color: Colors.black12, width: 1), + ), + child: _noAccepet_repair(false,index), + ), + + // 当 num > 1 时,左上角显示删除按钮 + if (index > 0) + Positioned( + top: -20, + left: -20, + child: IconButton( + padding: EdgeInsets.zero, + constraints: const BoxConstraints(), + icon: const Icon(Icons.cancel, color: Colors.red, size: 25), + onPressed: () { + _removeDepartment(index); + // 这里处理删除逻辑,比如: + // setState(() => _items.removeAt(num)); + }, + ), + ), + ], + ), + ); + } + + + // 存储各单位的人员列表 + List> _personCache = []; + // #region 不整改 + Widget _noAccepet_repair(bool _accept,int index, ) { + + return Column( + children: [ + GestureDetector( + onTap: () { + showModalBottomSheet( + context: context, + isScrollControlled: true, + barrierColor: Colors.black54, + backgroundColor: Colors.transparent, + builder: (ctx) => DepartmentPicker(onSelected: (id, name) async { + + setState(() { + // buMenId=id; + // buMenName=name; + // + // // 清空已选人员 + // renYuanId=""; + // renYuanName=""; + departments[index].department=name; + departments[index].departmentId=id; + + departments[index].responsible=""; + departments[index].responsibleId=""; + + }); + + // 拉取该单位的人员列表并缓存 + final result = await ApiService.getListTreePersonList(id); + _personCache=List>.from( + result['userList'] as List, + ); + + + }), + ); + }, + child: Container( + padding: EdgeInsets.symmetric(horizontal: 10), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(5), + ), + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "整改部门", + rightText: departments[index].department.isNotEmpty?departments[index].department:"请选择", + isRight: true, + ), + ), + ), + + Divider( + height: 10, + color: _accept ? h_backGroundColor() : Colors.transparent, + ), + + GestureDetector( + onTap: () { + if ( departments[index].departmentId.isEmpty) { + ToastUtil.showNormal(context, '请先选择部门'); + return; + } + DepartmentPersonPicker.show( + context, + personsData: _personCache, + onSelected: (userId, name) { + setState(() { + // renYuanId = userId; + // renYuanName = name; + + departments[index].responsible=name; + departments[index].responsibleId=userId; + departments[index].index=(index-1).toString(); + }); + + }, + ); + }, + child:Container( + padding: EdgeInsets.symmetric(horizontal: 10), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(5), + ), + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "整改负责人", + rightText: departments[index].responsible.isNotEmpty?departments[index].responsible:"请选择", + isRight: true, + ), + ), + ), + ], + ); + } + + /// 整改计划 + Widget _acceptPlan() { + return Padding( + padding: EdgeInsets.symmetric(horizontal: 10), + child: MediaPickerRow( + maxCount: 4, + onChanged: (List files) { + // images 列表更新 + // 上传 files 到服务器 + jiHuaImages.clear(); + for(int i=0;i { @override void initState() { super.initState(); - getData(); + if("2"==widget.item['HIDDEN_RISKSTANDARD']){ + getDataTwo(); + }else { + getData(); + } } @override @@ -85,7 +91,40 @@ class _HiddenRecordDetailPageState extends State { } } + Future getDataTwo() async { + try { + final data = await ApiService.getDangerDetailTwo(widget.item['HIDDEN_ID']); + if (data['result'] == 'success') { + setState(() { + pd = data['pd']; + hs = data['hs'] ?? {}; + + // 处理图片和视频 + for (var img in data['hImgs']) { + if (img['FILEPATH'].toString().endsWith('.mp4')) { + videoList.add(img); + } else { + files.add(img["FILEPATH"]); + } + } + + // List filesZheng = data['rImgs'] ?? []; + for (var img in data['rImgs']) { + files2.add(img["FILEPATH"]); + } + // files2=data['rImgs'] ?? []; + files4 = data['sImgs'] ?? []; + files5 = data['pImgs'] ?? []; + files6 = data['yImgs'] ?? []; + checkList = data['checkList'] ?? []; + }); + } + + } catch (e) { + print('Error fetching data: $e'); + } + } @@ -111,7 +150,7 @@ class _HiddenRecordDetailPageState extends State { @override Widget build(BuildContext context) { return Scaffold( - appBar: MyAppbar(title: "隐患记录-详情"), + appBar: MyAppbar(title: widget.dangerType.detailTitle), body: pd.isEmpty ? const Center(child: CircularProgressIndicator()) : LayoutBuilder( @@ -184,10 +223,16 @@ class _HiddenRecordDetailPageState extends State { const Text('隐患视频', style: TextStyle(fontWeight: FontWeight.bold)), GestureDetector( onTap: () { - present( - BigVideoViewer(videoUrl:ApiService.baseImgPath + videoList[0]['FILEPATH']), - context, + showDialog( + context: context, + barrierColor: Colors.black54, + builder: (_) => VideoPlayerPopup(videoUrl: ApiService.baseImgPath + videoList[0]['FILEPATH']), ); + + // present( + // BigVideoViewer(videoUrl:ApiService.baseImgPath + videoList[0]['FILEPATH']), + // context, + // ); }, // => _playVideo(ApiService.baseImgPath + videoList[0]['FILEPATH']), child: Image.asset( diff --git a/lib/pages/app/pending_rectification_detail_page.dart b/lib/pages/app/pending_rectification_detail_page.dart new file mode 100644 index 0000000..af233ee --- /dev/null +++ b/lib/pages/app/pending_rectification_detail_page.dart @@ -0,0 +1,707 @@ +import 'package:flutter/material.dart'; +import 'package:http/http.dart' as http; +import 'package:qhd_prevention/customWidget/big_video_viewer.dart'; +import 'package:qhd_prevention/pages/my_appbar.dart'; +import 'dart:convert'; +import 'package:video_player/video_player.dart'; + +import '../../customWidget/ItemWidgetFactory.dart'; +import '../../customWidget/custom_button.dart'; +import '../../customWidget/department_person_picker.dart'; +import '../../customWidget/department_picker.dart'; +import '../../customWidget/full_screen_video_page.dart'; +import '../../customWidget/single_image_viewer.dart'; +import '../../customWidget/toast_util.dart'; +import '../../customWidget/video_player_widget.dart'; +import '../../http/ApiService.dart'; +import '../../tools/h_colors.dart'; +import '../../tools/tools.dart'; +import 'danner_repair.dart'; +import 'danger_wait_list_page.dart'; + + +class PendingRectificationDetailPage extends StatefulWidget { + const PendingRectificationDetailPage(this.dangerType, this.item, {super.key,required this.onClose}); + + final Function(String) onClose; // 回调函数 + final DangerType dangerType; + final item; + + @override + _PendingRectificationDetailPageState createState() => _PendingRectificationDetailPageState(); +} + +class _PendingRectificationDetailPageState extends State { + late Map pd = {}; + late Map hs = {}; + List files = []; + List files2 = []; + List files4 = []; + List files5 = []; + List files6 = []; + List videoList = []; + List checkList = []; + + bool modalShow = false; + String videoSrc = ""; + VideoPlayerController? _videoController; + + // 是否整改 + bool _accepted = true; + + String buMenId=""; + String buMenName=""; + String renYuanId=""; + String renYuanName=""; + // 存储各单位的人员列表 + late List> _personCache = []; + + late DannerRepair dannerRepair; + // 在父组件中 + final GlobalKey dannerRepairKey = GlobalKey(); + + @override + void initState() { + super.initState(); + if("2"==widget.item['HIDDEN_RISKSTANDARD']){ + getDataTwo(); + }else { + getData(); + } + } + + @override + void dispose() { + _videoController?.dispose(); + super.dispose(); + } + + Future getData() async { + try { + final data = await ApiService.getDangerDetail(widget.item['HIDDEN_ID']); + if (data['result'] == 'success') { + + setState(() { + pd = data['pd']; + + buMenId=pd["RECTIFICATIONDEPT"]; + buMenName=pd["RECTIFICATIONDEPTNAME"]; + + renYuanId=pd["RECTIFICATIONOR"]; + renYuanName=pd["RECTIFICATIONORNAME"]; + + hs = data['hs'] ?? {}; + + // 处理图片和视频 + for (var img in data['hImgs']) { + if (img['FILEPATH'].toString().endsWith('.mp4')) { + videoList.add(img); + } else { + files.add(img["FILEPATH"]); + } + } + + // List filesZheng = data['rImgs'] ?? []; + for (var img in data['rImgs']) { + files2.add(img["FILEPATH"]); + } + // files2=data['rImgs'] ?? []; + files4 = data['sImgs'] ?? []; + files5 = data['pImgs'] ?? []; + files6 = data['yImgs'] ?? []; + checkList = data['checkList'] ?? []; + }); + } + + } catch (e) { + print('Error fetching data: $e'); + } + } + + Future getDataTwo() async { + try { + final data = await ApiService.getDangerDetailTwo(widget.item['HIDDEN_ID']); + if (data['result'] == 'success') { + + setState(() { + pd = data['pd']; + + buMenId=pd["RECTIFICATIONDEPT"]; + buMenName=pd["RECTIFICATIONDEPTNAME"]; + + renYuanId=pd["RECTIFICATIONOR"]; + renYuanName=pd["RECTIFICATIONORNAME"]; + + hs = data['hs'] ?? {}; + + // 处理图片和视频 + for (var img in data['hImgs']) { + if (img['FILEPATH'].toString().endsWith('.mp4')) { + videoList.add(img); + } else { + files.add(img["FILEPATH"]); + } + } + + // List filesZheng = data['rImgs'] ?? []; + for (var img in data['rImgs']) { + files2.add(img["FILEPATH"]); + } + // files2=data['rImgs'] ?? []; + files4 = data['sImgs'] ?? []; + files5 = data['pImgs'] ?? []; + files6 = data['yImgs'] ?? []; + checkList = data['checkList'] ?? []; + }); + } + + } catch (e) { + print('Error fetching data: $e'); + } + } + + + + Widget _buildInfoItem(String title, String value) { + return Padding( + padding: const EdgeInsets.symmetric(vertical: 8), + child: Row( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + SizedBox( + width: 120, + child: Text( + title, + style: const TextStyle(fontWeight: FontWeight.bold), + ), + ), + Expanded(child: Text(value,textAlign: TextAlign.right,)), + ], + ), + ); + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: MyAppbar(title: widget.dangerType.detailTitle), + body: pd.isEmpty + ? const Center(child: CircularProgressIndicator()) + : LayoutBuilder( + builder: (context, constraints) { + return SingleChildScrollView( + child: ConstrainedBox( + constraints: BoxConstraints( + minHeight: constraints.maxHeight, + ), + child: Padding( + padding: const EdgeInsets.all(16), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + _buildInfoItem('隐患描述', pd['HIDDENDESCR'] ?? ''), + + Divider(height: 1), + // 隐患来源 + _buildInfoItem('隐患来源', _getSourceText(pd['SOURCE'])), + Divider(height: 1), + // 条件渲染部分 + if (pd['SOURCE'] == '2') ...[ + _buildInfoItem('风险点(单元)', pd['RISK_UNIT'] ?? ''), + Divider(height: 1), + _buildInfoItem('辨识部位', pd['IDENTIFICATION'] ?? ''), + Divider(height: 1), + _buildInfoItem('存在风险', pd['RISK_DESCR'] ?? ''), + Divider(height: 1), + _buildInfoItem('风险分级', pd['LEVEL'] ?? ''), + Divider(height: 1), + _buildInfoItem('检查内容', pd['CHECK_CONTENT'] ?? ''), + Divider(height: 1), + ], + + _buildInfoItem('隐患部位', pd['HIDDENPART'] ?? ''), + Divider(height: 1), + _buildInfoItem('发现人', pd['CREATORNAME'] ?? ''), + Divider(height: 1), + _buildInfoItem('发现时间', pd['CREATTIME'] ?? ''), + Divider(height: 1), + + if (pd['HIDDEN_CATEGORY']?.isNotEmpty == true) + _buildInfoItem('隐患类别', pd['HIDDEN_CATEGORY_NAME'] ?? ''), + + _buildInfoItem('隐患类型', pd['HIDDENTYPE_NAME'] ?? ''), + Divider(height: 1), + _buildInfoItem('整改类型', _getRectificationType(pd['RECTIFICATIONTYPE'])), + + if (pd['RECTIFICATIONTYPE'] == '2') + _buildInfoItem('整改期限', pd['RECTIFICATIONDEADLINE'] ?? ''), + Divider(height: 1), + // 隐患照片 + // const Text('隐患照片', style: TextStyle(fontWeight: FontWeight.bold)), + // _buildImageGrid(files, onTap: (index) => _showImageGallery(files, index)), + ListItemFactory.createTextImageItem( + text: "隐患照片", + imageUrls: files, + onImageTapped: (index) { + present( + SingleImageViewer(imageUrl:ApiService.baseImgPath + files[index]), + context, + ); + }, + ), + + + // 隐患视频 + if (videoList.isNotEmpty) ...[ + const SizedBox(height: 16), + const Text('隐患视频', style: TextStyle(fontWeight: FontWeight.bold)), + GestureDetector( + onTap: () { + + showDialog( + context: context, + barrierColor: Colors.black54, + builder: (_) => VideoPlayerPopup(videoUrl: ApiService.baseImgPath + videoList[0]['FILEPATH']), + ); + // present( + // BigVideoViewer(videoUrl:ApiService.baseImgPath + videoList[0]['FILEPATH']), + // context, + // ); + }, + // => _playVideo(ApiService.baseImgPath + videoList[0]['FILEPATH']), + child: Image.asset( + 'assets/image/videostart.png', // 替换为你的视频占位图 + color: Colors.blue, + width: 120, + height: 120, + ), + ), + ], + + SizedBox(height: 10,), + // 整改信息部分 + // if (pd['STATE'] != null && int.parse(pd['STATE']) >= 2 && int.parse(pd['STATE']) <= 4) ...[ + // // const Divider(height: 10,color: Colors.grey,), + // const Text('整改信息', style: TextStyle(fontSize: 18, fontWeight: FontWeight.bold)), + // Divider(height: 1), + // _buildInfoItem('整改描述', pd['RECTIFYDESCR'] ?? ''), + // Divider(height: 1), + // _buildInfoItem('整改部门', pd['RECTIFICATIONDEPTNAME'] ?? ''), + // Divider(height: 1), + // _buildInfoItem('整改人', pd['RECTIFICATIONORNAME'] ?? ''), + // Divider(height: 1), + // _buildInfoItem('整改时间', pd['RECTIFICATIONTIME'] ?? ''), + // Divider(height: 1), + // // const Text('整改后图片', style: TextStyle(fontWeight: FontWeight.bold)), + // // _buildImageGrid(files2, onTap: (index) => _showImageGallery(files2, index)), + // ListItemFactory.createTextImageItem( + // text: "整改后图片", + // imageUrls: files2, + // onImageTapped: (index) { + // present( + // SingleImageViewer(imageUrl: ApiService.baseImgPath +files2[index]), + // context, + // ); + // }, + // ), + // + // // ... 其他整改信息字段 + // ], + + // 添加底部安全区域间距 + + // 隐患整改 + _danner_type_wait(), + + SizedBox(height: MediaQuery.of(context).padding.bottom + 20), + ], + ), + ), + ), + ); + }, + ), + ); + } + + /// 隐患整改 + Widget _danner_type_wait() { + return SizedBox( + child: Column( + children: [ + ListItemFactory.createYesNoSection( + horizontalPadding: 0, + + title: '是否正常整改', + yesLabel: '是', + noLabel: '否', + groupValue: _accepted, + onChanged: (val) { + setState(() { + _accepted = val; + }); + }, + ), + + + // 整改选项 + _accepted ? _getRepairState() : _noAccepet_repair(_accepted), + + const SizedBox(height: 20), + CustomButton( + text: "提交", + backgroundColor: Colors.blue, + onPressed: () { + // ToastUtil + //接口请求 + // _submitToServer(); + _accepted ? _normalRectificationSubmission() : _rectificationSubmission(); + + }, + ), + ], + ), + ); + } + + + Widget _getRepairState() { + dannerRepair= DannerRepair(pd,key: dannerRepairKey, ); + return dannerRepair; + } + + // #region 不整改 + Widget _noAccepet_repair(bool _accept) { + return Column( + children: [ + GestureDetector( + onTap: () { + showModalBottomSheet( + context: context, + isScrollControlled: true, + barrierColor: Colors.black54, + backgroundColor: Colors.transparent, + builder: (ctx) => DepartmentPicker(onSelected: (id, name) async { + + setState(() { + buMenId=id; + buMenName=name; + + // 清空已选人员 + renYuanId=""; + renYuanName=""; + + }); + // 拉取该单位的人员列表并缓存 + final result = await ApiService.getListTreePersonList(id); + _personCache=List>.from( + result['userList'] as List, + ); + + + }), + ); + }, + child: Container( + padding: EdgeInsets.symmetric(horizontal: 10), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(5), + ), + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "整改部门", + rightText: buMenName, + isRight: true, + ), + ), + ), + Divider( + height: 10, + color: _accept ? h_backGroundColor() : Colors.transparent, + ), + GestureDetector( + onTap: () { + if ( renYuanId.isEmpty) { + ToastUtil.showNormal(context, '请先选择部门'); + return; + } + DepartmentPersonPicker.show( + context, + personsData: _personCache, + onSelected: (userId, name) { + setState(() { + renYuanId = userId; + renYuanName = name; + }); + }, + ); + + }, + child: Container( + padding: EdgeInsets.symmetric(horizontal: 10), + decoration: BoxDecoration( + color: Colors.white, + borderRadius: BorderRadius.circular(5), + ), + child: ListItemFactory.createRowSpaceBetweenItem( + leftText: "整改负责人", + rightText: renYuanName, + isRight: true, + ), + ), + ), + ], + ); + } + + + Future _normalRectificationSubmission() async { + try { + + + String dataTime= dannerRepairKey.currentState!.dataTime; + if(dataTime.isEmpty){ + ToastUtil.showNormal(context, "请选择整改时间"); + return; + } + + String miaoShu= dannerRepairKey.currentState!.miaoShuController.text.trim(); + if(miaoShu.isEmpty){ + ToastUtil.showNormal(context, "请填整改描述"); + return; + } + + + List gaiHouImages = dannerRepairKey.currentState!.gaiHouImages; + if(gaiHouImages.isEmpty){ + ToastUtil.showNormal(context, "请上传整改后照片"); + return; + } + + String acceptedPrepareType="0"; + String standard=""; + String method=""; + String fund=""; + String person=""; + String workTime=""; + String time=""; + String work=""; + String other=""; + List fangAnImages =[]; + // 是否有整改方案 + bool acceptedPrepare= dannerRepairKey.currentState!.acceptedPrepare; + if(acceptedPrepare){ + acceptedPrepareType="1"; + + standard= dannerRepairKey.currentState!.standardController.text; + if(standard.isEmpty){ + ToastUtil.showNormal(context, "请输入治理标准要求"); + return; + } + + method= dannerRepairKey.currentState!.methodController.text; + if(method.isEmpty){ + ToastUtil.showNormal(context, "请输入治理方法"); + return; + } + + fund= dannerRepairKey.currentState!.fundController.text; + if(fund.isEmpty){ + ToastUtil.showNormal(context, "请输入经费和物资的落实"); + return; + } + + person= dannerRepairKey.currentState!.personController.text; + if(person.isEmpty){ + ToastUtil.showNormal(context, "请输入负责治理人员"); + return; + } + + workTime= dannerRepairKey.currentState!.workTimeController.text; + if(workTime.isEmpty){ + ToastUtil.showNormal(context, "请输入工时安排"); + return; + } + + time= dannerRepairKey.currentState!.timeController.text; + if(time.isEmpty){ + ToastUtil.showNormal(context, "请输入时限要求"); + return; + } + + work= dannerRepairKey.currentState!.workController.text; + if(work.isEmpty){ + ToastUtil.showNormal(context, "请输入工作要求"); + return; + } + + other= dannerRepairKey.currentState!.otherController.text; + if(other.isEmpty){ + ToastUtil.showNormal(context, "请输入工作要求"); + return; + } + + fangAnImages = dannerRepairKey.currentState!.fangAnImages; + if(fangAnImages.isEmpty){ + ToastUtil.showNormal(context, "请上传方案照片"); + return; + } + + } + + String acceptedPlanType="0"; + List jiHuaImages =[]; + // 是否有整改计划 + bool acceptedPlan= dannerRepairKey.currentState!.acceptedPlan; + if(acceptedPlan){ + acceptedPlanType="1"; + jiHuaImages = dannerRepairKey.currentState!.jiHuaImages; + if(jiHuaImages.isEmpty){ + ToastUtil.showNormal(context, "请上传计划照片"); + return; + } + } + + + + List departments = dannerRepairKey.currentState!.departments; + bool departmentsAllInput=true; + bool departmentsSameMan=false; + for(int i=0;i _rectificationSubmission() async { + try { + + final result = await ApiService.rectificationSubmission(pd["HIDDEN_ID"],buMenId,renYuanId); + if (result['result'] == 'success') { + setState(() { + + ToastUtil.showNormal(context, "提交成功"); + Navigator.of(context).pop(); + widget.onClose('关闭详情'); // 触发回调 + }); + + }else{ + ToastUtil.showNormal(context, "加载数据失败"); + } + } catch (e) { + // 出错时可以 Toast 或者在页面上显示错误状态 + print('加载数据失败:$e'); + } + } + + String _convertDepartmentsToJson(List departments) { + // 1. 将列表中的每个对象转换为 Map + List> jsonList = departments + .map((dept) => dept.toJson()) + .toList(); + + // 2. 使用 jsonEncode 转换为 JSON 字符串 + return jsonEncode(jsonList); + } + + + Future _addImgFiles(String imagePath,String type,String id) async { + try { + + final raw = await ApiService.addImgFiles( imagePath, type, id); + if (raw['result'] == 'success') { + return raw['imgPath']; + }else{ + // _showMessage('反馈提交失败'); + return ""; + } + + } catch (e) { + // 出错时可以 Toast 或者在页面上显示错误状态 + print('加载首页数据失败:$e'); + return ""; + } + } + + String _getSourceText(String? source) { + switch (source) { + case '1': return '隐患快报'; + case '2': return '隐患排查清单检查'; + case '3': return '标准排查清单检查'; + case '4': return '专项检查'; + case '5': return '安全检查'; + default: return ''; + } + } + + String _getRectificationType(String? type) { + switch (type) { + case '1': return '立即整改'; + case '2': return '限期整改'; + default: return ''; + } + } +} + + + diff --git a/lib/pages/home/home_page.dart b/lib/pages/home/home_page.dart index 4fdb7fd..aada2ee 100644 --- a/lib/pages/home/home_page.dart +++ b/lib/pages/home/home_page.dart @@ -8,7 +8,7 @@ import 'package:qhd_prevention/pages/home/study/study_garden_page.dart'; import 'package:qhd_prevention/pages/home/tap/tabList/work_tab_list_page.dart'; import 'package:qhd_prevention/pages/home/userInfo_page.dart'; import 'package:qhd_prevention/pages/home/work/danger_page.dart'; -import 'package:qhd_prevention/pages/home/work/danger_wait_list_page.dart'; +import 'package:qhd_prevention/pages/app/danger_wait_list_page.dart'; import 'package:qhd_prevention/pages/home/work/laws_regulations_page.dart'; import 'package:qhd_prevention/pages/home/workSet_page.dart'; diff --git a/lib/pages/home/work/dangerTypeItems/wait/danner_repair.dart b/lib/pages/home/work/dangerTypeItems/wait/danner_repair.dart deleted file mode 100644 index b23251d..0000000 --- a/lib/pages/home/work/dangerTypeItems/wait/danner_repair.dart +++ /dev/null @@ -1,285 +0,0 @@ -import 'dart:io'; -import 'package:flutter/material.dart'; -import '../../../../../customWidget/ItemWidgetFactory.dart'; -import '../../../../../customWidget/custom_button.dart'; -import '../../../../../customWidget/date_picker_dialog.dart'; -import '../../../../../customWidget/photo_picker_row.dart'; -import '../../../../../tools/h_colors.dart'; -import '../../../../../tools/tools.dart'; - -/// 隐患整改 -class DannerRepair extends StatefulWidget { - const DannerRepair({super.key}); - - @override - State createState() => _DannerRepairState(); -} - -class _DannerRepairState extends State { - - // 是否有整改方案 - bool _acceptedPrepare = false; - - // 是否有整改计划 - bool _acceptedPlan = false; - final _standardController = TextEditingController(); - final _methodController = TextEditingController(); - final _fundController = TextEditingController(); - final _personController = TextEditingController(); - final _workTimeController = TextEditingController(); - final _timeController = TextEditingController(); - final _workController = TextEditingController(); - final _otherController = TextEditingController(); - - late var _selectData = DateTime.now(); - - @override - void dispose() { - // 释放资源 - _standardController.dispose(); - _methodController.dispose(); - _fundController.dispose(); - _personController.dispose(); - _workTimeController.dispose(); - _timeController.dispose(); - _workController.dispose(); - _otherController.dispose(); - - super.dispose(); - } - - @override - Widget build(BuildContext context) { - return Container( - padding: EdgeInsets.only(bottom: 10), - decoration: BoxDecoration( - color: Colors.white, - borderRadius: BorderRadius.circular(5), - ), - child: Column( - children: [ - ListItemFactory.createBuildSimpleSection("隐患整改"), - Divider(height: 1), - Container( - height: 130, - padding: EdgeInsets.all(15), - child: Column( - children: [ - Row( - children: [HhTextStyleUtils.mainTitle("隐患描述", fontSize: 15)], - ), - TextField( - keyboardType: TextInputType.multiline, - maxLines: null, // 不限制行数,输入多少文字就撑开多少行 - style: TextStyle(fontSize: 15), - decoration: InputDecoration( - hintText: '请对隐患进行详细描述(必填项)', - border: InputBorder.none, - ), - ), - ], - ), - ), - Divider(height: 1), - GestureDetector( - onTap: () { - showDialog( - context: context, - builder: - (_) => HDatePickerDialog( - initialDate: DateTime.now(), - onCancel: () => Navigator.of(context).pop(), - onConfirm: (selected) { - Navigator.of(context).pop(); - setState(() { - _selectData = selected; - }); - }, - ), - ); - }, - child: Padding( - padding: EdgeInsets.symmetric(horizontal: 15), - child: ListItemFactory.createRowSpaceBetweenItem( - leftText: "整改日期", - rightText: "请选择", - isRight: true, - ), - ), - ), - Divider(), - RepairedPhotoSection( - title: "整改后照片", - maxCount: 4, - mediaType: MediaType.image, - onChanged: (files) { - // 上传 files 到服务器 - }, - onAiIdentify: () {}, - ), - - Divider(), - Row( - mainAxisAlignment: MainAxisAlignment.end, - children: [ - CustomButton( - onPressed: () {}, - text: "添加", - backgroundColor: Colors.blue, - borderRadius: 17, - height: 34, - padding: EdgeInsets.symmetric(horizontal: 20), - ), - ], - ), - _departmentItem(1), - _departmentItem(2), - Divider(), - ListItemFactory.createYesNoSection( - title: "是否有整改方案", - yesLabel: "是", - noLabel: "否", - groupValue: _acceptedPrepare, - onChanged: (val) { - setState(() { - _acceptedPrepare = val; - }); - }, - ), - _acceptedPrepare ? _acceptPrepare() : SizedBox(height: 1), - Divider(), - ListItemFactory.createYesNoSection( - title: "是否有整改计划", - yesLabel: "是", - noLabel: "否", - groupValue: _acceptedPlan, - onChanged: (val) { - setState(() { - _acceptedPlan = val; - }); - }, - ), - _acceptedPlan ? _acceptPlan() : SizedBox(height: 1), - ], - ), - ); - } - /// 整改方案 - Widget _acceptPrepare() { - final fields = [ - _buildReadOnlyRow("排查日期", "2025-1-2 11:22:30"), - _buildReadOnlyRow("隐患清单", "-----"), - ListItemFactory.createBuildMultilineInput("治理标准", "请输入治理标准", _standardController), - ListItemFactory.createBuildMultilineInput("治理方法", "请输入治理方法", _methodController), - ListItemFactory.createBuildMultilineInput("经费落实", "请输入经费落实", _fundController), - ListItemFactory.createBuildMultilineInput("负责人员", "请输入负责人员", _personController), - ListItemFactory.createBuildMultilineInput("工时安排", "请输入工时安排", _workTimeController), - ListItemFactory.createBuildMultilineInput("时限要求", "请输入时限要求", _timeController), - ListItemFactory.createBuildMultilineInput("工作要求", "请输入工作要求", _workController), - ListItemFactory.createBuildMultilineInput("其他事项", "请输入其他事项", _otherController), - ]; - - return ListView.separated( - shrinkWrap: true, - physics: const NeverScrollableScrollPhysics(), - itemCount: fields.length, - separatorBuilder: - (_, __) => const Divider(height: 1, color: Colors.black12), - itemBuilder: - (_, index) => Padding( - padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 8), - child: fields[index], - ), - ); - } - - Widget _buildReadOnlyRow(String left, String right) { - return ListItemFactory.createRowSpaceBetweenItem( - leftText: left, - rightText: right, - ); - } - - - /// 验收部门和负责人选择的item - Widget _departmentItem(int num) { - return Padding( - padding: const EdgeInsets.all(10), - child: Stack( - clipBehavior: Clip.none, - children: [ - Container( - decoration: BoxDecoration( - border: Border.all(color: Colors.black12, width: 1), - ), - child: _noAccepet_repair(false), - ), - - // 当 num > 1 时,左上角显示删除按钮 - if (num > 1) - Positioned( - top: -20, - left: -20, - child: IconButton( - padding: EdgeInsets.zero, - constraints: const BoxConstraints(), - icon: const Icon(Icons.cancel, color: Colors.red, size: 25), - onPressed: () { - // 这里处理删除逻辑,比如: - // setState(() => _items.removeAt(num)); - }, - ), - ), - ], - ), - ); - } - // #region 不整改 - Widget _noAccepet_repair(bool _accept) { - return Column( - children: [ - Container( - padding: EdgeInsets.symmetric(horizontal: 10), - decoration: BoxDecoration( - color: Colors.white, - borderRadius: BorderRadius.circular(5), - ), - child: ListItemFactory.createRowSpaceBetweenItem( - leftText: "整改部门", - rightText: "测试啊", - isRight: true, - ), - ), - Divider( - height: 10, - color: _accept ? h_backGroundColor() : Colors.transparent, - ), - Container( - padding: EdgeInsets.symmetric(horizontal: 10), - decoration: BoxDecoration( - color: Colors.white, - borderRadius: BorderRadius.circular(5), - ), - child: ListItemFactory.createRowSpaceBetweenItem( - leftText: "整改负责人", - rightText: "测试啊", - isRight: true, - ), - ), - ], - ); - } - - /// 整改计划 - Widget _acceptPlan() { - return Padding( - padding: EdgeInsets.symmetric(horizontal: 10), - child: MediaPickerRow( - maxCount: 4, - onChanged: (List images) { - // images 列表更新 - }, - ), - ); - } -} diff --git a/lib/pages/home/work/danger_repair_page.dart b/lib/pages/home/work/danger_repair_page.dart index 1bcb15c..013baab 100644 --- a/lib/pages/home/work/danger_repair_page.dart +++ b/lib/pages/home/work/danger_repair_page.dart @@ -3,8 +3,8 @@ import 'package:qhd_prevention/customWidget/custom_button.dart'; import 'package:qhd_prevention/pages/home/work/dangerTypeItems/danger_detail.dart'; import 'package:qhd_prevention/pages/home/work/dangerTypeItems/finish/danger_acceptance_finish.dart'; import 'package:qhd_prevention/pages/home/work/dangerTypeItems/wait/danger_acceptance.dart'; -import 'package:qhd_prevention/pages/home/work/dangerTypeItems/wait/danner_repair.dart'; -import 'package:qhd_prevention/pages/home/work/danger_wait_list_page.dart'; +import 'package:qhd_prevention/pages/app/danner_repair.dart'; +import 'package:qhd_prevention/pages/app/danger_wait_list_page.dart'; import 'package:qhd_prevention/pages/my_appbar.dart'; import 'package:qhd_prevention/tools/h_colors.dart'; import '../../../customWidget/ItemWidgetFactory.dart'; @@ -130,7 +130,7 @@ class _DangerRepairPageState extends State { }, ), // 整改选项 - _accepted ? DannerRepair() : _noAccepet_repair(_accepted), + // _accepted ? DannerRepair() : _noAccepet_repair(_accepted), const SizedBox(height: 20), CustomButton( @@ -230,63 +230,75 @@ class _DangerRepairPageState extends State { children: [ ListItemFactory.createRowSpaceBetweenItem( leftText: "隐患描述", - rightText: pd["HIDDENDESCR"], + rightText: "", + // rightText: pd["HIDDENDESCR"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "隐患来源", - rightText: _getSourceDangers(pd), + rightText: "", + // rightText: _getSourceDangers(pd), ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "风险点(单元)", - rightText: pd["RISK_UNIT"], + rightText: "", + // rightText: pd["RISK_UNIT"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "辨识部位", - rightText: pd["IDENTIFICATION"], + rightText: "", + // rightText: pd["IDENTIFICATION"]??"", ), Divider(height: 1), ListItemFactory.createColumnTextItem( topText: "存在风险", - bottomText: pd["RISK_DESCR"], + bottomText: "", + // bottomText: pd["RISK_DESCR"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "风险分级", - rightText: pd["LEVEL"], + rightText: "", + // rightText: pd["LEVEL"]??"", ), Divider(height: 1), ListItemFactory.createColumnTextItem( topText: "检测内容", - bottomText: pd["CHECK_CONTENT"], + bottomText: "", + // bottomText: pd["CHECK_CONTENT"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "隐患部位", - rightText: pd["HIDDENPART"], + rightText: "", + // rightText: pd["HIDDENPART"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "发现人", - rightText: pd["CREATORNAME"], + rightText: "", + // rightText: pd["CREATORNAME"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "发现时间", - rightText: pd["CREATTIME"], + rightText: "", + // rightText: pd["CREATTIME"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "隐患类型", - rightText: pd["HIDDENTYPE_NAME"], + rightText: "", + // rightText: pd["HIDDENTYPE_NAME"]??"", ), Divider(height: 1), ListItemFactory.createRowSpaceBetweenItem( leftText: "整改类型", - rightText: _getZhengGaiType(pd), + rightText: "", + // rightText: _getZhengGaiType(pd), ), Divider(height: 1), diff --git a/lib/pages/home/work/laws_list_picker.dart b/lib/pages/home/work/laws_list_picker.dart index ceb82df..d991ff5 100644 --- a/lib/pages/home/work/laws_list_picker.dart +++ b/lib/pages/home/work/laws_list_picker.dart @@ -73,9 +73,10 @@ class _LawsListPickerState extends State { } else { expandedSet.add(cat.id); } + }else { + selectedId = cat.id; + widget.onSelected(cat); } - selectedId = cat.id; - widget.onSelected(cat); }); }, child: Container( diff --git a/lib/pages/home/work/laws_regulations_page.dart b/lib/pages/home/work/laws_regulations_page.dart index ccd0730..1ccb002 100644 --- a/lib/pages/home/work/laws_regulations_page.dart +++ b/lib/pages/home/work/laws_regulations_page.dart @@ -4,15 +4,19 @@ import 'dart:convert'; import 'package:path_provider/path_provider.dart'; import 'package:intl/intl.dart'; import 'package:dio/dio.dart'; +import 'package:qhd_prevention/pages/home/work/read_file_page.dart'; import 'package:qhd_prevention/pages/home/work/risk_list_picker.dart'; import 'package:qhd_prevention/pages/my_appbar.dart'; +import '../../../customWidget/remote_file_page.dart'; import '../../../customWidget/search_bar_widget.dart'; import '../../../http/ApiService.dart'; import '../../../tools/tools.dart'; import 'laws_list_picker.dart'; import 'package:url_launcher/url_launcher.dart'; +import 'laws_regulations_two_page.dart'; + class LawsRegulationsPage extends StatefulWidget { const LawsRegulationsPage({super.key}); @@ -90,28 +94,20 @@ class _LawsRegulationsPage extends State { //预览 try { final url =ApiService.baseImgPath + file["FILEPATH"]; - openFile(url); + + pushPage( + ReadFilePage( + fileUrl: url, + ), + context, + ); } catch (e) { print(e.toString()); } } - Future openFile(String url) async { - // final Uri _url = Uri.file(filePath); - if (await canLaunch(url)) { - await launch( - url, - forceSafariVC: false, // 仅在iOS上有效,设置为false以避免使用SafariViewController。 - forceWebView: false, // 设置为false以避免使用WebView。 - enableJavaScript: true, // 启用JavaScript。 - ); - } else { - throw 'Could not launch $url'; - } - - } String _formatDate(String dateString) { try { @@ -147,7 +143,9 @@ class _LawsRegulationsPage extends State { body: Column( children: [ // 搜索栏 - Padding( + Container( + color: Colors.white, + child:Padding( padding: EdgeInsets.all(10), child: SearchBarWidget( controller: _searchController, @@ -157,7 +155,7 @@ class _LawsRegulationsPage extends State { }, ), ), - + ), Expanded(child: _buildContent()), ], ), @@ -169,11 +167,13 @@ class _LawsRegulationsPage extends State { return LawsListPicker( data: _accordionList, onSelected: (item) { - setState(() { + pushPage(LawsRegulationsTwoPage(item!.id), context); + // setState(() { // riskId = item?.id; // itemNameTwo=item!.name; // setResult(); - }); + + // }); }, ); diff --git a/lib/pages/home/work/laws_regulations_two_page.dart b/lib/pages/home/work/laws_regulations_two_page.dart new file mode 100644 index 0000000..441b68d --- /dev/null +++ b/lib/pages/home/work/laws_regulations_two_page.dart @@ -0,0 +1,321 @@ +import 'package:flutter/material.dart'; + +import 'dart:convert'; +import 'package:path_provider/path_provider.dart'; +import 'package:intl/intl.dart'; +import 'package:dio/dio.dart'; +import 'package:qhd_prevention/pages/home/work/read_file_page.dart'; +import 'package:qhd_prevention/pages/home/work/risk_list_picker.dart'; +import 'package:qhd_prevention/pages/my_appbar.dart'; + +import '../../../customWidget/remote_file_page.dart'; +import '../../../customWidget/search_bar_widget.dart'; +import '../../../http/ApiService.dart'; +import '../../../tools/tools.dart'; +import 'laws_list_picker.dart'; +import 'package:url_launcher/url_launcher.dart'; + +class LawsRegulationsTwoPage extends StatefulWidget { + const LawsRegulationsTwoPage(this.id, {super.key}); + + final String id; + + @override + State createState() => _LawsRegulationsTwoPage(); +} + +class _LawsRegulationsTwoPage extends State { + final TextEditingController _searchController = TextEditingController(); + + List _fileList = []; + + + + @override + void initState() { + super.initState(); + _getLowListByParentId(); + } + + Future _getLowListByParentId() async { + try { + _fileList.clear(); + final result = await ApiService.getLowListByParentId(widget.id); + if (result['result'] == 'success') { + final List newList = result['varList'] ?? []; + setState(() { + + _fileList.addAll(newList); + + }); + } else { + _showMessage('加载数据失败'); + } + } catch (e) { + // 出错时可以 Toast 或者在页面上显示错误状态 + print('加载数据失败:$e'); + } + } + + + + + + Future _downloadFile(String filePath) async { + //下载 + try { + final Uri url = Uri.parse(ApiService.baseImgPath + filePath); + if (!await launchUrl(url,mode:LaunchMode.externalApplication)) { + throw Exception('Could not launch $url'); + } + } catch (e) { + print(e.toString()); + } + } + + Future _showFile(dynamic file) async { + //预览 + try { + final url =ApiService.baseImgPath + file["FILEPATH"]; + + pushPage( + ReadFilePage( + fileUrl: url, + ), + context, + ); + } catch (e) { + print(e.toString()); + } + } + + + + + String _formatDate(String dateString) { + try { + final date = DateTime.parse(dateString); + return DateFormat('yyyy-MM-dd HH:mm:ss').format(date); + } catch (e) { + return dateString; + } + } + + String _getFileType(String extension) { + switch (extension) { + case '.docx': + case '.doc': + return '文件属性:文档'; + case '.xls': + case '.xlsx': + return '文件属性:表格'; + case '.ppt': + case '.pptx': + return '文件属性:幻灯片'; + case '.pdf': + return '文件属性:PDF'; + default: + return '文件属性:'; + } + } + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: MyAppbar(title: "法律法规"), + body: Column( + children: [ + // 搜索栏 + // Container( + // color: Colors.white, + // child:Padding( + // padding: EdgeInsets.all(10), + // child: SearchBarWidget( + // controller: _searchController, + // onSearch: (keyboard) { + // // 输入请求接口 + // _getLowListByParentId(); + // }, + // ), + // ), + // ), + Expanded(child: _buildContent()), + ], + ), + ); + } + + Widget _buildContent() { + if (_fileList.isNotEmpty) { + return ListView.builder( + itemCount: _fileList.length, + itemBuilder: + (context, index) => FileCard( + file: _fileList[index], + onDownload: _downloadFile, + onPreview: _showFile, + formatDate: _formatDate, + getFileType: _getFileType, + ), + ); + } else { + return const Center( + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Icon(Icons.folder_open, size: 64, color: Colors.grey), + SizedBox(height: 16), + Text('暂无数据', style: TextStyle(fontSize: 18, color: Colors.grey)), + ], + ), + ); + } + } + + void _showMessage(String msg) { + ScaffoldMessenger.of(context).showSnackBar(SnackBar(content: Text(msg))); + } +} + +class CustomCollapse extends StatefulWidget { + final dynamic item; + final Function(String) onTap; + + const CustomCollapse({super.key, required this.item, required this.onTap}); + + @override + State createState() => _CustomCollapseState(); +} + +class _CustomCollapseState extends State { + bool _isExpanded = false; + + @override + Widget build(BuildContext context) { + final hasChildren = + widget.item['children'] != null && + (widget.item['children'] as List).isNotEmpty; + + return Card( + margin: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + child: Column( + children: [ + ListTile( + title: Text( + widget.item['name'] ?? '未命名', + style: const TextStyle(fontWeight: FontWeight.bold, fontSize: 16), + ), + trailing: + hasChildren + ? Icon( + _isExpanded ? Icons.expand_less : Icons.expand_more, + color: Colors.blue, + ) + : null, + onTap: + hasChildren + ? () => setState(() => _isExpanded = !_isExpanded) + : () => widget.onTap(widget.item['id'] ?? ''), + ), + if (_isExpanded && hasChildren) + ...(widget.item['children'] as List).map((child) { + return Padding( + padding: const EdgeInsets.only(left: 16.0), + child: ListTile( + title: Text(child['name']), + leading: const Icon(Icons.description, color: Colors.blue), + onTap: () => widget.onTap(child['id'] ?? ''), + ), + ); + }).toList(), + ], + ), + ); + } +} + +class FileCard extends StatelessWidget { + final dynamic file; + final Function(String) onDownload; + final Function(dynamic) onPreview; + final String Function(String) formatDate; + final String Function(String) getFileType; + + const FileCard({ + super.key, + required this.file, + required this.onDownload, + required this.onPreview, + required this.formatDate, + required this.getFileType, + }); + + @override + Widget build(BuildContext context) { + return Card( + margin: const EdgeInsets.symmetric(horizontal: 16, vertical: 8), + child: Padding( + padding: const EdgeInsets.all(12.0), + child: Column( + crossAxisAlignment: CrossAxisAlignment.start, + children: [ + Text( + file['NAME'] ?? '未命名文件', + style: const TextStyle(fontSize: 18, fontWeight: FontWeight.bold), + ), + const SizedBox(height: 10), + Text( + getFileType(file['extension_name'] ?? ''), + style: TextStyle(color: Colors.grey[600]), + ), + const SizedBox(height: 5), + Text( + '上传日期:${formatDate(file['CTIME'] ?? '')}', + style: TextStyle(color: Colors.grey[600]), + ), + const SizedBox(height: 15), + Row( + mainAxisAlignment: MainAxisAlignment.end, + children: [ + OutlinedButton.icon( + icon: const Icon(Icons.download, size: 18), + label: const Text('下载'), + onPressed: () => onDownload(file['FILEPATH'] ?? ''), + style: OutlinedButton.styleFrom( + foregroundColor: Colors.blue, + side: const BorderSide(color: Colors.blue), + ), + ), + const SizedBox(width: 10), + ElevatedButton.icon( + icon: const Icon(Icons.visibility, size: 18), + label: const Text('预览'), + onPressed: () => onPreview(file), + style: ElevatedButton.styleFrom( + backgroundColor: Colors.cyan, + foregroundColor: Colors.white, + ), + ), + ], + ), + ], + ), + ), + ); + } +} + +//详情页 +class StudyDetailPage extends StatelessWidget { + final String parentId; + + const StudyDetailPage({super.key, required this.parentId}); + + @override + Widget build(BuildContext context) { + return Scaffold( + appBar: AppBar(title: const Text('详情页面')), + body: Center(child: Text('父级ID: $parentId')), + ); + } +} diff --git a/lib/pages/home/work/read_file_page.dart b/lib/pages/home/work/read_file_page.dart new file mode 100644 index 0000000..100a5bb --- /dev/null +++ b/lib/pages/home/work/read_file_page.dart @@ -0,0 +1,148 @@ +import 'dart:async'; +import 'dart:io'; +import 'package:flutter/material.dart'; +import 'package:pdfx/pdfx.dart'; +import 'package:path_provider/path_provider.dart'; +import 'package:qhd_prevention/pages/my_appbar.dart'; +import 'package:qhd_prevention/customWidget/custom_button.dart'; +import 'package:dio/dio.dart'; +import 'package:qhd_prevention/tools/tools.dart'; + +class ReadFilePage extends StatefulWidget { + final String fileUrl; + // final int countdownSeconds; + + const ReadFilePage({ + Key? key, + required this.fileUrl, + // this.countdownSeconds = 3, + }) : super(key: key); + + @override + _ReadFilePageState createState() => _ReadFilePageState(); +} + +class _ReadFilePageState extends State { + String? _localPath; + bool _isLoading = true; + bool _hasScrolledToBottom = false; + bool _timerFinished = false; + late int _secondsRemaining; + Timer? _countdownTimer; + late PdfControllerPinch _pdfController; + int _totalPages = 0; + + @override + void initState() { + super.initState(); + // _secondsRemaining = widget.countdownSeconds; + _startCountdown(); + _downloadAndLoad(); + } + + Future _downloadAndLoad() async { + try { + final url = widget.fileUrl; + final filename = url.split('/').last; + final dir = await getTemporaryDirectory(); + final filePath = '${dir.path}/$filename'; + + final dio = Dio(); + final response = await dio.get>( + url, + options: Options(responseType: ResponseType.bytes), + ); + final file = File(filePath); + await file.writeAsBytes(response.data!); + + // 加载 PDF 控制器 + _pdfController = PdfControllerPinch( + document: PdfDocument.openFile(filePath), + ); + + setState(() { + _localPath = filePath; + _isLoading = false; + }); + } catch (e) { + // 下载或加载失败 + setState(() { + _isLoading = false; + }); + ScaffoldMessenger.of(context).showSnackBar( + SnackBar(content: Text('文件加载失败: \$e')), + ); + } + } + + void _startCountdown() { + _countdownTimer = Timer.periodic(const Duration(seconds: 1), (timer) { + setState(() { + if (_secondsRemaining > 1) { + _secondsRemaining--; + } else { + _secondsRemaining = 0; + _timerFinished = true; + _countdownTimer?.cancel(); + } + }); + }); + } + + @override + void dispose() { + _countdownTimer?.cancel(); + if (!_isLoading) { + _pdfController.dispose(); + } + super.dispose(); + } + + @override + Widget build(BuildContext context) { + final isButtonEnabled = _timerFinished && _hasScrolledToBottom; + return Scaffold( + appBar: MyAppbar(title: '文件详情'), + backgroundColor: Colors.white, + body: SafeArea( + child: Column( + children: [ + Expanded( + child: _isLoading + ? const Center(child: CircularProgressIndicator()) + : PdfViewPinch( + controller: _pdfController, + scrollDirection: Axis.vertical, + onDocumentLoaded: (document) { + setState(() { + _totalPages = document.pagesCount; + }); + }, + onPageChanged: (page) { + if (page == _totalPages - 1) { + setState(() => _hasScrolledToBottom = true); + } + }, + ), + ), + // Padding( + // padding: const EdgeInsets.all(16), + // child: CustomButton( + // backgroundColor: isButtonEnabled ? Colors.blue : Colors.grey, + // text: isButtonEnabled + // ? '我已学习完毕' + // : _secondsRemaining == 0 ? '我已学习完毕' : '($_secondsRemaining s)我已学习完毕', + // onPressed: isButtonEnabled + // ? () { + // // TODO: 完成回调 + // Navigator.pop(context); + // } + // : null, + // ), + // ), + ], + ), + ), + ); + } +} diff --git a/lib/tools/tools.dart b/lib/tools/tools.dart index 08741cf..3f7805e 100644 --- a/lib/tools/tools.dart +++ b/lib/tools/tools.dart @@ -165,6 +165,7 @@ class SessionService { String? dangerJson; String? riskJson; String? departmentJsonStr; + String? departmentHiddenTypeJsonStr; /// 如果以下任何一项为空,则跳转到登录页 void loginSession(BuildContext context) { diff --git a/pubspec.lock b/pubspec.lock index 6c4fe48..937467d 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -6,7 +6,7 @@ packages: description: name: args sha256: d0481093c50b1da8910eb0bb301626d4d8eb7284aa739614d2b394ee09e3ea04 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.7.0" asn1lib: @@ -14,7 +14,7 @@ packages: description: name: asn1lib sha256: "9a8f69025044eb466b9b60ef3bc3ac99b4dc6c158ae9c56d25eeccf5bc56d024" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.6.5" async: @@ -22,7 +22,7 @@ packages: description: name: async sha256: "758e6d74e971c3e5aceb4110bfd6698efc7f501675bcfe0c775459a8140750eb" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.13.0" boolean_selector: @@ -30,7 +30,7 @@ packages: description: name: boolean_selector sha256: "8aab1771e1243a5063b8b0ff68042d67334e3feab9e95b9490f9a6ebf73b42ea" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.2" camera: @@ -38,7 +38,7 @@ packages: description: name: camera sha256: d6ec2cbdbe2fa8f5e0d07d8c06368fe4effa985a4a5ddade9cc58a8cd849557d - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.11.2" camera_android_camerax: @@ -46,23 +46,23 @@ packages: description: name: camera_android_camerax sha256: "4b6c1bef4270c39df96402c4d62f2348c3bb2bbaefd0883b9dbd58f426306ad0" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.6.19" camera_avfoundation: dependency: transitive description: name: camera_avfoundation - sha256: cabc6cbdeadca9cbcac178110c9d5e87bde5c646c3b8c9c4c2747fc500a12432 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + sha256: b389be4a325742a3950e50475067d95a3de2fb32ba3f31bfcc62b0b6d19907a6 + url: "https://pub.flutter-io.cn" source: hosted - version: "0.9.20+5" + version: "0.9.20+4" camera_platform_interface: dependency: transitive description: name: camera_platform_interface sha256: "2f757024a48696ff4814a789b0bd90f5660c0fb25f393ab4564fb483327930e2" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.10.0" camera_web: @@ -70,7 +70,7 @@ packages: description: name: camera_web sha256: "595f28c89d1fb62d77c73c633193755b781c6d2e0ebcd8dc25b763b514e6ba8f" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.3.5" characters: @@ -78,7 +78,7 @@ packages: description: name: characters sha256: f71061c654a3380576a52b451dd5532377954cf9dbd272a78fc8479606670803 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.4.0" clock: @@ -86,7 +86,7 @@ packages: description: name: clock sha256: fddb70d9b5277016c77a80201021d40a2247104d9f4aa7bab7157b7e3f05b84b - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.1.2" collection: @@ -94,7 +94,7 @@ packages: description: name: collection sha256: "2f5709ae4d3d59dd8f7cd309b4e023046b57d8a6c82130785d2b0e5868084e76" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.19.1" connectivity_plus: @@ -102,7 +102,7 @@ packages: description: name: connectivity_plus sha256: "051849e2bd7c7b3bc5844ea0d096609ddc3a859890ec3a9ac4a65a2620cc1f99" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.1.4" connectivity_plus_platform_interface: @@ -110,7 +110,7 @@ packages: description: name: connectivity_plus_platform_interface sha256: "42657c1715d48b167930d5f34d00222ac100475f73d10162ddf43e714932f204" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.0.1" convert: @@ -118,7 +118,7 @@ packages: description: name: convert sha256: b30acd5944035672bc15c6b7a8b47d773e41e2f17de064350988c5d02adb1c68 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.1.2" cross_file: @@ -126,7 +126,7 @@ packages: description: name: cross_file sha256: "7caf6a750a0c04effbb52a676dce9a4a592e10ad35c34d6d2d0e4811160d5670" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.3.4+2" crypto: @@ -134,7 +134,7 @@ packages: description: name: crypto sha256: "1e445881f28f22d6140f181e07737b22f1e099a5e1ff94b0af2f9e4a463f4855" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.0.6" csslib: @@ -142,7 +142,7 @@ packages: description: name: csslib sha256: "09bad715f418841f976c77db72d5398dc1253c21fb9c0c7f0b0b985860b2d58e" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.0.2" cupertino_icons: @@ -150,7 +150,7 @@ packages: description: name: cupertino_icons sha256: ba631d1c7f7bef6b729a622b7b752645a2d076dba9976925b8f25725a30e1ee6 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.0.8" dbus: @@ -158,7 +158,7 @@ packages: description: name: dbus sha256: "79e0c23480ff85dc68de79e2cd6334add97e48f7f4865d17686dd6ea81a47e8c" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.7.11" dio: @@ -166,7 +166,7 @@ packages: description: name: dio sha256: "253a18bbd4851fecba42f7343a1df3a9a4c1d31a2c1b37e221086b4fa8c8dbc9" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "5.8.0+1" dio_web_adapter: @@ -174,23 +174,15 @@ packages: description: name: dio_web_adapter sha256: "7586e476d70caecaf1686d21eee7247ea43ef5c345eab9e0cc3583ff13378d78" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.1" - dotted_line: - dependency: "direct main" - description: - name: dotted_line - sha256: "41e3d655939559815daa1370fc1e07673a205fa628cf40ce3af45d90029a77b6" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" - source: hosted - version: "3.2.3" encrypt: dependency: "direct main" description: name: encrypt sha256: "62d9aa4670cc2a8798bab89b39fc71b6dfbacf615de6cf5001fb39f7e4a996a2" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "5.0.3" extended_image: @@ -198,7 +190,7 @@ packages: description: name: extended_image sha256: f6cbb1d798f51262ed1a3d93b4f1f2aa0d76128df39af18ecb77fa740f88b2e0 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "10.0.1" extended_image_library: @@ -206,7 +198,7 @@ packages: description: name: extended_image_library sha256: "1f9a24d3a00c2633891c6a7b5cab2807999eb2d5b597e5133b63f49d113811fe" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "5.0.1" extension: @@ -214,7 +206,7 @@ packages: description: name: extension sha256: be3a6b7f8adad2f6e2e8c63c895d19811fcf203e23466c6296267941d0ff4f24 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.6.0" fake_async: @@ -222,7 +214,7 @@ packages: description: name: fake_async sha256: "5368f224a74523e8d2e7399ea1638b37aecfca824a3cc4dfdf77bf1fa905ac44" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.3.3" ffi: @@ -230,7 +222,7 @@ packages: description: name: ffi sha256: "289279317b4b16eb2bb7e271abccd4bf84ec9bdcbe999e278a94b804f5630418" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.4" file: @@ -238,7 +230,7 @@ packages: description: name: file sha256: a3b4f84adafef897088c160faf7dfffb7696046cb13ae90b508c2cbc95d3b8d4 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "7.0.1" file_selector_linux: @@ -246,7 +238,7 @@ packages: description: name: file_selector_linux sha256: "54cbbd957e1156d29548c7d9b9ec0c0ebb6de0a90452198683a7d23aed617a33" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.9.3+2" file_selector_macos: @@ -254,7 +246,7 @@ packages: description: name: file_selector_macos sha256: "8c9250b2bd2d8d4268e39c82543bacbaca0fda7d29e0728c3c4bbb7c820fd711" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.9.4+3" file_selector_platform_interface: @@ -262,7 +254,7 @@ packages: description: name: file_selector_platform_interface sha256: a3994c26f10378a039faa11de174d7b78eb8f79e4dd0af2a451410c1a5c3f66b - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.6.2" file_selector_windows: @@ -270,7 +262,7 @@ packages: description: name: file_selector_windows sha256: "320fcfb6f33caa90f0b58380489fc5ac05d99ee94b61aa96ec2bff0ba81d3c2b" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.9.3+4" fixnum: @@ -278,7 +270,7 @@ packages: description: name: fixnum sha256: b6dc7065e46c974bc7c5f143080a6764ec7a4be6da1285ececdc37be96de53be - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.1.1" flutter: @@ -291,7 +283,7 @@ packages: description: name: flutter_html sha256: "38a2fd702ffdf3243fb7441ab58aa1bc7e6922d95a50db76534de8260638558d" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.0.0" flutter_lints: @@ -299,7 +291,7 @@ packages: description: name: flutter_lints sha256: "5398f14efa795ffb7a33e9b6a08798b26a180edac4ad7db3f231e40f82ce11e1" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "5.0.0" flutter_plugin_android_lifecycle: @@ -307,7 +299,7 @@ packages: description: name: flutter_plugin_android_lifecycle sha256: f948e346c12f8d5480d2825e03de228d0eb8c3a737e4cdaa122267b89c022b5e - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.0.28" flutter_test: @@ -325,7 +317,7 @@ packages: description: name: fluttertoast sha256: "25e51620424d92d3db3832464774a6143b5053f15e382d8ffbfd40b6e795dcf1" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "8.2.12" geolocator: @@ -333,7 +325,7 @@ packages: description: name: geolocator sha256: f4efb8d3c4cdcad2e226af9661eb1a0dd38c71a9494b22526f9da80ab79520e5 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "10.1.1" geolocator_android: @@ -341,7 +333,7 @@ packages: description: name: geolocator_android sha256: fcb1760a50d7500deca37c9a666785c047139b5f9ee15aa5469fae7dbbe3170d - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "4.6.2" geolocator_apple: @@ -349,7 +341,7 @@ packages: description: name: geolocator_apple sha256: dbdd8789d5aaf14cf69f74d4925ad1336b4433a6efdf2fce91e8955dc921bf22 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.3.13" geolocator_platform_interface: @@ -357,7 +349,7 @@ packages: description: name: geolocator_platform_interface sha256: "30cb64f0b9adcc0fb36f628b4ebf4f731a2961a0ebd849f4b56200205056fe67" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "4.2.6" geolocator_web: @@ -365,7 +357,7 @@ packages: description: name: geolocator_web sha256: "102e7da05b48ca6bf0a5bda0010f886b171d1a08059f01bfe02addd0175ebece" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.2.1" geolocator_windows: @@ -373,7 +365,7 @@ packages: description: name: geolocator_windows sha256: "175435404d20278ffd220de83c2ca293b73db95eafbdc8131fe8609be1421eb6" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.2.5" html: @@ -381,7 +373,7 @@ packages: description: name: html sha256: "6d1264f2dffa1b1101c25a91dff0dc2daee4c18e87cd8538729773c073dbf602" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.15.6" http: @@ -389,7 +381,7 @@ packages: description: name: http sha256: "2c11f3f94c687ee9bad77c171151672986360b2b001d109814ee7140b2cf261b" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.4.0" http_client_helper: @@ -397,7 +389,7 @@ packages: description: name: http_client_helper sha256: "8a9127650734da86b5c73760de2b404494c968a3fd55602045ffec789dac3cb1" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.0.0" http_parser: @@ -405,7 +397,7 @@ packages: description: name: http_parser sha256: "178d74305e7866013777bab2c3d8726205dc5a4dd935297175b19a23a2e66571" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "4.1.2" image_picker: @@ -413,7 +405,7 @@ packages: description: name: image_picker sha256: "021834d9c0c3de46bf0fe40341fa07168407f694d9b2bb18d532dc1261867f7a" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.1.2" image_picker_android: @@ -421,7 +413,7 @@ packages: description: name: image_picker_android sha256: "6fae381e6af2bbe0365a5e4ce1db3959462fa0c4d234facf070746024bb80c8d" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.8.12+24" image_picker_for_web: @@ -429,7 +421,7 @@ packages: description: name: image_picker_for_web sha256: "717eb042ab08c40767684327be06a5d8dbb341fe791d514e4b92c7bbe1b7bb83" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.0.6" image_picker_ios: @@ -437,7 +429,7 @@ packages: description: name: image_picker_ios sha256: "05da758e67bc7839e886b3959848aa6b44ff123ab4b28f67891008afe8ef9100" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.8.12+2" image_picker_linux: @@ -445,7 +437,7 @@ packages: description: name: image_picker_linux sha256: "34a65f6740df08bbbeb0a1abd8e6d32107941fd4868f67a507b25601651022c9" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.2.1+2" image_picker_macos: @@ -453,7 +445,7 @@ packages: description: name: image_picker_macos sha256: "1b90ebbd9dcf98fb6c1d01427e49a55bd96b5d67b8c67cf955d60a5de74207c1" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.2.1+2" image_picker_platform_interface: @@ -461,7 +453,7 @@ packages: description: name: image_picker_platform_interface sha256: "886d57f0be73c4b140004e78b9f28a8914a09e50c2d816bdd0520051a71236a0" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.10.1" image_picker_windows: @@ -469,7 +461,7 @@ packages: description: name: image_picker_windows sha256: "6ad07afc4eb1bc25f3a01084d28520496c4a3bb0cb13685435838167c9dcedeb" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.2.1+1" intl: @@ -477,7 +469,7 @@ packages: description: name: intl sha256: "3df61194eb431efc39c4ceba583b95633a403f46c9fd341e550ce0bfa50e9aa5" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.20.2" js: @@ -485,7 +477,7 @@ packages: description: name: js sha256: "53385261521cc4a0c4658fd0ad07a7d14591cf8fc33abbceae306ddb974888dc" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.7.2" leak_tracker: @@ -493,7 +485,7 @@ packages: description: name: leak_tracker sha256: "6bb818ecbdffe216e81182c2f0714a2e62b593f4a4f13098713ff1685dfb6ab0" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "10.0.9" leak_tracker_flutter_testing: @@ -501,7 +493,7 @@ packages: description: name: leak_tracker_flutter_testing sha256: f8b613e7e6a13ec79cfdc0e97638fddb3ab848452eff057653abd3edba760573 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.0.9" leak_tracker_testing: @@ -509,7 +501,7 @@ packages: description: name: leak_tracker_testing sha256: "6ba465d5d76e67ddf503e1161d1f4a6bc42306f9d66ca1e8f079a47290fb06d3" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.0.1" lints: @@ -517,7 +509,7 @@ packages: description: name: lints sha256: c35bb79562d980e9a453fc715854e1ed39e24e7d0297a880ef54e17f9874a9d7 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "5.1.1" list_counter: @@ -525,7 +517,7 @@ packages: description: name: list_counter sha256: c447ae3dfcd1c55f0152867090e67e219d42fe6d4f2807db4bbe8b8d69912237 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.0.2" matcher: @@ -533,7 +525,7 @@ packages: description: name: matcher sha256: dc58c723c3c24bf8d3e2d3ad3f2f9d7bd9cf43ec6feaa64181775e60190153f2 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.12.17" material_color_utilities: @@ -541,7 +533,7 @@ packages: description: name: material_color_utilities sha256: f7142bb1154231d7ea5f96bc7bde4bda2a0945d2806bb11670e30b850d56bdec - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.11.1" meta: @@ -549,7 +541,7 @@ packages: description: name: meta sha256: e3641ec5d63ebf0d9b41bd43201a66e3fc79a65db5f61fc181f04cd27aab950c - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.16.0" mime: @@ -557,7 +549,7 @@ packages: description: name: mime sha256: "41a20518f0cb1256669420fdba0cd90d21561e560ac240f26ef8322e45bb7ed6" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.0.0" mobile_scanner: @@ -565,7 +557,7 @@ packages: description: name: mobile_scanner sha256: "54005bdea7052d792d35b4fef0f84ec5ddc3a844b250ecd48dc192fb9b4ebc95" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "7.0.1" ndef_record: @@ -573,7 +565,7 @@ packages: description: name: ndef_record sha256: "0c72dfac0d5c16fc264846d103ee5d8249cd3858261a5a537b455a24c1bd5857" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.2.1" nested: @@ -581,7 +573,7 @@ packages: description: name: nested sha256: "03bac4c528c64c95c722ec99280375a6f2fc708eec17c7b3f07253b626cd2a20" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.0.0" nfc_manager: @@ -589,7 +581,7 @@ packages: description: name: nfc_manager sha256: "164cc0223dee528d4d05a542da921f0b3a31ca0312400701c93ebf4ce757f676" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "4.0.2" nfc_manager_ndef: @@ -597,7 +589,7 @@ packages: description: name: nfc_manager_ndef sha256: "676e741c42b63ab1fda5a981015cb706ab4fdb76e5d0eec6611993bb27d7e7bf" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.0.1" nm: @@ -605,7 +597,7 @@ packages: description: name: nm sha256: "2c9aae4127bdc8993206464fcc063611e0e36e72018696cd9631023a31b24254" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.5.0" package_info_plus: @@ -613,7 +605,7 @@ packages: description: name: package_info_plus sha256: "7976bfe4c583170d6cdc7077e3237560b364149fcd268b5f53d95a991963b191" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "8.3.0" package_info_plus_platform_interface: @@ -621,7 +613,7 @@ packages: description: name: package_info_plus_platform_interface sha256: "6c935fb612dff8e3cc9632c2b301720c77450a126114126ffaafe28d2e87956c" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.2.0" path: @@ -629,7 +621,7 @@ packages: description: name: path sha256: "75cca69d1490965be98c73ceaea117e8a04dd21217b37b292c9ddbec0d955bc5" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.9.1" path_provider: @@ -637,7 +629,7 @@ packages: description: name: path_provider sha256: "50c5dd5b6e1aaf6fb3a78b33f6aa3afca52bf903a8a5298f53101fdaee55bbcd" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.5" path_provider_android: @@ -645,7 +637,7 @@ packages: description: name: path_provider_android sha256: d0d310befe2c8ab9e7f393288ccbb11b60c019c6b5afc21973eeee4dda2b35e9 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.2.17" path_provider_foundation: @@ -653,7 +645,7 @@ packages: description: name: path_provider_foundation sha256: "4843174df4d288f5e29185bd6e72a6fbdf5a4a4602717eed565497429f179942" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.1" path_provider_linux: @@ -661,7 +653,7 @@ packages: description: name: path_provider_linux sha256: f7a1fe3a634fe7734c8d3f2766ad746ae2a2884abe22e241a8b301bf5cac3279 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.2.1" path_provider_platform_interface: @@ -669,7 +661,7 @@ packages: description: name: path_provider_platform_interface sha256: "88f5779f72ba699763fa3a3b06aa4bf6de76c8e5de842cf6f29e2e06476c2334" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.2" path_provider_windows: @@ -677,7 +669,7 @@ packages: description: name: path_provider_windows sha256: bd6f00dbd873bfb70d0761682da2b3a2c2fccc2b9e84c495821639601d81afe7 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.3.0" pdfx: @@ -685,7 +677,7 @@ packages: description: name: pdfx sha256: "29db9b71d46bf2335e001f91693f2c3fbbf0760e4c2eb596bf4bafab211471c1" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.9.2" petitparser: @@ -693,7 +685,7 @@ packages: description: name: petitparser sha256: "07c8f0b1913bcde1ff0d26e57ace2f3012ccbf2b204e070290dad3bb22797646" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.1.0" photo_manager: @@ -701,7 +693,7 @@ packages: description: name: photo_manager sha256: a0d9a7a9bc35eda02d33766412bde6d883a8b0acb86bbe37dac5f691a0894e8a - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.7.1" photo_manager_image_provider: @@ -709,7 +701,7 @@ packages: description: name: photo_manager_image_provider sha256: b6015b67b32f345f57cf32c126f871bced2501236c405aafaefa885f7c821e4f - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.2.0" photo_view: @@ -717,7 +709,7 @@ packages: description: name: photo_view sha256: "1fc3d970a91295fbd1364296575f854c9863f225505c28c46e0a03e48960c75e" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.15.0" platform: @@ -725,7 +717,7 @@ packages: description: name: platform sha256: "5d6b1b0036a5f331ebc77c850ebc8506cbc1e9416c27e59b439f917a902a4984" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.1.6" plugin_platform_interface: @@ -733,7 +725,7 @@ packages: description: name: plugin_platform_interface sha256: "4820fbfdb9478b1ebae27888254d445073732dae3d6ea81f0b7e06d5dedc3f02" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.8" pointycastle: @@ -741,7 +733,7 @@ packages: description: name: pointycastle sha256: "4be0097fcf3fd3e8449e53730c631200ebc7b88016acecab2b0da2f0149222fe" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.9.1" provider: @@ -749,7 +741,7 @@ packages: description: name: provider sha256: "4abbd070a04e9ddc287673bf5a030c7ca8b685ff70218720abab8b092f53dd84" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.1.5" shared_preferences: @@ -757,7 +749,7 @@ packages: description: name: shared_preferences sha256: "6e8bf70b7fef813df4e9a36f658ac46d107db4b4cfe1048b477d4e453a8159f5" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.5.3" shared_preferences_android: @@ -765,7 +757,7 @@ packages: description: name: shared_preferences_android sha256: "20cbd561f743a342c76c151d6ddb93a9ce6005751e7aa458baad3858bfbfb6ac" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.10" shared_preferences_foundation: @@ -773,7 +765,7 @@ packages: description: name: shared_preferences_foundation sha256: "6a52cfcdaeac77cad8c97b539ff688ccfc458c007b4db12be584fbe5c0e49e03" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.5.4" shared_preferences_linux: @@ -781,7 +773,7 @@ packages: description: name: shared_preferences_linux sha256: "580abfd40f415611503cae30adf626e6656dfb2f0cee8f465ece7b6defb40f2f" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.1" shared_preferences_platform_interface: @@ -789,7 +781,7 @@ packages: description: name: shared_preferences_platform_interface sha256: "57cbf196c486bc2cf1f02b85784932c6094376284b3ad5779d1b1c6c6a816b80" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.1" shared_preferences_web: @@ -797,7 +789,7 @@ packages: description: name: shared_preferences_web sha256: c49bd060261c9a3f0ff445892695d6212ff603ef3115edbb448509d407600019 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.3" shared_preferences_windows: @@ -805,7 +797,7 @@ packages: description: name: shared_preferences_windows sha256: "94ef0f72b2d71bc3e700e025db3710911bd51a71cefb65cc609dd0d9a982e3c1" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.1" simple_gesture_detector: @@ -813,7 +805,7 @@ packages: description: name: simple_gesture_detector sha256: ba2cd5af24ff20a0b8d609cec3f40e5b0744d2a71804a2616ae086b9c19d19a3 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.2.1" sky_engine: @@ -826,7 +818,7 @@ packages: description: name: source_span sha256: "254ee5351d6cb365c859e20ee823c3bb479bf4a293c22d17a9f1bf144ce86f7c" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.10.1" sprintf: @@ -834,7 +826,7 @@ packages: description: name: sprintf sha256: "1fc9ffe69d4df602376b52949af107d8f5703b77cda567c4d7d86a0693120f23" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "7.0.0" stack_trace: @@ -842,7 +834,7 @@ packages: description: name: stack_trace sha256: "8b27215b45d22309b5cddda1aa2b19bdfec9df0e765f2de506401c071d38d1b1" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.12.1" stream_channel: @@ -850,7 +842,7 @@ packages: description: name: stream_channel sha256: "969e04c80b8bcdf826f8f16579c7b14d780458bd97f56d107d3950fdbeef059d" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.4" stream_transform: @@ -858,7 +850,7 @@ packages: description: name: stream_transform sha256: ad47125e588cfd37a9a7f86c7d6356dde8dfe89d071d293f80ca9e9273a33871 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.1" string_scanner: @@ -866,7 +858,7 @@ packages: description: name: string_scanner sha256: "921cd31725b72fe181906c6a94d987c78e3b98c2e205b397ea399d4054872b43" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.4.1" synchronized: @@ -874,7 +866,7 @@ packages: description: name: synchronized sha256: c254ade258ec8282947a0acbbc90b9575b4f19673533ee46f2f6e9b3aeefd7c0 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.4.0" table_calendar: @@ -882,7 +874,7 @@ packages: description: name: table_calendar sha256: "0c0c6219878b363a2d5f40c7afb159d845f253d061dc3c822aa0d5fe0f721982" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.2.0" term_glyph: @@ -890,7 +882,7 @@ packages: description: name: term_glyph sha256: "7f554798625ea768a7518313e58f83891c7f5024f88e46e7182a4558850a4b8e" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.2.2" test_api: @@ -898,7 +890,7 @@ packages: description: name: test_api sha256: fb31f383e2ee25fbbfe06b40fe21e1e458d14080e3c67e7ba0acfde4df4e0bbd - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.7.4" typed_data: @@ -906,7 +898,7 @@ packages: description: name: typed_data sha256: f9049c039ebfeb4cf7a7104a675823cd72dba8297f264b6637062516699fa006 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.4.0" universal_platform: @@ -914,7 +906,7 @@ packages: description: name: universal_platform sha256: "64e16458a0ea9b99260ceb5467a214c1f298d647c659af1bff6d3bf82536b1ec" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.1.0" url_launcher: @@ -922,7 +914,7 @@ packages: description: name: url_launcher sha256: f6a7e5c4835bb4e3026a04793a4199ca2d14c739ec378fdfe23fc8075d0439f8 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.3.2" url_launcher_android: @@ -930,7 +922,7 @@ packages: description: name: url_launcher_android sha256: "8582d7f6fe14d2652b4c45c9b6c14c0b678c2af2d083a11b604caeba51930d79" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.3.16" url_launcher_ios: @@ -938,7 +930,7 @@ packages: description: name: url_launcher_ios sha256: "7f2022359d4c099eea7df3fdf739f7d3d3b9faf3166fb1dd390775176e0b76cb" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.3.3" url_launcher_linux: @@ -946,7 +938,7 @@ packages: description: name: url_launcher_linux sha256: "4e9ba368772369e3e08f231d2301b4ef72b9ff87c31192ef471b380ef29a4935" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.2.1" url_launcher_macos: @@ -954,7 +946,7 @@ packages: description: name: url_launcher_macos sha256: "17ba2000b847f334f16626a574c702b196723af2a289e7a93ffcb79acff855c2" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.2.2" url_launcher_platform_interface: @@ -962,7 +954,7 @@ packages: description: name: url_launcher_platform_interface sha256: "552f8a1e663569be95a8190206a38187b531910283c3e982193e4f2733f01029" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.3.2" url_launcher_web: @@ -970,7 +962,7 @@ packages: description: name: url_launcher_web sha256: "4bd2b7b4dc4d4d0b94e5babfffbca8eac1a126c7f3d6ecbc1a11013faa3abba2" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.1" url_launcher_windows: @@ -978,7 +970,7 @@ packages: description: name: url_launcher_windows sha256: "3284b6d2ac454cf34f114e1d3319866fdd1e19cdc329999057e44ffe936cfa77" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.1.4" uuid: @@ -986,7 +978,7 @@ packages: description: name: uuid sha256: a5be9ef6618a7ac1e964353ef476418026db906c4facdedaa299b7a2e71690ff - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "4.5.1" vector_math: @@ -994,7 +986,7 @@ packages: description: name: vector_math sha256: "80b3257d1492ce4d091729e3a67a60407d227c27241d6927be0130c98e741803" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.1.4" video_player: @@ -1002,7 +994,7 @@ packages: description: name: video_player sha256: "0d55b1f1a31e5ad4c4967bfaa8ade0240b07d20ee4af1dfef5f531056512961a" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.10.0" video_player_android: @@ -1010,7 +1002,7 @@ packages: description: name: video_player_android sha256: "4a5135754a62dbc827a64a42ef1f8ed72c962e191c97e2d48744225c2b9ebb73" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.8.7" video_player_avfoundation: @@ -1018,7 +1010,7 @@ packages: description: name: video_player_avfoundation sha256: "9fedd55023249f3a02738c195c906b4e530956191febf0838e37d0dac912f953" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.8.0" video_player_platform_interface: @@ -1026,7 +1018,7 @@ packages: description: name: video_player_platform_interface sha256: cf2a1d29a284db648fd66cbd18aacc157f9862d77d2cc790f6f9678a46c1db5a - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.4.0" video_player_web: @@ -1034,7 +1026,7 @@ packages: description: name: video_player_web sha256: "9f3c00be2ef9b76a95d94ac5119fb843dca6f2c69e6c9968f6f2b6c9e7afbdeb" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.4.0" visibility_detector: @@ -1042,7 +1034,7 @@ packages: description: name: visibility_detector sha256: dd5cc11e13494f432d15939c3aa8ae76844c42b723398643ce9addb88a5ed420 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "0.4.0+2" vm_service: @@ -1050,7 +1042,7 @@ packages: description: name: vm_service sha256: ddfa8d30d89985b96407efce8acbdd124701f96741f2d981ca860662f1c0dc02 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "15.0.0" web: @@ -1058,7 +1050,7 @@ packages: description: name: web sha256: "868d88a33d8a87b18ffc05f9f030ba328ffefba92d6c127917a2ba740f9cfe4a" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.1.1" webview_flutter: @@ -1066,7 +1058,7 @@ packages: description: name: webview_flutter sha256: c3e4fe614b1c814950ad07186007eff2f2e5dd2935eba7b9a9a1af8e5885f1ba - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "4.13.0" webview_flutter_android: @@ -1074,7 +1066,7 @@ packages: description: name: webview_flutter_android sha256: "9573ad97890d199ac3ab32399aa33a5412163b37feb573eb5b0a76b35e9ffe41" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "4.8.2" webview_flutter_platform_interface: @@ -1082,7 +1074,7 @@ packages: description: name: webview_flutter_platform_interface sha256: f0dc2dc3a2b1e3a6abdd6801b9355ebfeb3b8f6cde6b9dc7c9235909c4a1f147 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "2.13.1" webview_flutter_wkwebview: @@ -1090,7 +1082,7 @@ packages: description: name: webview_flutter_wkwebview sha256: "71523b9048cf510cfa1fd4e0a3fa5e476a66e0884d5df51d59d5023dba237107" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "3.22.1" wechat_assets_picker: @@ -1098,7 +1090,7 @@ packages: description: name: wechat_assets_picker sha256: cafe3d32564ed3cacf9822f251941f7b44fe9885c17c8de4fca7e939a459e1ef - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "9.5.1" wechat_picker_library: @@ -1106,7 +1098,7 @@ packages: description: name: wechat_picker_library sha256: a42e09cb85b15fc9410f6a69671371cc60aa99c4a1f7967f6593a7f665f6f47a - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.0.5" win32: @@ -1114,7 +1106,7 @@ packages: description: name: win32 sha256: "66814138c3562338d05613a6e368ed8cfb237ad6d64a9e9334be3f309acfca03" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "5.14.0" xdg_directories: @@ -1122,7 +1114,7 @@ packages: description: name: xdg_directories sha256: "7a3f37b05d989967cdddcbb571f1ea834867ae2faa29725fd085180e0883aa15" - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "1.1.0" xml: @@ -1130,7 +1122,7 @@ packages: description: name: xml sha256: b015a8ad1c488f66851d762d3090a21c600e479dc75e68328c52774040cf9226 - url: "https://mirrors.tuna.tsinghua.edu.cn/dart-pub/" + url: "https://pub.flutter-io.cn" source: hosted version: "6.5.0" sdks: