2025-08-04 15:24:40 +08:00
|
|
|
|
import 'package:flustars/flustars.dart';
|
2025-08-26 11:16:50 +08:00
|
|
|
|
import 'package:flutter/gestures.dart';
|
2024-01-09 14:05:18 +08:00
|
|
|
|
import 'package:flutter/material.dart';
|
2024-04-28 15:00:13 +08:00
|
|
|
|
import 'package:flutter_easyloading/flutter_easyloading.dart';
|
2024-01-09 14:05:18 +08:00
|
|
|
|
import 'package:flutter_screenutil/flutter_screenutil.dart';
|
|
|
|
|
|
import 'package:get/get.dart';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
import 'package:intl/intl.dart';
|
2024-01-26 16:19:01 +08:00
|
|
|
|
import 'package:star_lock/appRouters.dart';
|
2025-08-18 14:48:24 +08:00
|
|
|
|
import 'package:star_lock/app_settings/app_settings.dart';
|
|
|
|
|
|
import 'package:star_lock/main/lockDetail/doorLockLog/date_time_extensions.dart';
|
2024-02-26 17:56:30 +08:00
|
|
|
|
import 'package:star_lock/main/lockDetail/doorLockLog/doorLockLog_entity.dart';
|
2024-01-09 14:05:18 +08:00
|
|
|
|
import 'package:star_lock/main/lockDetail/doorLockLog/doorLockLog_logic.dart';
|
2024-06-15 17:20:55 +08:00
|
|
|
|
import 'package:star_lock/main/lockDetail/doorLockLog/doorLockLog_state.dart';
|
2024-06-19 11:02:21 +08:00
|
|
|
|
import 'package:star_lock/main/lockDetail/doorLockLog/exportRecordDialog/exportRecordDialog_page.dart';
|
2025-08-18 14:48:24 +08:00
|
|
|
|
import 'package:star_lock/main/lockDetail/doorLockLog/week_calendar_view.dart';
|
2025-01-10 14:46:51 +08:00
|
|
|
|
import 'package:star_lock/main/lockDetail/videoLog/videoLog/videoLog_entity.dart';
|
|
|
|
|
|
import 'package:star_lock/main/lockDetail/videoLog/widget/full_screenImage_page.dart';
|
2025-01-21 14:09:09 +08:00
|
|
|
|
import 'package:star_lock/main/lockDetail/videoLog/widget/video_thumbnail_image.dart';
|
2024-05-16 10:19:01 +08:00
|
|
|
|
import 'package:star_lock/tools/EasyRefreshTool.dart';
|
2024-01-26 14:28:02 +08:00
|
|
|
|
import 'package:star_lock/tools/advancedCalendar/src/widget.dart';
|
2024-05-16 16:45:00 +08:00
|
|
|
|
import 'package:star_lock/tools/commonDataManage.dart';
|
2025-01-10 14:46:51 +08:00
|
|
|
|
import 'package:star_lock/tools/dateTool.dart';
|
2024-01-26 16:19:01 +08:00
|
|
|
|
import 'package:star_lock/tools/menuItem/xsDropDownWidget.dart';
|
2024-04-03 09:53:30 +08:00
|
|
|
|
import 'package:star_lock/tools/noData.dart';
|
2024-05-16 11:59:37 +08:00
|
|
|
|
import 'package:star_lock/tools/showCupertinoAlertView.dart';
|
2024-06-15 17:20:55 +08:00
|
|
|
|
import 'package:star_lock/tools/showTipView.dart';
|
2024-01-18 11:24:12 +08:00
|
|
|
|
import 'package:timelines/timelines.dart';
|
2024-01-09 14:05:18 +08:00
|
|
|
|
|
|
|
|
|
|
import '../../../app_settings/app_colors.dart';
|
2024-04-28 15:00:13 +08:00
|
|
|
|
import '../../../tools/appRouteObserver.dart';
|
2024-01-09 14:05:18 +08:00
|
|
|
|
import '../../../tools/titleAppBar.dart';
|
|
|
|
|
|
|
|
|
|
|
|
class DoorLockLogPage extends StatefulWidget {
|
|
|
|
|
|
const DoorLockLogPage({Key? key}) : super(key: key);
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
State<DoorLockLogPage> createState() => _DoorLockLogPageState();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-05-13 17:43:44 +08:00
|
|
|
|
class _DoorLockLogPageState extends State<DoorLockLogPage> with RouteAware {
|
2025-08-19 10:16:20 +08:00
|
|
|
|
final ScrollController _scrollController = ScrollController();
|
2024-06-15 17:20:55 +08:00
|
|
|
|
final DoorLockLogLogic logic = Get.put(DoorLockLogLogic());
|
|
|
|
|
|
final DoorLockLogState state = Get.find<DoorLockLogLogic>().state;
|
2025-08-19 10:16:20 +08:00
|
|
|
|
bool _isAtBottom = false;
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
void initState() {
|
|
|
|
|
|
super.initState();
|
|
|
|
|
|
_scrollController.addListener(_onScroll);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _onScroll() {
|
|
|
|
|
|
final max = _scrollController.position.maxScrollExtent;
|
|
|
|
|
|
final current = _scrollController.position.pixels;
|
|
|
|
|
|
|
|
|
|
|
|
AppLog.log('current:${current}');
|
|
|
|
|
|
// 判断是否接近底部(例如 5 像素内)
|
|
|
|
|
|
if (current >= max - 5) {
|
|
|
|
|
|
if (!_isAtBottom) {
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
_isAtBottom = true;
|
|
|
|
|
|
});
|
|
|
|
|
|
print('✅ 已滑动到 timelines 列表底部!');
|
|
|
|
|
|
// 可以在这里触发加载更多、发送事件等
|
|
|
|
|
|
}
|
|
|
|
|
|
} else {
|
|
|
|
|
|
if (_isAtBottom) {
|
|
|
|
|
|
setState(() {
|
|
|
|
|
|
_isAtBottom = false;
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2024-01-09 14:05:18 +08:00
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
|
return Scaffold(
|
|
|
|
|
|
backgroundColor: AppColors.mainBackgroundColor,
|
|
|
|
|
|
appBar: TitleAppBar(
|
2024-07-25 14:40:02 +08:00
|
|
|
|
barTitle: '操作记录'.tr,
|
2024-01-09 14:05:18 +08:00
|
|
|
|
haveBack: true,
|
|
|
|
|
|
backgroundColor: AppColors.mainColor,
|
2024-06-15 17:20:55 +08:00
|
|
|
|
actionsList: <Widget>[
|
2024-05-16 16:45:00 +08:00
|
|
|
|
Visibility(
|
2025-09-06 14:50:08 +08:00
|
|
|
|
visible: CommonDataManage().currentKeyInfo.isLockOwner == 1 || CommonDataManage().currentKeyInfo.keyRight == 1,
|
2024-06-15 17:20:55 +08:00
|
|
|
|
child: GestureDetector(
|
|
|
|
|
|
child: Image.asset(
|
|
|
|
|
|
'images/icon_tips_Q.png',
|
|
|
|
|
|
width: 34.w,
|
|
|
|
|
|
height: 32.w,
|
|
|
|
|
|
color: Colors.white,
|
|
|
|
|
|
),
|
|
|
|
|
|
onTap: () {
|
2025-09-06 14:50:08 +08:00
|
|
|
|
ShowTipView().showSureAlertDialog('1.锁没有联网,密码、IC卡、指纹等开门记录无法实时上传,可以点击右上角按钮,然后读取记录。'.tr + '\n' + '2.如果您需要保留历史记录,可以点击右上角按钮,然后导出记录'.tr,
|
|
|
|
|
|
tipTitle: '看不到操作记录,可能原因有'.tr, sureStr: '我知道了'.tr);
|
2024-06-15 17:20:55 +08:00
|
|
|
|
},
|
|
|
|
|
|
)),
|
|
|
|
|
|
Visibility(
|
2025-09-06 14:50:08 +08:00
|
|
|
|
visible: CommonDataManage().currentKeyInfo.isLockOwner == 1 || CommonDataManage().currentKeyInfo.keyRight == 1,
|
2024-06-15 17:20:55 +08:00
|
|
|
|
child: PopupMenuButton<String>(
|
|
|
|
|
|
onSelected: _onMenuItemSelected,
|
|
|
|
|
|
color: Colors.black,
|
|
|
|
|
|
itemBuilder: (BuildContext context) {
|
|
|
|
|
|
return <PopupMenuEntry<String>>[
|
2024-08-21 18:31:19 +08:00
|
|
|
|
_buildCustomPopupMenuItem('读取记录'.tr),
|
2025-09-06 14:50:08 +08:00
|
|
|
|
if (CommonDataManage().currentKeyInfo.isLockOwner == 1) const PopupMenuDivider(),
|
|
|
|
|
|
if (CommonDataManage().currentKeyInfo.isLockOwner == 1) _buildCustomPopupMenuItem('清空记录'.tr),
|
2024-06-15 17:20:55 +08:00
|
|
|
|
const PopupMenuDivider(),
|
2024-08-21 18:31:19 +08:00
|
|
|
|
_buildCustomPopupMenuItem('导出记录'.tr),
|
2024-06-15 17:20:55 +08:00
|
|
|
|
];
|
|
|
|
|
|
},
|
|
|
|
|
|
icon: Image.asset(
|
|
|
|
|
|
'images/icon_bar_more.png',
|
|
|
|
|
|
height: 30.h,
|
|
|
|
|
|
width: 10.w,
|
|
|
|
|
|
),
|
|
|
|
|
|
offset: Offset(0, 70.h), // 设置弹出框位置偏移量
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
2024-05-16 11:59:37 +08:00
|
|
|
|
],
|
2024-01-09 14:05:18 +08:00
|
|
|
|
),
|
2024-01-18 11:24:12 +08:00
|
|
|
|
body: Column(
|
|
|
|
|
|
mainAxisSize: MainAxisSize.max,
|
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
2025-09-06 14:50:08 +08:00
|
|
|
|
children: <Widget>[topAdvancedCalendarWidget(), eventDropDownWidget(), Expanded(child: timeLineView())],
|
2024-01-11 09:41:19 +08:00
|
|
|
|
),
|
2025-08-19 10:16:20 +08:00
|
|
|
|
floatingActionButton: Visibility(
|
|
|
|
|
|
visible: _isAtBottom,
|
|
|
|
|
|
child: FloatingActionButton(
|
|
|
|
|
|
onPressed: () {
|
|
|
|
|
|
_scrollController.animateTo(
|
|
|
|
|
|
0.0,
|
|
|
|
|
|
duration: const Duration(milliseconds: 300),
|
|
|
|
|
|
curve: Curves.easeOut,
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
shape: RoundedRectangleBorder(
|
|
|
|
|
|
borderRadius: BorderRadius.circular(48.w),
|
|
|
|
|
|
),
|
|
|
|
|
|
backgroundColor: AppColors.mainColor,
|
|
|
|
|
|
child: Icon(
|
|
|
|
|
|
Icons.arrow_upward,
|
|
|
|
|
|
color: Colors.white,
|
|
|
|
|
|
size: 48.w,
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
2024-01-11 09:41:19 +08:00
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-06-15 17:20:55 +08:00
|
|
|
|
//
|
|
|
|
|
|
PopupMenuItem<String> _buildCustomPopupMenuItem(String value) {
|
|
|
|
|
|
return PopupMenuItem<String>(
|
|
|
|
|
|
value: value,
|
|
|
|
|
|
height: 46.h,
|
|
|
|
|
|
child: SizedBox(
|
|
|
|
|
|
height: 46.h,
|
|
|
|
|
|
child: Row(
|
|
|
|
|
|
mainAxisAlignment: MainAxisAlignment.center,
|
|
|
|
|
|
children: <Widget>[
|
2025-01-22 13:53:17 +08:00
|
|
|
|
Text(
|
|
|
|
|
|
value,
|
|
|
|
|
|
style: TextStyle(color: Colors.white, fontSize: 22.sp),
|
|
|
|
|
|
),
|
2024-06-15 17:20:55 +08:00
|
|
|
|
],
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void _onMenuItemSelected(String value) {
|
2024-12-04 15:21:27 +08:00
|
|
|
|
if (value == '读取记录'.tr) {
|
2025-01-11 18:55:40 +08:00
|
|
|
|
logic.getLockRecordLastUploadDataTime();
|
2024-12-04 15:21:27 +08:00
|
|
|
|
} else if (value == '清空记录'.tr) {
|
|
|
|
|
|
ShowCupertinoAlertView().showClearOperationRecordAlert(clearClick: () {
|
|
|
|
|
|
logic.clearOperationRecordRequest();
|
|
|
|
|
|
});
|
|
|
|
|
|
} else if (value == '导出记录'.tr) {
|
2024-08-21 18:31:19 +08:00
|
|
|
|
showDialog(
|
|
|
|
|
|
context: context,
|
|
|
|
|
|
builder: (BuildContext context) {
|
|
|
|
|
|
return ExportRecordDialog(
|
|
|
|
|
|
onExport: (String filePath) {
|
2025-09-06 14:50:08 +08:00
|
|
|
|
Get.toNamed(Routers.exportSuccessPage, arguments: <String, String>{'filePath': filePath});
|
2024-06-19 11:02:21 +08:00
|
|
|
|
},
|
|
|
|
|
|
);
|
2024-08-21 18:31:19 +08:00
|
|
|
|
},
|
|
|
|
|
|
);
|
2024-06-15 17:20:55 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-01-10 14:46:51 +08:00
|
|
|
|
|
2024-01-25 17:46:46 +08:00
|
|
|
|
//顶部日历小部件
|
|
|
|
|
|
Widget topAdvancedCalendarWidget() {
|
2025-08-18 14:48:24 +08:00
|
|
|
|
return Container(
|
|
|
|
|
|
margin: EdgeInsets.only(top: 20.h, left: 30.w, bottom: 10.h, right: 20.w),
|
|
|
|
|
|
child: Column(
|
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
|
children: [
|
|
|
|
|
|
_buildWeekCalendar(),
|
2024-05-16 10:19:01 +08:00
|
|
|
|
],
|
2024-01-25 17:46:46 +08:00
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//事件下拉框组件
|
|
|
|
|
|
Widget eventDropDownWidget() {
|
2024-01-26 16:19:01 +08:00
|
|
|
|
return Container(
|
2024-02-27 13:52:49 +08:00
|
|
|
|
margin: EdgeInsets.only(top: 20.h, left: 30.w, bottom: 10.h, right: 20.w),
|
2024-01-26 16:19:01 +08:00
|
|
|
|
child: Row(
|
|
|
|
|
|
mainAxisAlignment: MainAxisAlignment.start,
|
2024-06-15 17:20:55 +08:00
|
|
|
|
children: <Widget>[
|
2025-01-22 13:53:17 +08:00
|
|
|
|
Obx(
|
|
|
|
|
|
() => XSDropDownWidget(
|
2024-01-26 16:19:01 +08:00
|
|
|
|
items: state.getDropDownItemList,
|
2024-02-01 10:52:23 +08:00
|
|
|
|
value: state.dropdownTitle.value,
|
2025-01-22 13:53:17 +08:00
|
|
|
|
valueChanged: (value) async {
|
2024-02-01 10:52:23 +08:00
|
|
|
|
state.dropdownValue.value = int.parse(value);
|
2025-01-22 13:53:17 +08:00
|
|
|
|
await logic.mockNetworkDataRequest(isRefresh: true);
|
|
|
|
|
|
},
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
2024-01-26 16:19:01 +08:00
|
|
|
|
],
|
2024-01-25 17:46:46 +08:00
|
|
|
|
),
|
2024-01-26 16:19:01 +08:00
|
|
|
|
);
|
2024-01-25 17:46:46 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
//时间轴组件
|
2024-05-16 11:59:37 +08:00
|
|
|
|
Widget timeLineView() {
|
2024-01-25 17:46:46 +08:00
|
|
|
|
return Container(
|
2024-01-26 14:28:02 +08:00
|
|
|
|
margin: EdgeInsets.only(left: 20.w, right: 20.w, bottom: 20.h, top: 20.h),
|
2024-01-25 17:46:46 +08:00
|
|
|
|
decoration: BoxDecoration(
|
2024-02-26 17:56:30 +08:00
|
|
|
|
color: Colors.white,
|
|
|
|
|
|
borderRadius: BorderRadius.circular(16.w),
|
|
|
|
|
|
),
|
2025-08-18 18:17:41 +08:00
|
|
|
|
child: Obx(
|
2025-08-19 10:16:20 +08:00
|
|
|
|
() => state.lockLogItemList.isNotEmpty
|
|
|
|
|
|
? Timeline.tileBuilder(
|
|
|
|
|
|
controller: _scrollController,
|
|
|
|
|
|
builder: _timelineBuilderWidget(),
|
|
|
|
|
|
theme: TimelineThemeData(
|
|
|
|
|
|
nodePosition: 0.04, //居左侧距离
|
|
|
|
|
|
connectorTheme: const ConnectorThemeData(
|
|
|
|
|
|
thickness: 1.0,
|
|
|
|
|
|
color: AppColors.greyLineColor,
|
|
|
|
|
|
indent: 0.5,
|
2025-01-22 13:53:17 +08:00
|
|
|
|
),
|
2025-08-19 10:16:20 +08:00
|
|
|
|
indicatorTheme: const IndicatorThemeData(
|
|
|
|
|
|
size: 8.0,
|
|
|
|
|
|
color: AppColors.greyLineColor,
|
|
|
|
|
|
position: 0.4,
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
)
|
|
|
|
|
|
: NoData(),
|
2025-08-18 18:17:41 +08:00
|
|
|
|
),
|
2024-05-16 11:59:37 +08:00
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-18 18:17:41 +08:00
|
|
|
|
String formatTimestampToDateTimeYYYYMMDD(int timestampMs) {
|
|
|
|
|
|
DateTime dateTime = DateTime.fromMillisecondsSinceEpoch(timestampMs);
|
2025-09-06 14:50:08 +08:00
|
|
|
|
DateFormat formatter = DateFormat('MM${'月'.tr}dd${'日'.tr}'); // 格式:2025-08-18 14:30
|
2025-08-18 18:17:41 +08:00
|
|
|
|
return formatter.format(dateTime);
|
|
|
|
|
|
}
|
2025-08-04 15:24:40 +08:00
|
|
|
|
|
2025-08-18 18:17:41 +08:00
|
|
|
|
String formatTimestampToHHmm(int timestampMs) {
|
2025-08-04 15:24:40 +08:00
|
|
|
|
// 2. 创建 DateTime 对象
|
|
|
|
|
|
DateTime dateTime = DateTime.fromMillisecondsSinceEpoch(timestampMs);
|
|
|
|
|
|
|
|
|
|
|
|
// 3. 使用 DateFormat 格式化为 "HH:mm"
|
|
|
|
|
|
DateFormat formatter = DateFormat('HH:mm');
|
|
|
|
|
|
return formatter.format(dateTime);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-26 11:16:50 +08:00
|
|
|
|
bool _checkIsVideoOrImagesType(DoorLockLogDataItem item) {
|
|
|
|
|
|
final recordType = item.recordType;
|
|
|
|
|
|
switch (recordType) {
|
|
|
|
|
|
case 130:
|
|
|
|
|
|
case 220:
|
|
|
|
|
|
return true;
|
|
|
|
|
|
default:
|
|
|
|
|
|
return false;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-04 15:24:40 +08:00
|
|
|
|
String _buildIDByType(DoorLockLogDataItem item) {
|
|
|
|
|
|
final recordType = item.recordType;
|
|
|
|
|
|
switch (recordType) {
|
|
|
|
|
|
case 10:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '指纹'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 20:
|
|
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' +
|
|
|
|
|
|
'密码'.tr +
|
|
|
|
|
|
'开锁'.tr +
|
2025-09-06 14:50:08 +08:00
|
|
|
|
'(${_formatUserNameOrId(item)})' +
|
2025-08-18 14:48:24 +08:00
|
|
|
|
'(${'密码'.tr}:${item.keyboardPwd})';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 30:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '卡'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 40:
|
2025-08-04 18:02:14 +08:00
|
|
|
|
if (item.username != null && item.username != '') {
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '蓝牙'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
} else {
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '蓝牙'.tr + '开锁'.tr + '(' + 'ID'.tr + ':${item.userid})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
}
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 50:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '组合模式'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
case 60:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '添加'.tr + '指纹'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
case 70:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '添加'.tr + '密码'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
case 80:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '添加'.tr + '卡'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
case 90:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '删除'.tr + '指纹'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
case 100:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '删除'.tr + '密码'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
case 110:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '删除'.tr + '卡'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 18:02:14 +08:00
|
|
|
|
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 160:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '人脸'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 190:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '胁迫指纹'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 200:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '胁迫密码'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
case 210:
|
2025-09-06 14:50:08 +08:00
|
|
|
|
return '${formatTimestampToHHmm(item.operateDate!)} ' + '胁迫卡片'.tr + '开锁'.tr + '(${_formatUserNameOrId(item)})';
|
2025-08-04 15:24:40 +08:00
|
|
|
|
default:
|
|
|
|
|
|
return item.recordStr ?? '';
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-06 14:50:08 +08:00
|
|
|
|
// 提取为私有方法,更清晰
|
|
|
|
|
|
String _formatUserNameOrId(DoorLockLogDataItem item) {
|
|
|
|
|
|
final recordStr = item.recordStr;
|
|
|
|
|
|
final idMatch = RegExp(r'ID[::](\w+)').firstMatch(recordStr ?? '');
|
|
|
|
|
|
final idValue = idMatch?.group(1) ?? '';
|
|
|
|
|
|
|
|
|
|
|
|
final name = item.username ?? '';
|
|
|
|
|
|
|
|
|
|
|
|
// 如果用户名不为空,并且是以 ID: 或 ID: 开头的格式,则直接返回 ID:xxx
|
|
|
|
|
|
if (name.isNotEmpty) {
|
|
|
|
|
|
final idInNameMatch = RegExp(r'^ID[::].+').hasMatch(name);
|
|
|
|
|
|
if (idInNameMatch) {
|
|
|
|
|
|
return name; // 直接返回,例如 "ID:123"
|
|
|
|
|
|
} else {
|
|
|
|
|
|
return '${'昵称'.tr}:$name'; // 正常用户昵称,加前缀
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 如果用户名为空,使用 recordStr 中提取的 ID
|
|
|
|
|
|
return 'ID:$idValue';
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-08-05 14:46:11 +08:00
|
|
|
|
Color _buildTextColorByType(DoorLockLogDataItem item) {
|
|
|
|
|
|
final recordType = item.recordType;
|
|
|
|
|
|
switch (recordType) {
|
|
|
|
|
|
case 120:
|
|
|
|
|
|
case 150:
|
|
|
|
|
|
case 130:
|
|
|
|
|
|
case 190:
|
|
|
|
|
|
case 200:
|
|
|
|
|
|
case 210:
|
|
|
|
|
|
case 220:
|
|
|
|
|
|
return Colors.red;
|
|
|
|
|
|
default:
|
|
|
|
|
|
return Colors.black;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-05-16 11:59:37 +08:00
|
|
|
|
TimelineTileBuilder _timelineBuilderWidget() {
|
|
|
|
|
|
return TimelineTileBuilder.fromStyle(
|
|
|
|
|
|
contentsAlign: ContentsAlign.basic,
|
|
|
|
|
|
itemCount: state.lockLogItemList.length,
|
2024-06-15 17:20:55 +08:00
|
|
|
|
contentsBuilder: (BuildContext context, int index) {
|
|
|
|
|
|
final DoorLockLogDataItem timelineData = state.lockLogItemList[index];
|
2025-08-26 11:16:50 +08:00
|
|
|
|
// 👇 提前计算第一个有 videoUrl 的索引(可以在 build 外层计算一次,避免重复)
|
2025-09-06 14:50:08 +08:00
|
|
|
|
int? firstVideoIndex = state.lockLogItemList.indexWhere((item) => _checkIsVideoOrImagesType(item));
|
2025-08-26 11:16:50 +08:00
|
|
|
|
bool isInvalid = _checkIsVideoOrImagesType(timelineData) &&
|
2025-09-06 14:50:08 +08:00
|
|
|
|
((timelineData.imagesUrl == null && timelineData.videoUrl == null) || (timelineData.videoUrl == '' && timelineData.imagesUrl == ''));
|
2025-08-28 13:59:22 +08:00
|
|
|
|
|
|
|
|
|
|
String typeText = '';
|
|
|
|
|
|
if (timelineData.recordType == 130) {
|
|
|
|
|
|
typeText = '图像'.tr;
|
|
|
|
|
|
} else if (timelineData.recordType == 220) {
|
|
|
|
|
|
typeText = '视频'.tr;
|
|
|
|
|
|
}
|
2024-07-17 16:29:25 +08:00
|
|
|
|
return GestureDetector(
|
|
|
|
|
|
onTap: () {
|
2025-01-22 13:53:17 +08:00
|
|
|
|
Get.toNamed(
|
|
|
|
|
|
Routers.doorLockLogDetailPage,
|
|
|
|
|
|
arguments: {'doorLockLogDataItem': timelineData},
|
|
|
|
|
|
);
|
2024-07-17 16:29:25 +08:00
|
|
|
|
},
|
|
|
|
|
|
child: Padding(
|
2025-08-05 14:46:11 +08:00
|
|
|
|
padding: EdgeInsets.only(left: 20.w, top: 20.h, right: 20.w),
|
2024-07-17 16:29:25 +08:00
|
|
|
|
child: Column(
|
|
|
|
|
|
mainAxisAlignment: MainAxisAlignment.start,
|
|
|
|
|
|
crossAxisAlignment: CrossAxisAlignment.start,
|
|
|
|
|
|
children: <Widget>[
|
2025-09-06 14:50:08 +08:00
|
|
|
|
Text('${formatTimestampToDateTimeYYYYMMDD(timelineData.operateDate!)}',
|
2025-08-18 18:17:41 +08:00
|
|
|
|
style: TextStyle(
|
|
|
|
|
|
fontSize: 20.sp,
|
|
|
|
|
|
)),
|
2025-08-04 18:02:14 +08:00
|
|
|
|
// 使用 SingleChildScrollView 实现横向滚动
|
|
|
|
|
|
SingleChildScrollView(
|
|
|
|
|
|
scrollDirection: Axis.horizontal, // 横向滚动
|
2025-08-26 11:16:50 +08:00
|
|
|
|
child: RichText(
|
2025-08-04 18:02:14 +08:00
|
|
|
|
textAlign: TextAlign.left,
|
2025-08-26 11:16:50 +08:00
|
|
|
|
text: TextSpan(
|
|
|
|
|
|
style: TextStyle(
|
|
|
|
|
|
color: _buildTextColorByType(timelineData),
|
|
|
|
|
|
fontSize: 24.sp,
|
|
|
|
|
|
fontWeight: FontWeight.w600,
|
|
|
|
|
|
),
|
|
|
|
|
|
children: [
|
|
|
|
|
|
TextSpan(
|
2025-09-06 14:50:08 +08:00
|
|
|
|
text: _buildIDByType(timelineData) + (isInvalid ? '(${typeText}' + '已失效'.tr + ')' : ''),
|
2025-08-26 11:16:50 +08:00
|
|
|
|
),
|
|
|
|
|
|
WidgetSpan(
|
|
|
|
|
|
alignment: PlaceholderAlignment.middle,
|
|
|
|
|
|
child: Visibility(
|
|
|
|
|
|
visible: isInvalid,
|
|
|
|
|
|
child: Icon(
|
|
|
|
|
|
Icons.error,
|
|
|
|
|
|
size: 24.sp,
|
|
|
|
|
|
color: Colors.red,
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
],
|
2025-08-04 18:02:14 +08:00
|
|
|
|
),
|
|
|
|
|
|
maxLines: 1,
|
|
|
|
|
|
overflow: TextOverflow.ellipsis,
|
2024-07-17 16:29:25 +08:00
|
|
|
|
),
|
2024-05-16 11:59:37 +08:00
|
|
|
|
),
|
2024-07-17 16:29:25 +08:00
|
|
|
|
SizedBox(
|
2025-01-22 13:53:17 +08:00
|
|
|
|
height: 12.h,
|
|
|
|
|
|
),
|
|
|
|
|
|
videoItem(
|
|
|
|
|
|
RecordListData(
|
|
|
|
|
|
recordId: state.lockLogItemList.value[index].recordId,
|
|
|
|
|
|
recordType: state.lockLogItemList.value[index].recordType,
|
|
|
|
|
|
operateDate: state.lockLogItemList.value[index].operateDate,
|
|
|
|
|
|
imagesUrl: state.lockLogItemList.value[index].imagesUrl,
|
|
|
|
|
|
videoUrl: state.lockLogItemList.value[index].videoUrl,
|
|
|
|
|
|
),
|
2024-07-17 16:29:25 +08:00
|
|
|
|
),
|
|
|
|
|
|
SizedBox(
|
2025-08-26 11:16:50 +08:00
|
|
|
|
height: 12.h,
|
2024-07-17 16:29:25 +08:00
|
|
|
|
),
|
2025-08-26 11:16:50 +08:00
|
|
|
|
Visibility(
|
2025-09-06 14:50:08 +08:00
|
|
|
|
visible: _checkIsVideoOrImagesType(timelineData) && index == firstVideoIndex,
|
2025-08-26 11:16:50 +08:00
|
|
|
|
child: GestureDetector(
|
|
|
|
|
|
onTap: () async {
|
|
|
|
|
|
await logic.getWebPlayUrl();
|
|
|
|
|
|
},
|
|
|
|
|
|
child: Container(
|
|
|
|
|
|
padding: EdgeInsets.all(8.w),
|
|
|
|
|
|
decoration: BoxDecoration(
|
|
|
|
|
|
borderRadius: BorderRadius.all(Radius.circular(8.r)),
|
|
|
|
|
|
),
|
|
|
|
|
|
child: RichText(
|
|
|
|
|
|
textAlign: TextAlign.center,
|
|
|
|
|
|
text: TextSpan(
|
|
|
|
|
|
children: [
|
|
|
|
|
|
// 普通文字部分
|
|
|
|
|
|
TextSpan(
|
|
|
|
|
|
text:
|
|
|
|
|
|
'${'您的图像和视频数据仅保留'.tr} ${state.rollingStorageDays.value} ${'天'.tr} ,${state.rollingStorageDays.value} ${'天'.tr} ${'后图像和视频数据将会失效,开通'.tr}',
|
|
|
|
|
|
style: TextStyle(
|
|
|
|
|
|
color: Colors.grey,
|
|
|
|
|
|
fontSize: 16.sp,
|
|
|
|
|
|
fontWeight: FontWeight.w600,
|
|
|
|
|
|
height: 1.8,
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
// 🔥 高亮且可点击的“云存会员”文本
|
|
|
|
|
|
TextSpan(
|
|
|
|
|
|
text: '云存会员'.tr,
|
|
|
|
|
|
style: TextStyle(
|
|
|
|
|
|
color: AppColors.mainColor,
|
|
|
|
|
|
fontSize: 22.sp,
|
|
|
|
|
|
fontWeight: FontWeight.w800,
|
|
|
|
|
|
// 更粗
|
|
|
|
|
|
decoration: TextDecoration.underline,
|
|
|
|
|
|
decorationThickness: 1.5,
|
|
|
|
|
|
height: 1.8,
|
|
|
|
|
|
),
|
|
|
|
|
|
recognizer: TapGestureRecognizer()
|
|
|
|
|
|
..onTap = () async {
|
|
|
|
|
|
// 👉 点击回调:跳转或弹窗
|
|
|
|
|
|
print('点击了“云存会员”');
|
|
|
|
|
|
await logic.getWebPlayUrl();
|
|
|
|
|
|
// 例如:Navigator.push(context, MaterialPageRoute(builder: ...));
|
|
|
|
|
|
},
|
|
|
|
|
|
),
|
|
|
|
|
|
// 后续文字
|
|
|
|
|
|
TextSpan(
|
|
|
|
|
|
text: '服务,图像视频信息随心存!'.tr,
|
|
|
|
|
|
style: TextStyle(
|
|
|
|
|
|
color: Colors.grey,
|
|
|
|
|
|
fontSize: 16.sp,
|
|
|
|
|
|
fontWeight: FontWeight.w600,
|
|
|
|
|
|
height: 1.8,
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
],
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
)
|
2024-07-17 16:29:25 +08:00
|
|
|
|
],
|
|
|
|
|
|
),
|
2024-05-16 11:59:37 +08:00
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
},
|
2024-01-25 17:46:46 +08:00
|
|
|
|
);
|
2024-01-11 09:41:19 +08:00
|
|
|
|
}
|
2024-04-28 15:00:13 +08:00
|
|
|
|
|
2025-01-10 14:46:51 +08:00
|
|
|
|
Widget videoItem(RecordListData recordData) {
|
|
|
|
|
|
return GestureDetector(
|
|
|
|
|
|
onTap: () {
|
|
|
|
|
|
if (recordData.videoUrl != null && recordData.videoUrl!.isNotEmpty) {
|
|
|
|
|
|
final lockLogItemList = state.lockLogItemList.value;
|
|
|
|
|
|
final list = lockLogItemList
|
2025-09-06 14:50:08 +08:00
|
|
|
|
.where((e) => (e.videoUrl != null && e.videoUrl!.isNotEmpty) || (e.imagesUrl != null && e.imagesUrl!.isNotEmpty))
|
2025-01-22 13:53:17 +08:00
|
|
|
|
.map(
|
|
|
|
|
|
(e) => RecordListData(
|
|
|
|
|
|
videoUrl: e.videoUrl,
|
|
|
|
|
|
imagesUrl: e.imagesUrl,
|
|
|
|
|
|
operateDate: e.operateDate,
|
|
|
|
|
|
recordId: e.recordId,
|
|
|
|
|
|
recordType: e.recordType,
|
|
|
|
|
|
),
|
|
|
|
|
|
)
|
2025-01-10 14:46:51 +08:00
|
|
|
|
.toList();
|
2025-09-06 14:50:08 +08:00
|
|
|
|
final selectDateString = DateTool().dateToYMDString(state.startDate.value.toString());
|
|
|
|
|
|
final cloudStorageData = CloudStorageData(date: selectDateString, recordList: list);
|
2025-01-10 14:46:51 +08:00
|
|
|
|
Get.toNamed(Routers.videoLogDetailPage, arguments: <String, Object>{
|
|
|
|
|
|
'recordData': recordData,
|
|
|
|
|
|
'videoDataList': [cloudStorageData]
|
|
|
|
|
|
});
|
2025-09-06 14:50:08 +08:00
|
|
|
|
} else if (recordData.imagesUrl != null && recordData.imagesUrl!.isNotEmpty) {
|
2025-01-10 14:46:51 +08:00
|
|
|
|
Navigator.push(
|
|
|
|
|
|
context,
|
|
|
|
|
|
MaterialPageRoute(
|
|
|
|
|
|
builder: (context) => FullScreenImagePage(
|
|
|
|
|
|
imageUrl: recordData.imagesUrl!,
|
|
|
|
|
|
),
|
|
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
},
|
2025-09-06 14:50:08 +08:00
|
|
|
|
child: ((recordData.imagesUrl != null && recordData.imagesUrl != '') || (recordData.videoUrl != null && recordData.videoUrl != ''))
|
2025-01-22 13:53:17 +08:00
|
|
|
|
? Container(
|
2025-01-10 14:46:51 +08:00
|
|
|
|
width: 260.w,
|
|
|
|
|
|
height: 260.h,
|
|
|
|
|
|
margin: const EdgeInsets.all(0),
|
|
|
|
|
|
color: Colors.white,
|
|
|
|
|
|
child: ClipRRect(
|
|
|
|
|
|
borderRadius: BorderRadius.circular(10.w),
|
|
|
|
|
|
child: _buildImageOrVideoItem(recordData),
|
|
|
|
|
|
),
|
2025-01-22 13:53:17 +08:00
|
|
|
|
)
|
|
|
|
|
|
: SizedBox.shrink(),
|
2025-01-10 14:46:51 +08:00
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_buildImageOrVideoItem(RecordListData recordData) {
|
|
|
|
|
|
if (recordData.videoUrl != null && recordData.videoUrl!.isNotEmpty) {
|
|
|
|
|
|
return _buildVideoItem(recordData);
|
2025-09-06 14:50:08 +08:00
|
|
|
|
} else if (recordData.imagesUrl != null && recordData.imagesUrl!.isNotEmpty) {
|
2025-01-10 14:46:51 +08:00
|
|
|
|
return _buildImageItem(recordData);
|
2025-01-22 13:53:17 +08:00
|
|
|
|
} else {
|
|
|
|
|
|
return SizedBox.shrink();
|
2025-01-10 14:46:51 +08:00
|
|
|
|
}
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_buildVideoItem(RecordListData recordData) {
|
2025-01-21 14:09:09 +08:00
|
|
|
|
return VideoThumbnailImage(videoUrl: recordData.videoUrl!);
|
2025-01-10 14:46:51 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
_buildImageItem(RecordListData recordData) {
|
2025-01-21 10:24:47 +08:00
|
|
|
|
return RotatedBox(
|
|
|
|
|
|
quarterTurns: -1,
|
|
|
|
|
|
child: Image.network(
|
|
|
|
|
|
recordData.imagesUrl!,
|
|
|
|
|
|
fit: BoxFit.cover,
|
2025-09-06 14:50:08 +08:00
|
|
|
|
errorBuilder: (BuildContext context, Object error, StackTrace? stackTrace) {
|
2025-01-21 10:24:47 +08:00
|
|
|
|
// 图片加载失败时显示错误图片
|
2025-01-22 13:53:17 +08:00
|
|
|
|
return Image.asset(
|
|
|
|
|
|
'images/icon_unHaveData.png', // 错误图片路径
|
|
|
|
|
|
fit: BoxFit.cover,
|
2025-01-21 10:24:47 +08:00
|
|
|
|
);
|
|
|
|
|
|
},
|
|
|
|
|
|
),
|
2025-01-10 14:46:51 +08:00
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2024-04-28 15:00:13 +08:00
|
|
|
|
@override
|
|
|
|
|
|
void didChangeDependencies() {
|
|
|
|
|
|
super.didChangeDependencies();
|
|
|
|
|
|
|
|
|
|
|
|
/// 路由订阅
|
|
|
|
|
|
AppRouteObserver().routeObserver.subscribe(this, ModalRoute.of(context)!);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
|
void dispose() {
|
|
|
|
|
|
/// 取消路由订阅
|
|
|
|
|
|
AppRouteObserver().routeObserver.unsubscribe(this);
|
|
|
|
|
|
super.dispose();
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 从上级界面进入 当前界面即将出现
|
|
|
|
|
|
@override
|
|
|
|
|
|
void didPush() {
|
|
|
|
|
|
super.didPush();
|
|
|
|
|
|
state.ifCurrentScreen.value = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 返回上一个界面 当前界面即将消失
|
|
|
|
|
|
@override
|
|
|
|
|
|
void didPop() {
|
|
|
|
|
|
super.didPop();
|
|
|
|
|
|
logic.cancelBlueConnetctToastTimer();
|
2024-10-31 17:09:06 +08:00
|
|
|
|
if (EasyLoading.isShow) {
|
|
|
|
|
|
EasyLoading.dismiss(animation: true);
|
|
|
|
|
|
}
|
2024-04-28 15:00:13 +08:00
|
|
|
|
state.ifCurrentScreen.value = false;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 从下级返回 当前界面即将出现
|
|
|
|
|
|
@override
|
|
|
|
|
|
void didPopNext() {
|
|
|
|
|
|
super.didPopNext();
|
|
|
|
|
|
state.ifCurrentScreen.value = true;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// 进入下级界面 当前界面即将消失
|
|
|
|
|
|
@override
|
|
|
|
|
|
void didPushNext() {
|
|
|
|
|
|
super.didPushNext();
|
|
|
|
|
|
logic.cancelBlueConnetctToastTimer();
|
2024-06-15 17:20:55 +08:00
|
|
|
|
if (EasyLoading.isShow) {
|
|
|
|
|
|
EasyLoading.dismiss(animation: true);
|
|
|
|
|
|
}
|
2024-04-28 15:00:13 +08:00
|
|
|
|
state.ifCurrentScreen.value = false;
|
|
|
|
|
|
}
|
2025-08-18 14:48:24 +08:00
|
|
|
|
|
|
|
|
|
|
List<DateTime> getCurrentWeekDates() {
|
|
|
|
|
|
final now = DateTime.now();
|
|
|
|
|
|
// weekday: 1=周一, 2=周二, ..., 7=周日
|
|
|
|
|
|
// 计算距离上一个周日相差的天数
|
|
|
|
|
|
// 如果今天是周日,weekday == 7,偏移为 0
|
|
|
|
|
|
final int daysSinceSunday = now.weekday % 7; // 周一=1 -> %7=1, 周日=7 -> %7=0
|
|
|
|
|
|
|
|
|
|
|
|
final List<DateTime> weekDates = [];
|
|
|
|
|
|
for (int i = 0; i < 7; i++) {
|
|
|
|
|
|
final DateTime day = DateTime(
|
|
|
|
|
|
now.year,
|
|
|
|
|
|
now.month,
|
|
|
|
|
|
now.day - daysSinceSunday + i, // 从周日开始累加
|
|
|
|
|
|
);
|
|
|
|
|
|
weekDates.add(day);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return weekDates;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Widget _buildWeekCalendar() {
|
|
|
|
|
|
return Obx(() {
|
2025-08-18 18:17:41 +08:00
|
|
|
|
final list = state.weekEventList.value;
|
2025-08-18 14:48:24 +08:00
|
|
|
|
final dateSet = list
|
|
|
|
|
|
.map((e) => DateTime.fromMillisecondsSinceEpoch(e.operateDate!))
|
|
|
|
|
|
.map((dt) => dt.withoutTime) // 转为年月日
|
|
|
|
|
|
.toSet(); // 用 Set 提升查找性能
|
|
|
|
|
|
AppLog.log('dateSet:${dateSet}');
|
|
|
|
|
|
return WeekCalendarView(
|
|
|
|
|
|
hasData: (DateTime date) {
|
|
|
|
|
|
return dateSet.contains(date.withoutTime);
|
|
|
|
|
|
},
|
2025-08-18 18:17:41 +08:00
|
|
|
|
onDateSelected: (DateTime date) async {
|
|
|
|
|
|
print('外部收到选中: $date');
|
|
|
|
|
|
|
|
|
|
|
|
state.operateDate = date.millisecondsSinceEpoch;
|
2025-09-06 14:50:08 +08:00
|
|
|
|
state.startDate.value = DateTime(date.year, date.month, date.day).millisecondsSinceEpoch;
|
|
|
|
|
|
state.endDate.value = DateTime(date.year, date.month, date.day, 23, 59, 59, 999).millisecondsSinceEpoch;
|
2025-08-18 18:17:41 +08:00
|
|
|
|
await logic.mockNetworkDataRequest(isRefresh: true);
|
|
|
|
|
|
},
|
2025-08-18 14:48:24 +08:00
|
|
|
|
onWeekChanged: (DateTime start, DateTime end) {
|
2025-08-18 18:17:41 +08:00
|
|
|
|
state.startDate.value = start.millisecondsSinceEpoch;
|
|
|
|
|
|
state.endDate.value = end.millisecondsSinceEpoch;
|
|
|
|
|
|
logic.mockNetworkDataRequest(isRefresh: true);
|
2025-08-18 14:48:24 +08:00
|
|
|
|
},
|
|
|
|
|
|
);
|
|
|
|
|
|
});
|
|
|
|
|
|
}
|
2024-01-09 14:05:18 +08:00
|
|
|
|
}
|