reports page m4 ui done

This commit is contained in:
2026-02-18 18:44:16 +05:30
parent 93f2de2ab6
commit 917b4e213c
9 changed files with 410 additions and 437 deletions

View File

@@ -90,6 +90,7 @@ class ReportsRepositoryImpl implements ReportsRepository {
final List<SpendInvoice> spendInvoices = []; final List<SpendInvoice> spendInvoices = [];
final Map<DateTime, double> dailyAggregates = {}; final Map<DateTime, double> dailyAggregates = {};
final Map<String, double> industryAggregates = {};
for (final inv in invoices) { for (final inv in invoices) {
final amount = (inv.amount ?? 0.0).toDouble(); final amount = (inv.amount ?? 0.0).toDouble();
@@ -104,6 +105,9 @@ class ReportsRepositoryImpl implements ReportsRepository {
overdueInvoices++; overdueInvoices++;
} }
final industry = inv.vendor?.serviceSpecialty ?? 'Other';
industryAggregates[industry] = (industryAggregates[industry] ?? 0.0) + amount;
final issueDateTime = inv.issueDate.toDateTime(); final issueDateTime = inv.issueDate.toDateTime();
spendInvoices.add(SpendInvoice( spendInvoices.add(SpendInvoice(
id: inv.id, id: inv.id,
@@ -112,6 +116,7 @@ class ReportsRepositoryImpl implements ReportsRepository {
amount: amount, amount: amount,
status: statusStr, status: statusStr,
vendorName: inv.vendor?.companyName ?? 'Unknown', vendorName: inv.vendor?.companyName ?? 'Unknown',
industry: industry,
)); ));
// Chart data aggregation // Chart data aggregation
@@ -119,19 +124,40 @@ class ReportsRepositoryImpl implements ReportsRepository {
dailyAggregates[date] = (dailyAggregates[date] ?? 0.0) + amount; dailyAggregates[date] = (dailyAggregates[date] ?? 0.0) + amount;
} }
final List<SpendChartPoint> chartData = dailyAggregates.entries // Ensure chart data covers all days in range
final Map<DateTime, double> completeDailyAggregates = {};
for (int i = 0; i <= endDate.difference(startDate).inDays; i++) {
final date = startDate.add(Duration(days: i));
final normalizedDate = DateTime(date.year, date.month, date.day);
completeDailyAggregates[normalizedDate] =
dailyAggregates[normalizedDate] ?? 0.0;
}
final List<SpendChartPoint> chartData = completeDailyAggregates.entries
.map((e) => SpendChartPoint(date: e.key, amount: e.value)) .map((e) => SpendChartPoint(date: e.key, amount: e.value))
.toList() .toList()
..sort((a, b) => a.date.compareTo(b.date)); ..sort((a, b) => a.date.compareTo(b.date));
final List<SpendIndustryCategory> industryBreakdown = industryAggregates.entries
.map((e) => SpendIndustryCategory(
name: e.key,
amount: e.value,
percentage: totalSpend > 0 ? (e.value / totalSpend * 100) : 0,
))
.toList()
..sort((a, b) => b.amount.compareTo(a.amount));
final daysCount = endDate.difference(startDate).inDays + 1;
return SpendReport( return SpendReport(
totalSpend: totalSpend, totalSpend: totalSpend,
averageCost: invoices.isEmpty ? 0 : totalSpend / invoices.length, averageCost: daysCount > 0 ? totalSpend / daysCount : 0,
paidInvoices: paidInvoices, paidInvoices: paidInvoices,
pendingInvoices: pendingInvoices, pendingInvoices: pendingInvoices,
overdueInvoices: overdueInvoices, overdueInvoices: overdueInvoices,
invoices: spendInvoices, invoices: spendInvoices,
chartData: chartData, chartData: chartData,
industryBreakdown: industryBreakdown,
); );
}); });
} }

View File

@@ -34,6 +34,7 @@ class DailyOpsShift extends Equatable {
final int workersNeeded; final int workersNeeded;
final int filled; final int filled;
final String status; final String status;
final double? hourlyRate;
const DailyOpsShift({ const DailyOpsShift({
required this.id, required this.id,
@@ -44,6 +45,7 @@ class DailyOpsShift extends Equatable {
required this.workersNeeded, required this.workersNeeded,
required this.filled, required this.filled,
required this.status, required this.status,
this.hourlyRate,
}); });
@override @override
@@ -56,5 +58,6 @@ class DailyOpsShift extends Equatable {
workersNeeded, workersNeeded,
filled, filled,
status, status,
hourlyRate,
]; ];
} }

View File

