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-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':
|
|
|
|
|
return 'Tersedia';
|
|
|
|
|
case 'low_stock':
|
|
|
|
|
return 'Stok Rendah';
|
|
|
|
|
case 'out_of_stock':
|
|
|
|
|
return 'Habis';
|
|
|
|
|
default:
|
|
|
|
|
return 'Unknown';
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
return Scaffold(
|
|
|
|
|
backgroundColor: AppColor.background,
|
2025-08-17 23:54:28 +07:00
|
|
|
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(
|
|
|
|
|
tabBar: TabBar(
|
|
|
|
|
controller: _tabController,
|
|
|
|
|
indicator: BoxDecoration(
|
|
|
|
|
gradient: LinearGradient(
|
|
|
|
|
colors: AppColor.primaryGradient,
|
|
|
|
|
begin: Alignment.topLeft,
|
|
|
|
|
end: Alignment.bottomRight,
|
|
|
|
|
),
|
|
|
|
|
borderRadius: BorderRadius.circular(25),
|
|
|
|
|
boxShadow: [
|
|
|
|
|
BoxShadow(
|
|
|
|
|
color: AppColor.primary.withOpacity(0.3),
|
|
|
|
|
blurRadius: 12,
|
|
|
|
|
offset: const Offset(0, 4),
|
|
|
|
|
),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
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,
|
|
|
|
|
),
|
|
|
|
|
child: const Row(
|
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
|
mainAxisAlignment:
|
|
|
|
|
MainAxisAlignment.center,
|
|
|
|
|
children: [
|
|
|
|
|
Icon(
|
|
|
|
|
Icons.inventory_2_rounded,
|
|
|
|
|
size: 16,
|
|
|
|
|
),
|
|
|
|
|
SizedBox(width: 6),
|
|
|
|
|
Text('Produk'),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
Tab(
|
|
|
|
|
height: 40,
|
|
|
|
|
child: Container(
|
|
|
|
|
padding: const EdgeInsets.symmetric(
|
|
|
|
|
horizontal: 12,
|
|
|
|
|
),
|
|
|
|
|
child: const Row(
|
|
|
|
|
mainAxisSize: MainAxisSize.min,
|
|
|
|
|
mainAxisAlignment:
|
|
|
|
|
MainAxisAlignment.center,
|
|
|
|
|
children: [
|
|
|
|
|
Icon(
|
|
|
|
|
Icons.restaurant_menu_rounded,
|
|
|
|
|
size: 16,
|
|
|
|
|
),
|
|
|
|
|
SizedBox(width: 6),
|
|
|
|
|
Text('Bahan'),
|
|
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
2025-08-15 18:02:09 +07:00
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
2025-08-17 23:54:28 +07:00
|
|
|
];
|
|
|
|
|
},
|
|
|
|
|
body: TabBarView(
|
|
|
|
|
controller: _tabController,
|
|
|
|
|
children: [
|
|
|
|
|
_buildProductTab(state.inventoryAnalytic),
|
|
|
|
|
_buildIngredientTab(state.inventoryAnalytic),
|
2025-08-15 18:02:09 +07:00
|
|
|
],
|
|
|
|
|
),
|
|
|
|
|
),
|
|
|
|
|
),
|
2025-08-17 23:54:28 +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-16 00:39:09 +07:00
|
|
|
flexibleSpace: CustomAppBar(title: 'Inventaris'),
|
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(
|
|
|
|
|
'Total Produk',
|
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-17 23:54:28 +07:00
|
|
|
'Produk Terjual',
|
|
|
|
|
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(
|
|
|
|
|
'Stok Rendah',
|
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(
|
|
|
|
|
'Stok Kosong',
|
|
|
|
|
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(
|
|
|
|
|
'Total Bahan',
|
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-17 23:54:28 +07:00
|
|
|
'Bahan Terjual',
|
|
|
|
|
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(
|
|
|
|
|
'Stok Kurang',
|
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(
|
|
|
|
|
'Habis',
|
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,
|
|
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
},
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|