周报月报卡片弹窗

This commit is contained in:
czz
2025-08-14 17:48:14 +08:00
parent e5672d5fd6
commit aed666ea74
8 changed files with 433 additions and 15 deletions

View File

@@ -376,7 +376,7 @@
"入睡时间": "Sleep time",
"睡眠时长": "Sleep duration",
"起床时间": "Wake time",
"小时": "hours",
"小时": "h",
"分钟": "minutes",
"消息提醒设置": "Message notification settings",
"APP消息": "APP messages",
@@ -503,5 +503,8 @@
"本月睡眠时长": "Monthly Sleep Duration",
"知道了": "Back",
"本月睡眠时长是指从月初到月末,用户每天实际睡眠的时间总和": "The monthly sleep duration refers to the total actual sleep time of the user from the beginning to the end of the month.",
"选择生日": "Select Birthday"
"选择生日": "Select Birthday",
"sleep_duration": "Sleep Duration",
"deep_sleep": "Deep Sleep",
"light_sleep": "Light Sleep"
}

View File

@@ -515,5 +515,9 @@
"睡眠分数与上周分数进行对比,是通过量化分析近期睡眠质量变化,可了解自身睡眠状态的波动情况,进而调整用户的作息习惯。": "睡眠分数与上周分数进行对比,是通过量化分析近期睡眠质量变化,可了解自身睡眠状态的波动情况,进而调整用户的作息习惯。",
"本月睡眠时长": "本月睡眠时长",
"知道了": "返回",
"本月睡眠时长是指从月初到月末,用户每天实际睡眠的时间总和": "本月睡眠时长是指从月初到月末,用户每天实际睡眠的时间总和", "选择生日": "选择生日"
"本月睡眠时长是指从月初到月末,用户每天实际睡眠的时间总和": "本月睡眠时长是指从月初到月末,用户每天实际睡眠的时间总和",
"选择生日": "选择生日",
"sleep_duration": "睡眠时长",
"deep_sleep": "深睡",
"light_sleep": "浅睡"
}

View File