@@ -17,8 +17,11 @@ class SpendReport extends Equatable {
required this.overdueInvoices, required this.overdueInvoices,
required this.invoices, required this.invoices,
required this.chartData, required this.chartData,
required this.industryBreakdown,
}); });
final List<SpendIndustryCategory> industryBreakdown;
@override @override
List<Object?> get props => [ List<Object?> get props => [
totalSpend, totalSpend,
@@ -28,9 +31,25 @@ class SpendReport extends Equatable {
overdueInvoices, overdueInvoices,
invoices, invoices,
chartData, chartData,
industryBreakdown,
]; ];
} }
class SpendIndustryCategory extends Equatable {
final String name;
final double amount;
final double percentage;
const SpendIndustryCategory({
required this.name,
required this.amount,
required this.percentage,
});
@override
List<Object?> get props => [name, amount, percentage];
}
class SpendInvoice extends Equatable { class SpendInvoice extends Equatable {
final String id; final String id;
final String invoiceNumber; final String invoiceNumber;
@@ -46,10 +65,13 @@ class SpendInvoice extends Equatable {
required this.amount, required this.amount,
required this.status, required this.status,
required this.vendorName, required this.vendorName,
this.industry,
}); });
final String? industry;
@override @override
List<Object?> get props => [id, invoiceNumber, issueDate, amount, status, vendorName]; List<Object?> get props => [id, invoiceNumber, issueDate, amount, status, vendorName, industry];
} }
class SpendChartPoint extends Equatable { class SpendChartPoint extends Equatable {

View File

@@ -56,7 +56,7 @@ class _CoverageReportPageState extends State<CoverageReportPage> {
top: 60, top: 60,
left: 20, left: 20,
right: 20, right: 20,
bottom: 32, bottom: 80, // Increased bottom padding for overlap background
), ),
decoration: const BoxDecoration( decoration: const BoxDecoration(
gradient: LinearGradient( gradient: LinearGradient(
@@ -160,53 +160,60 @@ class _CoverageReportPageState extends State<CoverageReportPage> {
), ),
], ],
), ),
const SizedBox(height: 24),
// ── 3 summary stat chips (matches prototype) ──
Row(
children: [
_HeaderStatChip(
icon: UiIcons.trendingUp,
label: 'Avg Coverage',
value:
'${report.overallCoverage.toStringAsFixed(0)}%',
),
const SizedBox(width: 12),
_HeaderStatChip(
icon: UiIcons.checkCircle,
label: 'Full',
value: fullDays.toString(),
),
const SizedBox(width: 12),
_HeaderStatChip(
icon: UiIcons.warning,
label: 'Needs Help',
value: needsHelpDays.toString(),
isAlert: needsHelpDays > 0,
),
],
),
], ],
), ),
), ),
// ── 3 summary stat chips (Moved here for overlap) ──
Transform.translate(
offset: const Offset(0, -60), // Pull up to overlap header
child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 20),
child: Row(
children: [
_CoverageStatCard(
icon: UiIcons.trendingUp,
label: 'Avg Coverage',
value: '${report.overallCoverage.toStringAsFixed(0)}%',
iconColor: UiColors.primary,
),
const SizedBox(width: 12),
_CoverageStatCard(
icon: UiIcons.checkCircle,
label: 'Full',
value: fullDays.toString(),
iconColor: UiColors.success,
),
const SizedBox(width: 12),
_CoverageStatCard(
icon: UiIcons.warning,
label: 'Needs Help',
value: needsHelpDays.toString(),
iconColor: UiColors.error,
),
],
),
),
),
// ── Content ────────────────────────────────────────── // ── Content ──────────────────────────────────────────
Transform.translate( Transform.translate(
offset: const Offset(0, -16), offset: const Offset(0, -60), // Pull up to overlap header
child: Padding( child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 20), padding: const EdgeInsets.symmetric(horizontal: 20),
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
const SizedBox(height: 32),
// Section label // Section label
const Text( const Text(
'NEXT 7 DAYS', 'NEXT 7 DAYS',
style: TextStyle( style: TextStyle(
fontSize: 11, fontSize: 14,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: UiColors.textSecondary, color: UiColors.textPrimary,
letterSpacing: 1.2, letterSpacing: 0.5,
), ),
), ),
const SizedBox(height: 16), const SizedBox(height: 16),
@@ -250,57 +257,68 @@ class _CoverageReportPageState extends State<CoverageReportPage> {
} }
// ── Header stat chip (inside the blue header) ───────────────────────────────── // ── Header stat chip (inside the blue header) ─────────────────────────────────
class _HeaderStatChip extends StatelessWidget { // ── Header stat card (boxes inside the blue header overlap) ───────────────────
class _CoverageStatCard extends StatelessWidget {
final IconData icon; final IconData icon;
final String label; final String label;
final String value; final String value;
final bool isAlert; final Color iconColor;
const _HeaderStatChip({ const _CoverageStatCard({
required this.icon, required this.icon,
required this.label, required this.label,
required this.value, required this.value,
this.isAlert = false, required this.iconColor,
}); });
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Expanded( return Expanded(
child: Container( child: Container(
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 10), padding: const EdgeInsets.all(16), // Increased padding
decoration: BoxDecoration( decoration: BoxDecoration(
color: UiColors.white.withOpacity(0.15), color: UiColors.white,
borderRadius: BorderRadius.circular(10), borderRadius: BorderRadius.circular(16), // More rounded
boxShadow: [
BoxShadow(
color: UiColors.black.withOpacity(0.04),
blurRadius: 10,
offset: const Offset(0, 4),
),
],
), ),
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
Row( Row(
children: [ children: [
Icon( Icon(
icon, icon,
size: 12, size: 14,
color: isAlert color: iconColor,
? const Color(0xFFFFD580)
: UiColors.white.withOpacity(0.8),
), ),
const SizedBox(width: 4), const SizedBox(width: 6),
Text( Expanded(
label, child: Text(
style: TextStyle( label,
fontSize: 10, style: const TextStyle(
color: UiColors.white.withOpacity(0.8), fontSize: 11,
color: UiColors.textSecondary,
fontWeight: FontWeight.w500,
),
overflow: TextOverflow.ellipsis,
), ),
), ),
], ],
), ),
const SizedBox(height: 4), const SizedBox(height: 8),
Text( Text(
value, value,
style: const TextStyle( style: const TextStyle(
fontSize: 20, fontSize: 20, // Slightly smaller to fit if needed
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: UiColors.white, color: UiColors.textPrimary,
), ),
), ),
], ],
@@ -344,7 +362,7 @@ class _DayCoverageCard extends StatelessWidget {
final badgeBg = percentage >= 95 final badgeBg = percentage >= 95
? UiColors.tagSuccess ? UiColors.tagSuccess
: percentage >= 80 : percentage >= 80
? UiColors.tagInProgress ? UiColors.primary.withOpacity(0.1) // Blue tint
: UiColors.tagError; : UiColors.tagError;
return Container( return Container(

View File

@@ -243,7 +243,7 @@ class _DailyOpsReportPageState extends State<DailyOpsReportPage> {
physics: const NeverScrollableScrollPhysics(), physics: const NeverScrollableScrollPhysics(),
mainAxisSpacing: 12, mainAxisSpacing: 12,
crossAxisSpacing: 12, crossAxisSpacing: 12,
childAspectRatio: 1.4, childAspectRatio: 1.2,
children: [ children: [
_OpsStatCard( _OpsStatCard(
label: context.t.client_reports label: context.t.client_reports
@@ -314,16 +314,16 @@ class _DailyOpsReportPageState extends State<DailyOpsReportPage> {
], ],
), ),
const SizedBox(height: 24), const SizedBox(height: 8),
Text( Text(
context.t.client_reports.daily_ops_report context.t.client_reports.daily_ops_report
.all_shifts_title .all_shifts_title
.toUpperCase(), .toUpperCase(),
style: const TextStyle( style: const TextStyle(
fontSize: 12, fontSize: 14,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: UiColors.textSecondary, color: UiColors.textPrimary,
letterSpacing: 1.2, letterSpacing: 0.5,
), ),
), ),
const SizedBox(height: 12), const SizedBox(height: 12),
@@ -341,9 +341,12 @@ class _DailyOpsReportPageState extends State<DailyOpsReportPage> {
title: shift.title, title: shift.title,
location: shift.location, location: shift.location,
time: time:
'${DateFormat('hh:mm a').format(shift.startTime)} - ${DateFormat('hh:mm a').format(shift.endTime)}', '${DateFormat('HH:mm').format(shift.startTime)} - ${DateFormat('HH:mm').format(shift.endTime)}',
workers: workers:
'${shift.filled}/${shift.workersNeeded}', '${shift.filled}/${shift.workersNeeded}',
rate: shift.hourlyRate != null
? '\$${shift.hourlyRate!.toStringAsFixed(0)}/hr'
: '-',
status: shift.status.replaceAll('_', ' '), status: shift.status.replaceAll('_', ' '),
statusColor: shift.status == 'COMPLETED' statusColor: shift.status == 'COMPLETED'
? UiColors.success ? UiColors.success
@@ -399,15 +402,15 @@ class _OpsStatCard extends StatelessWidget {
offset: const Offset(0, 2), offset: const Offset(0, 2),
), ),
], ],
border: Border(left: BorderSide(color: color, width: 4)),
), ),
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
Row( Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
Icon(icon, size: 14, color: color),
const SizedBox(width: 8),
Expanded( Expanded(
child: Text( child: Text(
label, label,
@@ -420,7 +423,6 @@ class _OpsStatCard extends StatelessWidget {
overflow: TextOverflow.ellipsis, overflow: TextOverflow.ellipsis,
), ),
), ),
Icon(icon, size: 14, color: color),
], ],
), ),
Column( Column(
@@ -429,16 +431,29 @@ class _OpsStatCard extends StatelessWidget {
Text( Text(
value, value,
style: const TextStyle( style: const TextStyle(
fontSize: 24, fontSize: 28,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: UiColors.textPrimary, color: UiColors.textPrimary,
), ),
), ),
Text( const SizedBox(height: 6),
subValue, // Colored pill badge (matches prototype)
style: const TextStyle( Container(
fontSize: 10, padding: const EdgeInsets.symmetric(
color: UiColors.textSecondary, horizontal: 8,
vertical: 3,
),
decoration: BoxDecoration(
color: color.withOpacity(0.12),
borderRadius: BorderRadius.circular(20),
),
child: Text(
subValue,
style: TextStyle(
fontSize: 10,
fontWeight: FontWeight.bold,
color: color,
),
), ),
), ),
], ],
@@ -454,6 +469,7 @@ class _ShiftListItem extends StatelessWidget {
final String location; final String location;
final String time; final String time;
final String workers; final String workers;
final String rate;
final String status; final String status;
final Color statusColor; final Color statusColor;
@@ -462,6 +478,7 @@ class _ShiftListItem extends StatelessWidget {
required this.location, required this.location,
required this.time, required this.time,
required this.workers, required this.workers,
required this.rate,
required this.status, required this.status,
required this.statusColor, required this.statusColor,
}); });
@@ -557,6 +574,11 @@ class _ShiftListItem extends StatelessWidget {
UiIcons.users, UiIcons.users,
context.t.client_reports.daily_ops_report.shift_item.workers, context.t.client_reports.daily_ops_report.shift_item.workers,
workers), workers),
_infoItem(
context,
UiIcons.trendingUp,
'Rate',
rate),
], ],
), ),
], ],

