import 'package:flutter/material.dart'; import 'package:mileograph_flutter/components/legs/leg_card.dart'; import 'package:mileograph_flutter/objects/objects.dart'; import 'package:mileograph_flutter/services/data_service.dart'; import 'package:provider/provider.dart'; class LegsPage extends StatefulWidget { const LegsPage({super.key}); @override State createState() => _LegsPageState(); } class _LegsPageState extends State { int _sortDirection = 0; DateTime? _startDate; DateTime? _endDate; bool _initialised = false; @override void didChangeDependencies() { super.didChangeDependencies(); if (!_initialised) { _initialised = true; _refreshLegs(); } } Future _refreshLegs() async { final data = context.read(); await data.fetchLegs( sortDirection: _sortDirection, dateRangeStart: _formatDate(_startDate), dateRangeEnd: _formatDate(_endDate), ); } Future _loadMore() async { final data = context.read(); await data.fetchLegs( sortDirection: _sortDirection, dateRangeStart: _formatDate(_startDate), dateRangeEnd: _formatDate(_endDate), offset: data.legs.length, append: true, ); } double _pageMileage(List legs) { return legs.fold( 0, (prev, leg) => prev + (leg.mileage as double? ?? 0), ); } Future _pickDate({required bool start}) async { final initial = start ? _startDate ?? DateTime.now() : _endDate ?? _startDate ?? DateTime.now(); final picked = await showDatePicker( context: context, initialDate: initial, firstDate: DateTime(1970), lastDate: DateTime.now().add(const Duration(days: 365)), ); if (picked != null) { setState(() { if (start) { _startDate = picked; if (_endDate != null && _endDate!.isBefore(picked)) { _endDate = picked; } } else { _endDate = picked; } }); await _refreshLegs(); } } void _clearFilters() { setState(() { _startDate = null; _endDate = null; _sortDirection = 0; }); _refreshLegs(); } @override Widget build(BuildContext context) { final data = context.watch(); final legs = data.legs; final pageMileage = _pageMileage(legs); return RefreshIndicator( onRefresh: _refreshLegs, child: ListView( padding: const EdgeInsets.all(16), physics: const AlwaysScrollableScrollPhysics(), children: [ Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text('Logbook', style: Theme.of(context).textTheme.labelMedium), const SizedBox(height: 2), Text('Entries', style: Theme.of(context).textTheme.headlineSmall), ], ), Card( child: Padding( padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 8), child: Column( crossAxisAlignment: CrossAxisAlignment.end, children: [ Text('Page mileage', style: Theme.of(context).textTheme.labelSmall), Text('${pageMileage.toStringAsFixed(1)} mi', style: Theme.of(context) .textTheme .titleMedium ?.copyWith(fontWeight: FontWeight.w700)), ], ), ), ), ], ), const SizedBox(height: 12), Card( child: Padding( padding: const EdgeInsets.all(12), child: Column( crossAxisAlignment: CrossAxisAlignment.stretch, children: [ Row( mainAxisAlignment: MainAxisAlignment.spaceBetween, children: [ Text('Filters', style: Theme.of(context).textTheme.titleMedium), TextButton.icon( onPressed: _clearFilters, icon: const Icon(Icons.refresh), label: const Text('Clear'), ), ], ), const SizedBox(height: 8), Wrap( spacing: 12, runSpacing: 12, crossAxisAlignment: WrapCrossAlignment.center, children: [ FilledButton.tonalIcon( onPressed: () => _pickDate(start: true), icon: const Icon(Icons.calendar_month), label: Text( _startDate == null ? 'Start date' : _formatDate(_startDate!)!, ), ), FilledButton.tonalIcon( onPressed: () => _pickDate(start: false), icon: const Icon(Icons.event), label: Text( _endDate == null ? 'End date' : _formatDate(_endDate!)!, ), ), ], ), ], ), ), ), const SizedBox(height: 12), if (data.isLegsLoading && legs.isEmpty) const Center( child: Padding( padding: EdgeInsets.symmetric(vertical: 24.0), child: CircularProgressIndicator(), ), ) else if (legs.isEmpty) Card( child: Padding( padding: const EdgeInsets.all(16.0), child: Column( crossAxisAlignment: CrossAxisAlignment.start, children: [ Text( 'No entries found', style: Theme.of(context) .textTheme .titleMedium ?.copyWith(fontWeight: FontWeight.w700), ), const SizedBox(height: 8), const Text('Adjust the filters or add a new leg.'), ], ), ), ) else Column( children: [ ..._buildLegsWithDividers(context, legs), const SizedBox(height: 8), if (data.legsHasMore || data.isLegsLoading) Align( alignment: Alignment.center, child: OutlinedButton.icon( onPressed: data.isLegsLoading ? null : () => _loadMore(), icon: data.isLegsLoading ? const SizedBox( height: 14, width: 14, child: CircularProgressIndicator(strokeWidth: 2), ) : const Icon(Icons.expand_more), label: Text( data.isLegsLoading ? 'Loading...' : 'Load more', ), ), ), ], ), ], ), ); } List _buildLegsWithDividers(BuildContext context, List legs) { final widgets = []; String? currentDate; double dayMileage = 0; final dayLegs = []; void flushDay() { if (currentDate == null) return; widgets.add( Padding( padding: const EdgeInsets.symmetric(vertical: 8.0), child: Row( children: [ Expanded( child: Text( currentDate!, style: Theme.of(context).textTheme.labelMedium?.copyWith( fontWeight: FontWeight.w700, ), ), ), Text( '${dayMileage.toStringAsFixed(1)} mi', style: Theme.of(context).textTheme.labelMedium, ), ], ), ), ); widgets.add(const Divider()); widgets.addAll( dayLegs.map((leg) => LegCard(leg: leg, showDate: false)), ); dayLegs.clear(); } for (final leg in legs) { final dateStr = _formatDate(leg.beginTime) ?? ''; if (currentDate != null && dateStr != currentDate) { flushDay(); dayMileage = 0; } currentDate = dateStr; dayLegs.add(leg); dayMileage += leg.mileage; } flushDay(); return widgets; } String? _formatDate(DateTime? date) { if (date == null) return null; return '${date.year.toString().padLeft(4, '0')}-${date.month.toString().padLeft(2, '0')}-${date.day.toString().padLeft(2, '0')}'; } }