@@ -0,0 +1,252 @@
import 'package:ef/ef.dart';
import 'package:flutter/material.dart';
import 'package:flutter_svg/svg.dart';
import 'package:flutterflow_ui/flutterflow_ui.dart';
import 'package:vbvs_app/common/color/appConstants.dart';
import 'package:vbvs_app/common/util/FitTool.dart';
import 'package:vbvs_app/common/util/MyUtils.dart';
import 'package:vbvs_app/component/tool/ClickableContainer.dart';
import 'package:vbvs_app/controller/sleep/sleep_report_controller.dart';
import 'package:vbvs_app/controller/theme_controller/ThemeController.dart';
import 'package:vbvs_app/pages/device_bind/componnet/bind_dialog.dart';
class WeekSleepDataModule extends StatefulWidget {
final Map<String, dynamic> data;
final dynamic sleepReportData; // 可选参数,类型为 var/dynamic
const WeekSleepDataModule({
super.key,
required this.data,
this.sleepReportData, // 标记为可选参数
});
@override
State<WeekSleepDataModule> createState() => _WeekSleepDataModuleState();
}
class _WeekSleepDataModuleState extends State<WeekSleepDataModule> {
@override
void setState(VoidCallback callback) {
super.setState(callback);
}
@override
void initState() {
super.initState();
}
@override
void dispose() {
super.dispose();
}
@override
Widget build(BuildContext context) {
ThemeController themeController = Get.find();
return ClickableContainer(
backgroundColor: themeController.currentColor.sc5,
highlightColor: themeController.currentColor.sc21,
borderRadius: 20.rpx,
padding: EdgeInsetsDirectional.fromSTEB(18.rpx, 10.rpx, 18.rpx, 10.rpx),
onTap: () {
showTipDialog(
backgroundColor: stringToColor("#FFFFFF"),
context,
Container(
constraints: BoxConstraints(
maxHeight: 700.rpx,
),
child: SingleChildScrollView(
child: Column(
children: [
Text(
"${widget.data['name']}",
style: TextStyle(
color: stringToColor("#333333"),
fontSize: 36.rpx,
),
),
SizedBox(
height: 17.rpx,
),
// Text(
// (widget.data['tips']?.toString().trim().isNotEmpty ?? false)
// ? widget.data['tips'].toString()
// : "未知数据".tr,
// style: TextStyle(
// color: stringToColor("#C8CBD2"),
// fontSize: 26.rpx,
// ),
// ),
// SizedBox(
// height: 37.rpx,
// ),
Text(
"${widget.data['value']}" +
((widget.data['unit'] == null ||
widget.data['unit'].toString().isEmpty)
? ''
: widget.data['unit']),
style: TextStyle(
color: stringToColor("${widget.data['color']}"),
fontSize: 60.rpx,
),
),
// SizedBox(
// height: 81.rpx,
// ),
// IntrinsicHeight(
// child: Row(
// crossAxisAlignment: CrossAxisAlignment.stretch,
// children: [
// for (int i = 0; i < levelGroups.length; i++) ...[
// // 每个 levelGroup 区域
// Expanded(
// child: Column(
// mainAxisSize: MainAxisSize.min,
// children: [
// // Level 名称
// Text(
// levelGroups[i]['levelName'],
// style: TextStyle(
// fontSize: 30.rpx,
// color: stringToColor("#333333"),
// fontWeight: FontWeight.bold,
// ),
// ),
// SizedBox(height: 38.rpx),
],
),
),
),
);
if (widget.data['onto'] != null && widget.data['onto'] == true) {
//跳转睡眠报告
Get.toNamed("/newSleepReportPage", arguments: {
'date': widget.data['time'] != null
? int.parse(widget.data['time'].toString())
: DateTime.now().millisecondsSinceEpoch,
"mac": widget.data['mac'] != null && widget.data['mac'].isNotEmpty
? widget.data['mac']
: 'aaaaaaeeeeeq',
'type': 1,
'name': 'sleep', //'sleep', 'heartRate' 或 'breathe'
'itemName': widget.data['id'],
'person': widget.data['person'],
});
}
},
child: Container(
width: MediaQuery.sizeOf(context).width * 0.267,
constraints: BoxConstraints(
minWidth: 200.rpx,
minHeight: 161.rpx,
),
child: Column(
mainAxisSize: MainAxisSize.max,
crossAxisAlignment: CrossAxisAlignment.start,
mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [
Text(
'${widget.data['name']}',
style: TextStyle(
fontFamily: 'Inter',
fontSize: 26.rpx,
letterSpacing: 0.0,
color: themeController.currentColor.sc3,
),
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
Row(
mainAxisSize: MainAxisSize.min,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Expanded(
child: Text.rich(
TextSpan(
children: [
TextSpan(
text: '${widget.data['value']}',
style: TextStyle(
fontFamily: 'Inter',
fontSize: 36.rpx,
letterSpacing: 0.0,
color: themeController.currentColor.sc3,
),
),
WidgetSpan(child: SizedBox(width: 2.rpx)), // 可选间距
TextSpan(
text: widget.data['unit'] != null
? '${widget.data['unit']}'
: '',
style: TextStyle(
fontFamily: 'Inter',
fontSize: AppConstants().small_text_fontSize,
letterSpacing: 0.0,
color: themeController.currentColor.sc3,
),
),
],
),
maxLines: 1,
style: TextStyle(
color:
themeController.currentColor.sc3), // 强制 ellipsis 颜色
overflow: TextOverflow.ellipsis,
),
),
if (widget.data['level'] != null)
ClickableContainer(
backgroundColor: (widget.data['color'] == null ||
widget.data['color'].toString().isEmpty)
? Colors.transparent
: stringToColor(widget.data['color']),
highlightColor: themeController.currentColor.sc3,
padding: EdgeInsets.symmetric(
horizontal: 0.rpx,
vertical: 0.rpx,
),
borderRadius: 8.rpx,
onTap: () {
print('Button pressed ...');
},
child: Container(
alignment: Alignment.center,
constraints: BoxConstraints(
minWidth: 43.rpx,
minHeight: 25.rpx,
),
child: Text(
'${widget.data['level']}',
style: TextStyle(
fontFamily: 'Inter Tight',
color: themeController.currentColor.sc3,
letterSpacing: 0.0,
fontSize: 15.rpx,
),
),
),
),
].divide(SizedBox(width: 0.rpx)),
),
Text(
"正常值".tr +
'${(widget.data['range'] ?? '').toString().isEmpty ? '未知数据'.tr : widget.data['range']}',
style: TextStyle(
fontFamily: 'Inter',
fontSize: AppConstants().small_text_fontSize,
letterSpacing: 0.0,
color: themeController.currentColor.sc4,
),
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
],
),
),
);
}
}

View File

@@ -7,6 +7,7 @@ import 'package:vbvs_app/pages/sleep_report/component/SleepCard.dart';
import 'package:vbvs_app/pages/sleep_report/component/SleepChartWidget.dart';
import 'package:vbvs_app/pages/sleep_report/component/TrendDataTablePage.dart';
import 'package:vbvs_app/pages/sleep_report/component/TrendDataTextPage.dart';
import 'package:vbvs_app/pages/sleep_report/component/WeekSleepCard.dart';
import 'package:vbvs_app/pages/sleep_report/component/WeekSleepScoreWidget.dart';
Widget MonthDataWidget(
@@ -88,7 +89,7 @@ Widget MonthDataWidget(
),
showLabel: sleepReport['scoreList']['type'],
),
SleepCard(
WeekSleepCard(
sleepReport: sleepReport,
highlightItem: data['itemName'],
),

View File

@@ -7,6 +7,7 @@ import 'package:vbvs_app/pages/sleep_report/component/SleepCard.dart';
import 'package:vbvs_app/pages/sleep_report/component/SleepChartWidget.dart';
import 'package:vbvs_app/pages/sleep_report/component/TrendDataTablePage.dart';
import 'package:vbvs_app/pages/sleep_report/component/TrendDataTextPage.dart';
import 'package:vbvs_app/pages/sleep_report/component/WeekSleepCard.dart';
import 'package:vbvs_app/pages/sleep_report/component/WeekSleepScoreWidget.dart';
@@ -109,7 +110,7 @@ Widget WeekDataWidget(
),
showLabel: sleepReport['scoreList']['type'],
),
SleepCard(
WeekSleepCard(
sleepReport: sleepReport,
highlightItem: data['itemName'],
),

View File

@@ -0,0 +1,159 @@
import 'package:flutter/material.dart';
import 'package:vbvs_app/common/color/appConstants.dart';
import 'package:vbvs_app/common/util/FitTool.dart';
import 'package:vbvs_app/common/util/MyUtils.dart';
import 'package:vbvs_app/component/home_page/SleepDataModuleWidget.dart';
import 'package:EasyDartModule/EasyDartModule.dart' as es;
import 'package:vbvs_app/component/home_page/WeekSleepDataModule.dart';
import 'package:vbvs_app/enum/APPPackageType.dart';
import 'package:vbvs_app/language/AppLanguage.dart';
class WeekSleepCard extends StatefulWidget {
final dynamic sleepReport;
final int? highlightItem;
WeekSleepCard({super.key, required this.sleepReport, this.highlightItem});
@override
State<WeekSleepCard> createState() => _WeekSleepCardState();
}
class _WeekSleepCardState extends State<WeekSleepCard>
with TickerProviderStateMixin {
final GlobalKey _highlightKey = GlobalKey();
AnimationController? _animationController;
bool _shouldAnimate = false;
int? _highlightedId;
int _flashCount = 0; // 用于跟踪闪烁次数
@override
void initState() {
super.initState();
if (widget.highlightItem != null) {
_highlightedId = widget.highlightItem;
_shouldAnimate = true;
_initAnimation();
}
WidgetsBinding.instance.addPostFrameCallback((_) {
if (widget.highlightItem != null &&
_highlightKey.currentContext != null) {
Scrollable.ensureVisible(
_highlightKey.currentContext!,
duration: Duration(milliseconds: 500),
curve: Curves.easeInOut,
alignment: 0.3,
);
}
});
}
void _initAnimation() {
_animationController = AnimationController(
vsync: this,
duration: Duration(milliseconds: 300),
)..addStatusListener((status) {
if (status == AnimationStatus.completed) {
// 正向动画完成,开始反向动画
_animationController!.reverse();
} else if (status == AnimationStatus.dismissed) {
// 反向动画完成,增加计数
_flashCount++;
// 闪烁3次后停止
if (_flashCount >= 5) {
_animationController!.dispose();
setState(() {
_shouldAnimate = false;
_highlightedId = null;
});
} else {
// 继续下一次闪烁
_animationController!.forward();
}
}
});
_animationController!.forward();
}
@override
void dispose() {
_animationController?.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
try {
if (widget.sleepReport == null ||
widget.sleepReport is! Map ||
widget.sleepReport.isEmpty) {
return Container();
}
String? language = "";
if (AppConstants().ent_type == APPPackageType.MHT.code) {
if (mhLanguageController.selectLanguage != null) {
language = mhLanguageController.selectLanguage.value!.language_code;
}
} else {
if (languageController.selectLanguage != null) {
language = languageController.selectLanguage.value!.language_code;
}
}
int num = AppLanguage().isChinese() ? 3 : 2;
List data = widget.sleepReport['bs'] ?? [];
return Container(
width: double.infinity,
decoration: BoxDecoration(
color: themeController.currentColor.sc5,
borderRadius:
BorderRadius.circular(AppConstants().normal_container_radius),
),
child: Padding(
padding:
EdgeInsetsDirectional.fromSTEB(26.rpx, 29.rpx, 26.rpx, 45.rpx),
child: Wrap(
spacing: 23.rpx,
runSpacing: 25.rpx,
children: List.generate(data.length, (index) {
final item = data[index];
item['showTip'] = true;
final bool isHighlighted =
_shouldAnimate && item['id'] == _highlightedId;
return SizedBox(
width: (MediaQuery.of(context).size.width - 160.rpx) / num,
child: AnimatedBuilder(
animation: _animationController ?? AlwaysStoppedAnimation(0),
builder: (context, child) {
return Container(
key: isHighlighted ? _highlightKey : null,
decoration: isHighlighted
? BoxDecoration(
border: Border.all(
color: themeController.currentColor.sc2
.withOpacity(
_animationController?.value ?? 0),
width: 1.rpx,
),
borderRadius: BorderRadius.circular(8),
)
: null,
child: WeekSleepDataModule(
data: item,
sleepReportData: widget.sleepReport,
),
);
},
),
);
}),
),
),
);
} catch (e) {
es.EasyDartModule.logger.error("数据卡片渲染异常${e}");
return Container();
}
}
}

View File

@@ -108,16 +108,9 @@ class AvgSleepScoreWidget extends StatelessWidget {
centerWidget: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
// Text(
// mediumLabel.tr,
// style: TextStyle(
// color: Colors.white,
// fontSize: AppConstants().normal_text_fontSize,
// ),
// ),
Padding(
padding:
EdgeInsets.only(top: 12.rpx), // 👈 向下偏移的关键
EdgeInsets.only(top: 15.rpx), // 👈 向下偏移的关键
child: Text(
mediumLabel.tr,
style: TextStyle(

View File

@@ -907,10 +907,15 @@ class _NewSleepReportPageState extends State<NewSleepReportPage> {
breatheCardKey,
widget.data);
case 2:
return WeekDataWidget(sleepReport, widget.data);
return WeekDataWidget(
sleepReport,
widget.data,
);
case 3:
return MonthDataWidget(
sleepReport, widget.data);
sleepReport,
widget.data,
);
default:
return NullDataWidget();
}