View File

@@ -296,7 +296,7 @@ class _SummaryChip extends StatelessWidget {
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Container( return Container(
padding: const EdgeInsets.symmetric(vertical: 16, horizontal: 12), padding: const EdgeInsets.symmetric(vertical: 14, horizontal: 12),
decoration: BoxDecoration( decoration: BoxDecoration(
color: UiColors.white, color: UiColors.white,
borderRadius: BorderRadius.circular(12), borderRadius: BorderRadius.circular(12),
@@ -311,24 +311,32 @@ class _SummaryChip extends StatelessWidget {
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Icon(icon, size: 16, color: iconColor), Row(
children: [
Icon(icon, size: 12, color: iconColor),
const SizedBox(width: 4),
Expanded(
child: Text(
label,
style: TextStyle(
fontSize: 10,
color: iconColor,
fontWeight: FontWeight.w600,
),
overflow: TextOverflow.ellipsis,
),
),
],
),
const SizedBox(height: 8), const SizedBox(height: 8),
Text( Text(
value, value,
style: const TextStyle( style: const TextStyle(
fontSize: 22, fontSize: 26,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: UiColors.textPrimary, color: UiColors.textPrimary,
), ),
), ),
const SizedBox(height: 2),
Text(
label,
style: const TextStyle(
fontSize: 11,
color: UiColors.textSecondary,
),
),
], ],
), ),
); );

