432 lines
14 KiB
Dart
Raw Normal View History

2025-08-15 18:02:09 +07:00
import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
2025-08-17 23:54:28 +07:00
import 'package:flutter_bloc/flutter_bloc.dart';
2025-08-15 18:02:09 +07:00
2025-08-17 23:54:28 +07:00
import '../../../application/analytic/inventory_analytic_loader/inventory_analytic_loader_bloc.dart';
2025-08-20 13:52:49 +07:00
import '../../../common/extension/extension.dart';
2025-08-15 18:02:09 +07:00
import '../../../common/theme/theme.dart';
2025-08-17 23:54:28 +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 18:02:09 +07:00
import 'widgets/ingredient_tile.dart';
import 'widgets/product_tile.dart';
import 'widgets/stat_card.dart';
import 'widgets/tabbar_delegate.dart';
// Custom SliverPersistentHeaderDelegate untuk TabBar
@RoutePage()
2025-08-17 23:54:28 +07:00
class InventoryPage extends StatefulWidget implements AutoRouteWrapper {
2025-08-15 18:02:09 +07:00
const InventoryPage({super.key});
@override
State<InventoryPage> createState() => _InventoryPageState();
2025-08-17 23:54:28 +07:00
@override
Widget wrappedRoute(BuildContext context) => BlocProvider(
create: (_) =>
getIt<InventoryAnalyticLoaderBloc>()
..add(InventoryAnalyticLoaderEvent.fetched()),
child: this,
);
2025-08-15 18:02:09 +07:00
}
class _InventoryPageState extends State<InventoryPage>
with TickerProviderStateMixin {
late AnimationController _fadeAnimationController;
late AnimationController _slideAnimationController;
late Animation<double> _fadeAnimation;
late Animation<Offset> _slideAnimation;
late TabController _tabController;
@override
void initState() {
super.initState();
_tabController = TabController(length: 2, vsync: this);
_fadeAnimationController = AnimationController(
duration: const Duration(milliseconds: 1000),
vsync: this,
);
_slideAnimationController = AnimationController(
duration: const Duration(milliseconds: 800),
vsync: this,
);
_fadeAnimation = Tween<double>(begin: 0.0, end: 1.0).animate(
CurvedAnimation(
parent: _fadeAnimationController,
curve: Curves.easeInOut,
),
);
_slideAnimation =
Tween<Offset>(begin: const Offset(0.0, 0.3), end: Offset.zero).animate(
CurvedAnimation(
parent: _slideAnimationController,
curve: Curves.easeOutBack,
),
);
_fadeAnimationController.forward();
_slideAnimationController.forward();
}
@override
void dispose() {
_fadeAnimationController.dispose();
_slideAnimationController.dispose();
_tabController.dispose();
super.dispose();
}
Color getStatusColor(String status) {
switch (status) {
case 'available':
return AppColor.success;
case 'low_stock':
return AppColor.warning;
case 'out_of_stock':
return AppColor.error;
default:
return AppColor.textSecondary;
}
}
String getStatusText(String status) {
switch (status) {
case 'available':
2025-08-20 13:52:49 +07:00
return context.lang.available;
2025-08-15 18:02:09 +07:00
case 'low_stock':
2025-08-20 13:52:49 +07:00
return context.lang.low_stock;
2025-08-15 18:02:09 +07:00
case 'out_of_stock':
2025-08-20 13:52:49 +07:00
return context.lang.out_of_stock;
2025-08-15 18:02:09 +07:00
default:
return 'Unknown';
}
}
@override
Widget build(BuildContext context) {
2025-08-18 17:27:36 +07:00
return BlocListener<
InventoryAnalyticLoaderBloc,
InventoryAnalyticLoaderState
>(
listenWhen: (previous, current) =>
previous.dateFrom != current.dateFrom ||
previous.dateTo != current.dateTo,
listener: (context, state) {
context.read<InventoryAnalyticLoaderBloc>().add(
InventoryAnalyticLoaderEvent.fetched(),
);
},
child: Scaffold(
backgroundColor: AppColor.background,
body:
BlocBuilder<
InventoryAnalyticLoaderBloc,
InventoryAnalyticLoaderState
>(
builder: (context, state) {
return FadeTransition(
opacity: _fadeAnimation,
child: SlideTransition(
position: _slideAnimation,
child: NestedScrollView(
headerSliverBuilder: (context, innerBoxIsScrolled) {
return [
_buildSliverAppBar(),
SliverPersistentHeader(
pinned: true,
delegate: InventorySliverTabBarDelegate(
startDate: state.dateFrom,
endDate: state.dateTo,
onDateRangeChanged: (startDate, endDate) {
context.read<InventoryAnalyticLoaderBloc>().add(
InventoryAnalyticLoaderEvent.rangeDateChanged(
startDate!,
endDate!,
2025-08-17 23:54:28 +07:00
),
2025-08-18 17:27:36 +07:00
);
},
tabBar: TabBar(
controller: _tabController,
indicator: BoxDecoration(
gradient: LinearGradient(
colors: AppColor.primaryGradient,
begin: Alignment.topLeft,
end: Alignment.bottomRight,
2025-08-17 23:54:28 +07:00
),
2025-08-18 17:27:36 +07:00
borderRadius: BorderRadius.circular(25),
boxShadow: [
BoxShadow(
color: AppColor.primary.withOpacity(0.3),
blurRadius: 12,
offset: const Offset(0, 4),
),
],
2025-08-17 23:54:28 +07:00
),
2025-08-18 17:27:36 +07:00
indicatorSize: TabBarIndicatorSize.tab,
indicatorPadding: const EdgeInsets.all(6),
labelColor: AppColor.textWhite,
unselectedLabelColor: AppColor.textSecondary,
labelStyle: const TextStyle(
fontWeight: FontWeight.w700,
fontSize: 13,
),
unselectedLabelStyle: const TextStyle(
fontWeight: FontWeight.w500,
fontSize: 13,
),
dividerColor: Colors.transparent,
splashFactory: NoSplash.splashFactory,
overlayColor: MaterialStateProperty.all(
Colors.transparent,
),
tabs: [
Tab(
height: 40,
child: Container(
padding: const EdgeInsets.symmetric(
horizontal: 12,
),
2025-08-20 13:52:49 +07:00
child: Row(
2025-08-18 17:27:36 +07:00
mainAxisSize: MainAxisSize.min,
mainAxisAlignment:
MainAxisAlignment.center,
children: [
Icon(
Icons.inventory_2_rounded,
size: 16,
),
SizedBox(width: 6),
2025-08-20 13:52:49 +07:00
Text(context.lang.product),
2025-08-18 17:27:36 +07:00
],
),
2025-08-17 23:54:28 +07:00
),
2025-08-18 17:27:36 +07:00
),
Tab(
height: 40,
child: Container(
padding: const EdgeInsets.symmetric(
horizontal: 12,
),
2025-08-20 13:52:49 +07:00
child: Row(
2025-08-18 17:27:36 +07:00
mainAxisSize: MainAxisSize.min,
mainAxisAlignment:
MainAxisAlignment.center,
children: [
Icon(
Icons.restaurant_menu_rounded,
size: 16,
),
SizedBox(width: 6),
2025-08-20 13:52:49 +07:00
Text(context.lang.ingredients),
2025-08-18 17:27:36 +07:00
],
),
2025-08-17 23:54:28 +07:00
),
),
2025-08-18 17:27:36 +07:00
],
),
2025-08-15 18:02:09 +07:00
),
),
2025-08-18 17:27:36 +07:00
];
},
body: TabBarView(
controller: _tabController,
children: [
_buildProductTab(state.inventoryAnalytic),
_buildIngredientTab(state.inventoryAnalytic),
],
),
2025-08-15 18:02:09 +07:00
),
),
2025-08-18 17:27:36 +07:00
);
},
),
),
2025-08-15 18:02:09 +07:00
);
}
Widget _buildSliverAppBar() {
return SliverAppBar(
expandedHeight: 120,
floating: false,
pinned: true,
elevation: 0,
backgroundColor: AppColor.primary,
2025-08-20 13:52:49 +07:00
flexibleSpace: CustomAppBar(title: context.lang.inventory),
2025-08-15 18:02:09 +07:00
);
}
2025-08-17 23:54:28 +07:00
Widget _buildProductTab(InventoryAnalytic inventoryAnalytic) {
2025-08-15 18:02:09 +07:00
return CustomScrollView(
slivers: [
2025-08-17 23:54:28 +07:00
SliverToBoxAdapter(
child: _buildProductStats(inventoryAnalytic.summary),
),
2025-08-15 18:02:09 +07:00
SliverPadding(
padding: const EdgeInsets.all(16),
2025-08-17 23:54:28 +07:00
sliver: SliverList(
2025-08-15 18:02:09 +07:00
delegate: SliverChildBuilderDelegate(
(context, index) =>
2025-08-17 23:54:28 +07:00
InventoryProductTile(item: inventoryAnalytic.products[index]),
childCount: inventoryAnalytic.products.length,
2025-08-15 18:02:09 +07:00
),
),
),
],
);
}
2025-08-17 23:54:28 +07:00
Widget _buildIngredientTab(InventoryAnalytic inventoryAnalytic) {
2025-08-15 18:02:09 +07:00
return CustomScrollView(
slivers: [
2025-08-17 23:54:28 +07:00
SliverToBoxAdapter(
child: _buildIngredientStats(inventoryAnalytic.summary),
),
2025-08-15 18:02:09 +07:00
SliverPadding(
padding: const EdgeInsets.all(16),
sliver: SliverList(
delegate: SliverChildBuilderDelegate(
2025-08-17 23:54:28 +07:00
(context, index) => InventoryIngredientTile(
item: inventoryAnalytic.ingredients[index],
),
childCount: inventoryAnalytic.ingredients.length,
2025-08-15 18:02:09 +07:00
),
),
),
],
);
}
2025-08-17 23:54:28 +07:00
Widget _buildProductStats(InventorySummary inventory) {
2025-08-15 18:02:09 +07:00
return Container(
margin: const EdgeInsets.all(16),
child: Column(
children: [
Row(
children: [
Expanded(
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.total_products,
2025-08-17 23:54:28 +07:00
inventory.totalProducts.toString(),
2025-08-15 18:02:09 +07:00
Icons.inventory_2_rounded,
AppColor.primary,
),
),
const SizedBox(width: 16),
Expanded(
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.total_sold,
2025-08-17 23:54:28 +07:00
inventory.totalSoldProducts.toString(),
2025-08-15 18:02:09 +07:00
Icons.check_circle_rounded,
AppColor.success,
),
),
],
),
const SizedBox(height: 16),
Row(
children: [
Expanded(
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.low_stock,
2025-08-17 23:54:28 +07:00
inventory.lowStockProducts.toString(),
2025-08-15 18:02:09 +07:00
Icons.warning_rounded,
AppColor.warning,
),
),
const SizedBox(width: 16),
Expanded(
2025-08-17 23:54:28 +07:00
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.zero_stock,
2025-08-17 23:54:28 +07:00
inventory.zeroStockProducts.toString(),
Icons.error_rounded,
AppColor.error,
),
2025-08-15 18:02:09 +07:00
),
],
),
],
),
);
}
2025-08-17 23:54:28 +07:00
Widget _buildIngredientStats(InventorySummary inventory) {
2025-08-15 18:02:09 +07:00
return Container(
margin: const EdgeInsets.all(16),
child: Column(
children: [
Row(
children: [
Expanded(
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.total_ingredients,
2025-08-17 23:54:28 +07:00
inventory.totalIngredients.toString(),
2025-08-15 18:02:09 +07:00
Icons.restaurant_menu_rounded,
AppColor.primary,
),
),
const SizedBox(width: 16),
Expanded(
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.total_sold,
2025-08-17 23:54:28 +07:00
inventory.totalSoldIngredients.toString(),
2025-08-15 18:02:09 +07:00
Icons.check_circle_rounded,
AppColor.success,
),
),
],
),
const SizedBox(height: 16),
Row(
children: [
Expanded(
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.low_stock,
2025-08-17 23:54:28 +07:00
inventory.lowStockIngredients.toString(),
2025-08-15 18:02:09 +07:00
Icons.warning_rounded,
AppColor.warning,
),
),
const SizedBox(width: 16),
Expanded(
child: _buildStatCard(
2025-08-20 13:52:49 +07:00
context.lang.zero_stock,
2025-08-17 23:54:28 +07:00
inventory.zeroStockIngredients.toString(),
2025-08-15 18:02:09 +07:00
Icons.error_rounded,
AppColor.error,
),
),
],
),
],
),
);
}
Widget _buildStatCard(
String title,
String value,
IconData icon,
Color color,
) {
return TweenAnimationBuilder<double>(
tween: Tween<double>(begin: 0, end: 1),
duration: const Duration(milliseconds: 800),
builder: (context, animationValue, child) {
return Transform.scale(
scale: animationValue,
child: InventoryStatCard(
title: title,
value: value,
icon: icon,
color: color,
),
);
},
);
}
}