157 lines
4.8 KiB
Dart
157 lines
4.8 KiB
Dart
import 'package:flutter/material.dart';
|
||
import 'package:fl_chart/fl_chart.dart';
|
||
|
||
/// 学习趋势图表
|
||
class LearningTrendChart extends StatelessWidget {
|
||
final List<Map<String, dynamic>> weeklyData;
|
||
|
||
const LearningTrendChart({
|
||
super.key,
|
||
required this.weeklyData,
|
||
});
|
||
|
||
@override
|
||
Widget build(BuildContext context) {
|
||
if (weeklyData.isEmpty) {
|
||
return const Center(
|
||
child: Text(
|
||
'暂无学习数据',
|
||
style: TextStyle(color: Colors.grey),
|
||
),
|
||
);
|
||
}
|
||
|
||
return Container(
|
||
height: 200,
|
||
padding: const EdgeInsets.all(16),
|
||
child: LineChart(
|
||
LineChartData(
|
||
gridData: FlGridData(
|
||
show: true,
|
||
drawVerticalLine: false,
|
||
horizontalInterval: 20,
|
||
getDrawingHorizontalLine: (value) {
|
||
return FlLine(
|
||
color: Colors.grey.withOpacity(0.1),
|
||
strokeWidth: 1,
|
||
);
|
||
},
|
||
),
|
||
titlesData: FlTitlesData(
|
||
show: true,
|
||
rightTitles: const AxisTitles(
|
||
sideTitles: SideTitles(showTitles: false),
|
||
),
|
||
topTitles: const AxisTitles(
|
||
sideTitles: SideTitles(showTitles: false),
|
||
),
|
||
bottomTitles: AxisTitles(
|
||
sideTitles: SideTitles(
|
||
showTitles: true,
|
||
reservedSize: 30,
|
||
interval: 1,
|
||
getTitlesWidget: (value, meta) {
|
||
final index = value.toInt();
|
||
if (index < 0 || index >= weeklyData.length) {
|
||
return const Text('');
|
||
}
|
||
|
||
final date = DateTime.parse(weeklyData[index]['date']);
|
||
final dayLabel = ['周一', '周二', '周三', '周四', '周五', '周六', '周日'][date.weekday - 1];
|
||
|
||
return Text(
|
||
dayLabel,
|
||
style: const TextStyle(
|
||
color: Colors.grey,
|
||
fontSize: 10,
|
||
),
|
||
);
|
||
},
|
||
),
|
||
),
|
||
leftTitles: AxisTitles(
|
||
sideTitles: SideTitles(
|
||
showTitles: true,
|
||
interval: 20,
|
||
reservedSize: 35,
|
||
getTitlesWidget: (value, meta) {
|
||
return Text(
|
||
value.toInt().toString(),
|
||
style: const TextStyle(
|
||
color: Colors.grey,
|
||
fontSize: 10,
|
||
),
|
||
);
|
||
},
|
||
),
|
||
),
|
||
),
|
||
borderData: FlBorderData(
|
||
show: true,
|
||
border: Border(
|
||
bottom: BorderSide(color: Colors.grey.withOpacity(0.2)),
|
||
left: BorderSide(color: Colors.grey.withOpacity(0.2)),
|
||
),
|
||
),
|
||
minX: 0,
|
||
maxX: (weeklyData.length - 1).toDouble(),
|
||
minY: 0,
|
||
maxY: _getMaxY(),
|
||
lineBarsData: [
|
||
LineChartBarData(
|
||
spots: _generateSpots(),
|
||
isCurved: true,
|
||
gradient: const LinearGradient(
|
||
colors: [Color(0xFF2196F3), Color(0xFF1976D2)],
|
||
),
|
||
barWidth: 3,
|
||
isStrokeCapRound: true,
|
||
dotData: FlDotData(
|
||
show: true,
|
||
getDotPainter: (spot, percent, barData, index) {
|
||
return FlDotCirclePainter(
|
||
radius: 4,
|
||
color: Colors.white,
|
||
strokeWidth: 2,
|
||
strokeColor: const Color(0xFF2196F3),
|
||
);
|
||
},
|
||
),
|
||
belowBarData: BarAreaData(
|
||
show: true,
|
||
gradient: LinearGradient(
|
||
colors: [
|
||
const Color(0xFF2196F3).withOpacity(0.2),
|
||
const Color(0xFF2196F3).withOpacity(0.05),
|
||
],
|
||
begin: Alignment.topCenter,
|
||
end: Alignment.bottomCenter,
|
||
),
|
||
),
|
||
),
|
||
],
|
||
),
|
||
),
|
||
);
|
||
}
|
||
|
||
List<FlSpot> _generateSpots() {
|
||
return List.generate(
|
||
weeklyData.length,
|
||
(index) {
|
||
final wordsStudied = (weeklyData[index]['words_studied'] ?? 0) as int;
|
||
return FlSpot(index.toDouble(), wordsStudied.toDouble());
|
||
},
|
||
);
|
||
}
|
||
|
||
double _getMaxY() {
|
||
if (weeklyData.isEmpty) return 100;
|
||
|
||
final maxWords = weeklyData.map((data) => (data['words_studied'] ?? 0) as int).reduce((a, b) => a > b ? a : b);
|
||
|
||
// 向上取整到最近的10的倍数,并加20作为上边距
|
||
return ((maxWords / 10).ceil() * 10 + 20).toDouble();
|
||
}
|
||
}
|