View File

@@ -432,26 +432,27 @@ class _KpiRow extends StatelessWidget {
], ],
), ),
), ),
Column( // Value + badge inline (matches prototype)
crossAxisAlignment: CrossAxisAlignment.end, Row(
crossAxisAlignment: CrossAxisAlignment.center,
children: [ children: [
Text( Text(
kpi.displayValue, kpi.displayValue,
style: const TextStyle( style: const TextStyle(
fontSize: 15, fontSize: 16,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: UiColors.textPrimary, color: UiColors.textPrimary,
), ),
), ),
const SizedBox(height: 2), const SizedBox(width: 6),
Container( Container(
padding: const EdgeInsets.symmetric( padding: const EdgeInsets.symmetric(
horizontal: 8, horizontal: 7,
vertical: 3, vertical: 3,
), ),
decoration: BoxDecoration( decoration: BoxDecoration(
color: badgeBg, color: badgeBg,
borderRadius: BorderRadius.circular(10), borderRadius: BorderRadius.circular(8),
), ),
child: Text( child: Text(
badgeText, badgeText,
@@ -473,7 +474,7 @@ class _KpiRow extends StatelessWidget {
value: (kpi.value / 100).clamp(0.0, 1.0), value: (kpi.value / 100).clamp(0.0, 1.0),
backgroundColor: UiColors.bgSecondary, backgroundColor: UiColors.bgSecondary,
valueColor: AlwaysStoppedAnimation<Color>(kpi.barColor), valueColor: AlwaysStoppedAnimation<Color>(kpi.barColor),
minHeight: 5, minHeight: 6,
), ),
), ),
], ],

View File

@@ -8,6 +8,7 @@ import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:flutter_modular/flutter_modular.dart'; import 'package:flutter_modular/flutter_modular.dart';
import 'package:intl/intl.dart'; import 'package:intl/intl.dart';
import 'package:client_reports/src/domain/entities/spend_report.dart';
class SpendReportPage extends StatefulWidget { class SpendReportPage extends StatefulWidget {
const SpendReportPage({super.key}); const SpendReportPage({super.key});
@@ -17,8 +18,19 @@ class SpendReportPage extends StatefulWidget {
} }
class _SpendReportPageState extends State<SpendReportPage> { class _SpendReportPageState extends State<SpendReportPage> {
DateTime _startDate = DateTime.now().subtract(const Duration(days: 6)); late DateTime _startDate;
DateTime _endDate = DateTime.now(); late DateTime _endDate;
@override
void initState() {
super.initState();
final now = DateTime.now();
// Monday alignment logic
final diff = now.weekday - DateTime.monday;
final monday = now.subtract(Duration(days: diff));
_startDate = DateTime(monday.year, monday.month, monday.day);
_endDate = _startDate.add(const Duration(days: 6, hours: 23, minutes: 59, seconds: 59));
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
@@ -48,14 +60,10 @@ class _SpendReportPageState extends State<SpendReportPage> {
top: 60, top: 60,
left: 20, left: 20,
right: 20, right: 20,
bottom: 32, bottom: 80, // Overlap space
), ),
decoration: const BoxDecoration( decoration: const BoxDecoration(
gradient: LinearGradient( color: UiColors.primary, // Blue background per prototype
colors: [UiColors.success, UiColors.tagSuccess],
begin: Alignment.topLeft,
end: Alignment.bottomRight,
),
), ),
child: Row( child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween, mainAxisAlignment: MainAxisAlignment.spaceBetween,
@@ -128,7 +136,7 @@ class _SpendReportPageState extends State<SpendReportPage> {
const Icon( const Icon(
UiIcons.download, UiIcons.download,
size: 14, size: 14,
color: UiColors.success, color: UiColors.primary,
), ),
const SizedBox(width: 6), const SizedBox(width: 6),
Text( Text(
@@ -137,7 +145,7 @@ class _SpendReportPageState extends State<SpendReportPage> {
.split(' ') .split(' ')
.first, .first,
style: const TextStyle( style: const TextStyle(
color: UiColors.success, color: UiColors.primary,
fontSize: 12, fontSize: 12,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
), ),
@@ -152,50 +160,50 @@ class _SpendReportPageState extends State<SpendReportPage> {
// Content // Content
Transform.translate( Transform.translate(
offset: const Offset(0, -16), offset: const Offset(0, -60), // Pull up to overlap
child: Padding( child: Padding(
padding: const EdgeInsets.symmetric(horizontal: 20), padding: const EdgeInsets.symmetric(horizontal: 20),
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
// Summary Cards // Summary Cards (New Style)
Row( Row(
children: [ children: [
Expanded( Expanded(
child: _SpendSummaryCard( child: _SpendStatCard(
label: context.t.client_reports.spend_report label: context.t.client_reports.spend_report
.summary.total_spend, .summary.total_spend,
value: NumberFormat.currency(symbol: r'$') value: NumberFormat.currency(
symbol: r'$', decimalDigits: 0)
.format(report.totalSpend), .format(report.totalSpend),
change: '', // Can be calculated if needed pillText: context.t.client_reports
period: context.t.client_reports
.spend_report.summary.this_week, .spend_report.summary.this_week,
color: UiColors.textSuccess, themeColor: UiColors.success,
icon: UiIcons.dollar, icon: UiIcons.dollar,
), ),
), ),
const SizedBox(width: 12), const SizedBox(width: 12),
Expanded( Expanded(
child: _SpendSummaryCard( child: _SpendStatCard(
label: context.t.client_reports.spend_report label: context.t.client_reports.spend_report
.summary.avg_daily, .summary.avg_daily,
value: NumberFormat.currency(symbol: r'$') value: NumberFormat.currency(
symbol: r'$', decimalDigits: 0)
.format(report.averageCost), .format(report.averageCost),
change: '', pillText: context.t.client_reports
period: context.t.client_reports
.spend_report.summary.per_day, .spend_report.summary.per_day,
color: UiColors.primary, themeColor: UiColors.primary,
icon: UiIcons.chart, icon: UiIcons.trendingUp,
), ),
), ),
], ],
), ),
const SizedBox(height: 24), const SizedBox(height: 24),
// Chart Section // Daily Spend Trend Chart
Container( Container(
height: 300, height: 320,
padding: const EdgeInsets.all(16), padding: const EdgeInsets.all(20),
decoration: BoxDecoration( decoration: BoxDecoration(
color: UiColors.white, color: UiColors.white,
borderRadius: BorderRadius.circular(16), borderRadius: BorderRadius.circular(16),
@@ -210,50 +218,15 @@ class _SpendReportPageState extends State<SpendReportPage> {
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Row( const Text(
mainAxisAlignment: 'Daily Spend Trend',
MainAxisAlignment.spaceBetween, style: TextStyle(
children: [ fontSize: 14,
Text( fontWeight: FontWeight.bold,
context.t.client_reports.spend_report color: UiColors.textPrimary,
.chart_title, ),
style: const TextStyle(
fontSize: 14,
fontWeight: FontWeight.bold,
color: UiColors.textPrimary,
),
),
Container(
padding: const EdgeInsets.symmetric(
horizontal: 8,
vertical: 4,
),
decoration: BoxDecoration(
color: UiColors.bgSecondary,
borderRadius:
BorderRadius.circular(8),
),
child: Row(
children: [
Text(
context.t.client_reports.tabs.week,
style: const TextStyle(
fontSize: 10,
fontWeight: FontWeight.bold,
color: UiColors.textSecondary,
),
),
const Icon(
UiIcons.chevronDown,
size: 10,
color: UiColors.textSecondary,
),
],
),
),
],
), ),
const SizedBox(height: 24), const SizedBox(height: 32),
Expanded( Expanded(
child: _SpendBarChart( child: _SpendBarChart(
chartData: report.chartData), chartData: report.chartData),
@@ -263,71 +236,11 @@ class _SpendReportPageState extends State<SpendReportPage> {
), ),
const SizedBox(height: 24), const SizedBox(height: 24),
// Status Distribution
Row(
children: [
Expanded(
child: _StatusMiniCard(
label: 'Paid',
value: report.paidInvoices.toString(),
color: UiColors.success,
),
),
const SizedBox(width: 8),
Expanded(
child: _StatusMiniCard(
label: 'Pending',
value: report.pendingInvoices.toString(),
color: UiColors.textWarning,
),
),
const SizedBox(width: 8),
Expanded(
child: _StatusMiniCard(
label: 'Overdue',
value: report.overdueInvoices.toString(),
color: UiColors.error,
),
),
],
),
const SizedBox(height: 32), // Spend by Industry
Text( _SpendByIndustryCard(
'RECENT INVOICES', industries: report.industryBreakdown,
style: const TextStyle(
fontSize: 12,
fontWeight: FontWeight.bold,
color: UiColors.textSecondary,
letterSpacing: 1.2,
),
), ),
const SizedBox(height: 16),
// Invoice List
if (report.invoices.isEmpty)
const Padding(
padding: EdgeInsets.symmetric(vertical: 40),
child: Center(
child:
Text('No invoices found for this period'),
),
)
else
...report.invoices.map((inv) => _InvoiceListItem(
invoice: inv.invoiceNumber,
vendor: inv.vendorName,
date: DateFormat('MMM dd, yyyy')
.format(inv.issueDate),
amount: NumberFormat.currency(symbol: r'$')
.format(inv.amount),
status: inv.status,
statusColor: inv.status == 'PAID'
? UiColors.success
: inv.status == 'PENDING'
? UiColors.textWarning
: UiColors.error,
)),
const SizedBox(height: 100), const SizedBox(height: 100),
], ],
@@ -356,8 +269,9 @@ class _SpendBarChart extends StatelessWidget {
return BarChart( return BarChart(
BarChartData( BarChartData(
alignment: BarChartAlignment.spaceAround, alignment: BarChartAlignment.spaceAround,
maxY: (chartData.fold<double>( maxY: (chartData.fold<double>(0,
0, (prev, element) => element.amount > prev ? element.amount : prev) * (prev, element) =>
element.amount > prev ? element.amount : prev) *
1.2) 1.2)
.ceilToDouble(), .ceilToDouble(),
barTouchData: BarTouchData( barTouchData: BarTouchData(
@@ -379,14 +293,34 @@ class _SpendBarChart extends StatelessWidget {
bottomTitles: AxisTitles( bottomTitles: AxisTitles(
sideTitles: SideTitles( sideTitles: SideTitles(
showTitles: true, showTitles: true,
reservedSize: 30,
getTitlesWidget: (value, meta) { getTitlesWidget: (value, meta) {
if (value.toInt() >= chartData.length) return const SizedBox(); if (value.toInt() >= chartData.length) return const SizedBox();
final date = chartData[value.toInt()].date; final date = chartData[value.toInt()].date;
return SideTitleWidget( return SideTitleWidget(
axisSide: meta.axisSide, axisSide: meta.axisSide,
space: 4, space: 8,
child: Text( child: Text(
DateFormat('E').format(date), DateFormat('E').format(date),
style: const TextStyle(
color: UiColors.textSecondary,
fontSize: 11,
),
),
);
},
),
),
leftTitles: AxisTitles(
sideTitles: SideTitles(
showTitles: true,
reservedSize: 40,
getTitlesWidget: (value, meta) {
if (value == 0) return const SizedBox();
return SideTitleWidget(
axisSide: meta.axisSide,
child: Text(
'\$${(value / 1000).toStringAsFixed(0)}k',
style: const TextStyle( style: const TextStyle(
color: UiColors.textSecondary, color: UiColors.textSecondary,
fontSize: 10, fontSize: 10,
@@ -396,9 +330,6 @@ class _SpendBarChart extends StatelessWidget {
}, },
), ),
), ),
leftTitles: const AxisTitles(
sideTitles: SideTitles(showTitles: false),
),
topTitles: const AxisTitles( topTitles: const AxisTitles(
sideTitles: SideTitles(showTitles: false), sideTitles: SideTitles(showTitles: false),
), ),
@@ -406,7 +337,15 @@ class _SpendBarChart extends StatelessWidget {
sideTitles: SideTitles(showTitles: false), sideTitles: SideTitles(showTitles: false),
), ),
), ),
gridData: const FlGridData(show: false), gridData: FlGridData(
show: true,
drawVerticalLine: false,
horizontalInterval: 1000,
getDrawingHorizontalLine: (value) => FlLine(
color: UiColors.bgSecondary,
strokeWidth: 1,
),
),
borderData: FlBorderData(show: false), borderData: FlBorderData(show: false),
barGroups: List.generate( barGroups: List.generate(
chartData.length, chartData.length,
@@ -416,7 +355,7 @@ class _SpendBarChart extends StatelessWidget {
BarChartRodData( BarChartRodData(
toY: chartData[index].amount, toY: chartData[index].amount,
color: UiColors.success, color: UiColors.success,
width: 16, width: 12,
borderRadius: const BorderRadius.vertical( borderRadius: const BorderRadius.vertical(
top: Radius.circular(4), top: Radius.circular(4),
), ),
@@ -429,20 +368,18 @@ class _SpendBarChart extends StatelessWidget {
} }
} }
class _SpendSummaryCard extends StatelessWidget { class _SpendStatCard extends StatelessWidget {
final String label; final String label;
final String value; final String value;
final String change; final String pillText;
final String period; final Color themeColor;
final Color color;
final IconData icon; final IconData icon;
const _SpendSummaryCard({ const _SpendStatCard({
required this.label, required this.label,
required this.value, required this.value,
required this.change, required this.pillText,
required this.period, required this.themeColor,
required this.color,
required this.icon, required this.icon,
}); });
@@ -450,6 +387,78 @@ class _SpendSummaryCard extends StatelessWidget {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Container( return Container(
padding: const EdgeInsets.all(16), padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: UiColors.white,
borderRadius: BorderRadius.circular(16),
boxShadow: [
BoxShadow(
color: UiColors.black.withOpacity(0.06),
blurRadius: 8,
offset: const Offset(0, 4),
),
],
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
Icon(icon, size: 14, color: themeColor),
const SizedBox(width: 8),
Expanded(
child: Text(
label,
style: const TextStyle(
fontSize: 12,
color: UiColors.textSecondary,
fontWeight: FontWeight.w500,
),
maxLines: 1,
overflow: TextOverflow.ellipsis,
),
),
],
),
const SizedBox(height: 12),
Text(
value,
style: const TextStyle(
fontSize: 24,
fontWeight: FontWeight.bold,
color: UiColors.textPrimary,
),
),
const SizedBox(height: 12),
Container(
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
decoration: BoxDecoration(
color: themeColor.withOpacity(0.1),
borderRadius: BorderRadius.circular(6),
),
child: Text(
pillText,
style: TextStyle(
fontSize: 10,
fontWeight: FontWeight.bold,
color: themeColor,
),
),
),
],
),
);
}
}
class _SpendByIndustryCard extends StatelessWidget {
final List<SpendIndustryCategory> industries;
const _SpendByIndustryCard({required this.industries});
@override
Widget build(BuildContext context) {
return Container(
padding: const EdgeInsets.all(20),
decoration: BoxDecoration( decoration: BoxDecoration(
color: UiColors.white, color: UiColors.white,
borderRadius: BorderRadius.circular(16), borderRadius: BorderRadius.circular(16),
@@ -464,209 +473,73 @@ class _SpendSummaryCard extends StatelessWidget {
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
Row( const Text(
mainAxisAlignment: MainAxisAlignment.spaceBetween, 'Spend by Industry',
children: [ style: TextStyle(
Container( fontSize: 14,
padding: const EdgeInsets.all(8),
decoration: BoxDecoration(
color: color.withOpacity(0.1),
shape: BoxShape.circle,
),
child: Icon(icon, size: 16, color: color),
),
if (change.isNotEmpty)
Text(
change,
style: TextStyle(
fontSize: 12,
fontWeight: FontWeight.bold,
color: change.startsWith('+')
? UiColors.error
: UiColors.success,
),
),
],
),
const SizedBox(height: 12),
Text(
label,
style: const TextStyle(
fontSize: 12,
color: UiColors.textSecondary,
),
),
const SizedBox(height: 4),
Text(
value,
style: const TextStyle(
fontSize: 18,
fontWeight: FontWeight.bold, fontWeight: FontWeight.bold,
color: UiColors.textPrimary, color: UiColors.textPrimary,
), ),
), ),
const SizedBox(height: 4), const SizedBox(height: 24),
Text( if (industries.isEmpty)
period, const Center(
style: const TextStyle( child: Padding(
fontSize: 10, padding: EdgeInsets.all(16.0),
color: UiColors.textDescription,
),
),
],
),
);
}
}
class _StatusMiniCard extends StatelessWidget {
final String label;
final String value;
final Color color;
const _StatusMiniCard({
required this.label,
required this.value,
required this.color,
});
@override
Widget build(BuildContext context) {
return Container(
padding: const EdgeInsets.symmetric(vertical: 12, horizontal: 8),
decoration: BoxDecoration(
color: color.withOpacity(0.05),
borderRadius: BorderRadius.circular(12),
border: Border.all(color: color.withOpacity(0.1)),
),
child: Column(
children: [
Text(
value,
style: TextStyle(
fontSize: 16,
fontWeight: FontWeight.bold,
color: color,
),
),
const SizedBox(height: 2),
Text(
label,
style: const TextStyle(
fontSize: 10,
color: UiColors.textSecondary,
fontWeight: FontWeight.w500,
),
),
],
),
);
}
}
class _InvoiceListItem extends StatelessWidget {
final String invoice;
final String vendor;
final String date;
final String amount;
final String status;
final Color statusColor;
const _InvoiceListItem({
required this.invoice,
required this.vendor,
required this.date,
required this.amount,
required this.status,
required this.statusColor,
});
@override
Widget build(BuildContext context) {
return Container(
margin: const EdgeInsets.only(bottom: 12),
padding: const EdgeInsets.all(16),
decoration: BoxDecoration(
color: UiColors.white,
borderRadius: BorderRadius.circular(16),
boxShadow: [
BoxShadow(
color: UiColors.black.withOpacity(0.02),
blurRadius: 10,
),
],
),
child: Row(
children: [
Container(
padding: const EdgeInsets.all(10),
decoration: BoxDecoration(
color: statusColor.withOpacity(0.1),
borderRadius: BorderRadius.circular(12),
),
child: Icon(UiIcons.file, size: 20, color: statusColor),
),
const SizedBox(width: 16),
Expanded(
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
invoice,
style: const TextStyle(
fontWeight: FontWeight.bold,
fontSize: 14,
color: UiColors.textPrimary,
),
),
const SizedBox(height: 2),
Text(
vendor,
style: const TextStyle(
fontSize: 12,
color: UiColors.textSecondary,
),
),
const SizedBox(height: 4),
Text(
date,
style: const TextStyle(
fontSize: 11,
color: UiColors.textDescription,
),
),
],
),
),
Column(
crossAxisAlignment: CrossAxisAlignment.end,
children: [
Text(
amount,
style: const TextStyle(
fontWeight: FontWeight.bold,
fontSize: 14,
color: UiColors.textPrimary,
),
),
const SizedBox(height: 4),
Container(
padding: const EdgeInsets.symmetric(horizontal: 8, vertical: 4),
decoration: BoxDecoration(
color: statusColor.withOpacity(0.1),
borderRadius: BorderRadius.circular(6),
),
child: Text( child: Text(
status.toUpperCase(), 'No industry data available',
style: TextStyle( style: TextStyle(color: UiColors.textSecondary),
color: statusColor,
fontSize: 9,
fontWeight: FontWeight.bold,
),
), ),
), ),
], )
), else
...industries.map((ind) => Padding(
padding: const EdgeInsets.only(bottom: 24.0),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
ind.name,
style: const TextStyle(
fontSize: 13,
color: UiColors.textSecondary,
),
),
Text(
NumberFormat.currency(symbol: r'$', decimalDigits: 0)
.format(ind.amount),
style: const TextStyle(
fontSize: 13,
fontWeight: FontWeight.bold,
color: UiColors.textPrimary,
),
),
],
),
const SizedBox(height: 8),
ClipRRect(
borderRadius: BorderRadius.circular(4),
child: LinearProgressIndicator(
value: ind.percentage / 100,
backgroundColor: UiColors.bgSecondary,
color: UiColors.success,
minHeight: 6,
),
),
const SizedBox(height: 6),
Text(
'${ind.percentage.toStringAsFixed(1)}% of total',
style: const TextStyle(
fontSize: 10,
color: UiColors.textDescription,
),
),
],
),
)),
], ],
), ),
); );

View File

@@ -281,7 +281,7 @@ query listInvoicesForSpendByBusiness(
status status
invoiceNumber invoiceNumber
vendor { id companyName } vendor { id companyName serviceSpecialty }
business { id businessName } business { id businessName }
order { id eventName } order { id eventName }
} }
@@ -306,7 +306,7 @@ query listInvoicesForSpendByVendor(
status status
invoiceNumber invoiceNumber
vendor { id companyName } vendor { id companyName serviceSpecialty }
business { id businessName } business { id businessName }
order { id eventName } order { id eventName }
} }
@@ -332,7 +332,7 @@ query listInvoicesForSpendByOrder(
status status
invoiceNumber invoiceNumber
vendor { id companyName } vendor { id companyName serviceSpecialty }
business { id businessName } business { id businessName }
order { id eventName } order { id eventName }
} }