353 lines
10 KiB
Dart
Raw Normal View History

2025-08-15 23:53:05 +07:00
import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
2025-08-17 22:36:46 +07:00
import 'package:flutter_bloc/flutter_bloc.dart';
2025-08-15 23:53:05 +07:00
import 'package:line_icons/line_icons.dart';
2025-08-17 22:36:46 +07:00
import '../../../application/profit_loss/profit_loss_loader/profit_loss_loader_bloc.dart';
import '../../../common/extension/extension.dart';
2025-08-15 23:53:05 +07:00
import '../../../common/theme/theme.dart';
2025-08-17 22:36:46 +07:00
import '../../../domain/analytic/analytic.dart';
import '../../../injection.dart';
2025-08-16 00:39:09 +07:00
import '../../components/appbar/appbar.dart';
2025-08-15 23:53:05 +07:00
import 'widgets/cash_flow.dart';
import 'widgets/category.dart';
2025-08-17 22:36:46 +07:00
import 'widgets/product.dart';
2025-08-15 23:53:05 +07:00
import 'widgets/profit_loss.dart';
import 'widgets/summary_card.dart';
@RoutePage()
2025-08-17 22:36:46 +07:00
class FinancePage extends StatefulWidget implements AutoRouteWrapper {
2025-08-15 23:53:05 +07:00
const FinancePage({super.key});
@override
State<FinancePage> createState() => _FinancePageState();
2025-08-17 22:36:46 +07:00
@override
Widget wrappedRoute(BuildContext context) => BlocProvider(
create: (_) =>
getIt<ProfitLossLoaderBloc>()..add(ProfitLossLoaderEvent.fetched()),
child: this,
);
2025-08-15 23:53:05 +07:00
}
class _FinancePageState extends State<FinancePage>
with TickerProviderStateMixin {
late AnimationController _slideController;
late AnimationController _fadeController;
late AnimationController _scaleController;
late Animation<Offset> _slideAnimation;
late Animation<double> _fadeAnimation;
late Animation<double> _scaleAnimation;
String selectedPeriod = 'Hari ini';
final List<String> periods = [
'Hari ini',
'Minggu ini',
'Bulan ini',
'Tahun ini',
];
@override
void initState() {
super.initState();
_slideController = AnimationController(
duration: const Duration(milliseconds: 800),
vsync: this,
);
_fadeController = AnimationController(
duration: const Duration(milliseconds: 1000),
vsync: this,
);
_scaleController = AnimationController(
duration: const Duration(milliseconds: 600),
vsync: this,
);
_slideAnimation =
Tween<Offset>(begin: const Offset(0, 0.3), end: Offset.zero).animate(
CurvedAnimation(parent: _slideController, curve: Curves.easeOutCubic),
);
_fadeAnimation = Tween<double>(
begin: 0.0,
end: 1.0,
).animate(CurvedAnimation(parent: _fadeController, curve: Curves.easeIn));
_scaleAnimation = Tween<double>(begin: 0.8, end: 1.0).animate(
CurvedAnimation(parent: _scaleController, curve: Curves.elasticOut),
);
// Start animations
_fadeController.forward();
Future.delayed(const Duration(milliseconds: 200), () {
_slideController.forward();
});
Future.delayed(const Duration(milliseconds: 400), () {
_scaleController.forward();
});
}
@override
void dispose() {
_slideController.dispose();
_fadeController.dispose();
_scaleController.dispose();
super.dispose();
}
@override
Widget build(BuildContext context) {
return Scaffold(
backgroundColor: AppColor.background,
2025-08-17 22:36:46 +07:00
body: BlocBuilder<ProfitLossLoaderBloc, ProfitLossLoaderState>(
builder: (context, state) {
return CustomScrollView(
slivers: [
// SliverAppBar with animated background
SliverAppBar(
expandedHeight: 120,
floating: false,
pinned: true,
backgroundColor: AppColor.primary,
elevation: 0,
flexibleSpace: CustomAppBar(title: 'Keuangan'),
),
2025-08-15 23:53:05 +07:00
2025-08-17 22:36:46 +07:00
// Header dengan filter periode
SliverToBoxAdapter(
child: FadeTransition(
opacity: _fadeAnimation,
child: _buildPeriodSelector(),
),
),
2025-08-15 23:53:05 +07:00
2025-08-17 22:36:46 +07:00
// Summary Cards
SliverToBoxAdapter(
child: SlideTransition(
position: _slideAnimation,
child: _buildSummaryCards(state.profitLoss.summary),
),
),
2025-08-15 23:53:05 +07:00
2025-08-17 22:36:46 +07:00
// Cash Flow Analysis
SliverToBoxAdapter(
child: ScaleTransition(
scale: _scaleAnimation,
child: FinanceCashFlow(dailyData: state.profitLoss.data),
),
),
2025-08-15 23:53:05 +07:00
2025-08-17 22:36:46 +07:00
// Profit Loss Detail
SliverToBoxAdapter(
child: FadeTransition(
opacity: _fadeAnimation,
child: FinanceProfitLoss(data: state.profitLoss.summary),
),
),
2025-08-15 23:53:05 +07:00
2025-08-17 22:36:46 +07:00
SliverToBoxAdapter(
child: SlideTransition(
position: _slideAnimation,
child: FinanceCategory(),
),
),
2025-08-15 23:53:05 +07:00
2025-08-17 22:36:46 +07:00
// Product Analysis Section
SliverToBoxAdapter(
child: SlideTransition(
position: _slideAnimation,
child: _buildProductAnalysis(state.profitLoss.productData),
),
),
2025-08-15 23:53:05 +07:00
2025-08-17 22:36:46 +07:00
// Transaction Categories
// Bottom spacing
const SliverToBoxAdapter(child: SizedBox(height: 100)),
],
);
},
2025-08-15 23:53:05 +07:00
),
);
}
Widget _buildPeriodSelector() {
return Container(
padding: const EdgeInsets.all(16),
child: Row(
children: [
Expanded(
child: Container(
padding: const EdgeInsets.symmetric(horizontal: 16),
decoration: BoxDecoration(
color: AppColor.white,
borderRadius: BorderRadius.circular(12),
border: Border.all(color: AppColor.border),
),
child: DropdownButtonHideUnderline(
child: DropdownButton<String>(
value: selectedPeriod,
isExpanded: true,
icon: const Icon(
LineIcons.angleDown,
color: AppColor.primary,
),
style: AppStyle.md,
items: periods.map((String period) {
return DropdownMenuItem<String>(
value: period,
child: Text(period),
);
}).toList(),
onChanged: (String? newValue) {
setState(() {
selectedPeriod = newValue!;
});
},
),
),
),
),
const SizedBox(width: 12),
Container(
decoration: BoxDecoration(
color: AppColor.primary,
borderRadius: BorderRadius.circular(12),
),
child: IconButton(
onPressed: () {},
icon: const Icon(LineIcons.calendar, color: AppColor.white),
),
),
],
),
);
}
2025-08-17 22:36:46 +07:00
Widget _buildSummaryCards(ProfitLossSummary summary) {
2025-08-15 23:53:05 +07:00
return Padding(
padding: const EdgeInsets.symmetric(horizontal: 16),
child: Column(
children: [
Row(
children: [
Expanded(
child: FinanceSummaryCard(
title: 'Total Pendapatan',
2025-08-17 22:36:46 +07:00
amount: summary.totalRevenue.currencyFormatRp,
2025-08-15 23:53:05 +07:00
icon: LineIcons.arrowUp,
color: AppColor.success,
isPositive: true,
),
),
const SizedBox(width: 12),
Expanded(
child: FinanceSummaryCard(
title: 'Total Pengeluaran',
2025-08-17 22:36:46 +07:00
amount: summary.totalCost.currencyFormatRp,
2025-08-15 23:53:05 +07:00
icon: LineIcons.arrowDown,
color: AppColor.error,
isPositive: false,
),
),
],
),
const SizedBox(height: 12),
Row(
children: [
Expanded(
child: FinanceSummaryCard(
title: 'Keuntungan Bersih',
2025-08-17 22:36:46 +07:00
amount: summary.netProfit.currencyFormatRp,
2025-08-15 23:53:05 +07:00
icon: LineIcons.lineChart,
color: AppColor.info,
isPositive: true,
),
),
const SizedBox(width: 12),
Expanded(
child: FinanceSummaryCard(
title: 'Margin Profit',
2025-08-17 22:36:46 +07:00
amount: '${summary.profitabilityRatio.round()}%',
2025-08-15 23:53:05 +07:00
icon: LineIcons.percent,
color: AppColor.warning,
isPositive: true,
),
),
],
),
],
),
);
}
2025-08-17 22:36:46 +07:00
Widget _buildProductAnalysis(List<ProfitLossProductData> products) {
2025-08-15 23:53:05 +07:00
return Container(
margin: const EdgeInsets.all(16),
padding: const EdgeInsets.all(20),
decoration: BoxDecoration(
color: AppColor.white,
borderRadius: BorderRadius.circular(16),
boxShadow: [
BoxShadow(
color: AppColor.textLight.withOpacity(0.1),
spreadRadius: 1,
blurRadius: 8,
offset: const Offset(0, 2),
),
],
),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
children: [
Container(
padding: const EdgeInsets.all(8),
decoration: BoxDecoration(
2025-08-17 22:36:46 +07:00
color: AppColor.info.withOpacity(0.1),
2025-08-15 23:53:05 +07:00
borderRadius: BorderRadius.circular(8),
),
child: const Icon(
2025-08-17 22:36:46 +07:00
LineIcons.shoppingBag,
color: AppColor.info,
2025-08-15 23:53:05 +07:00
size: 20,
),
),
const SizedBox(width: 12),
Text(
2025-08-17 22:36:46 +07:00
'Analisis Produk',
2025-08-15 23:53:05 +07:00
style: AppStyle.lg.copyWith(fontWeight: FontWeight.bold),
),
2025-08-17 22:36:46 +07:00
const Spacer(),
TextButton(
onPressed: () {},
child: Text(
'Lihat Semua',
style: AppStyle.sm.copyWith(color: AppColor.primary),
2025-08-15 23:53:05 +07:00
),
),
],
),
2025-08-17 22:36:46 +07:00
// Product list
ListView.separated(
shrinkWrap: true,
physics: const NeverScrollableScrollPhysics(),
padding: const EdgeInsets.only(top: 12),
itemCount: products.length,
separatorBuilder: (context, index) => const SizedBox(height: 12),
itemBuilder: (context, index) {
final product = products[index];
return ProfitLossProduct(product: product);
},
2025-08-15 23:53:05 +07:00
),
],
),
);
}
}