checkout page

This commit is contained in:
efrilm 2025-10-27 01:54:35 +07:00
parent e8fd683077
commit 3135dde317
9 changed files with 530 additions and 64 deletions

View File

@ -150,6 +150,42 @@ class _OrderMenuState extends State<OrderMenu> {
), ),
], ],
), ),
SpaceHeight(8.0),
SizedBox(
height: 40,
child: Row(
children: [
Flexible(
child: TextFormField(
cursorColor: AppColor.primary,
controller: _controller,
style: const TextStyle(fontSize: 12, color: AppColor.black),
decoration: InputDecoration(hintText: 'Catatan Pesanan'),
),
),
const SpaceWidth(16.0),
GestureDetector(
onTap: () {
context.read<CheckoutFormBloc>().add(
CheckoutFormEvent.removeItem(
widget.data.product,
widget.data.variant,
),
);
},
child: Container(
height: 40,
width: 40,
decoration: BoxDecoration(
color: AppColor.primary,
borderRadius: BorderRadius.circular(8.0),
),
child: Icon(Icons.delete_outline, color: AppColor.white),
),
),
],
),
),
], ],
); );
} }

View File

@ -0,0 +1,107 @@
part of 'field.dart';
class CustomerAutocomplete extends StatefulWidget {
final TextEditingController controller;
final Customer? selectedCustomer;
final ValueChanged<Customer?>? onSelected;
const CustomerAutocomplete({
super.key,
required this.controller,
this.selectedCustomer,
this.onSelected,
});
@override
State<CustomerAutocomplete> createState() => _CustomerAutocompleteState();
}
class _CustomerAutocompleteState extends State<CustomerAutocomplete> {
final FocusNode _focusNode = FocusNode();
@override
void initState() {
super.initState();
context.read<CustomerLoaderBloc>().add(CustomerLoaderEvent.fetched());
}
@override
Widget build(BuildContext context) {
return Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Text(
'Pelanggan',
style: AppStyle.lg.copyWith(
color: AppColor.black,
fontWeight: FontWeight.w600,
),
),
const SpaceHeight(6.0),
BlocBuilder<CustomerLoaderBloc, CustomerLoaderState>(
builder: (context, state) {
return RawAutocomplete<Customer>(
textEditingController: widget.controller,
focusNode: _focusNode,
optionsBuilder: (TextEditingValue textEditingValue) {
if (textEditingValue.text.isEmpty) {
return const Iterable<Customer>.empty();
}
return state.customers.where((Customer customer) {
return customer.name.toLowerCase().contains(
textEditingValue.text.toLowerCase(),
);
});
},
displayStringForOption: (Customer option) => option.name,
onSelected: (Customer selection) {
widget.controller.text = selection.name;
if (widget.onSelected != null) {
widget.onSelected!(selection);
}
},
fieldViewBuilder:
(context, controller, focusNode, onFieldSubmitted) {
return TextFormField(
controller: controller,
focusNode: focusNode,
decoration: const InputDecoration(
hintText: 'Nama Pelanggan',
),
onChanged: (value) {
if (widget.onSelected != null) {
widget.onSelected!(null); // reset jika ketik manual
}
},
);
},
optionsViewBuilder: (context, onSelected, options) {
return Material(
elevation: 1,
borderRadius: BorderRadius.circular(8),
color: Colors.white,
child: SizedBox(
height: context.deviceHeight * 0.4,
child: ListView.separated(
padding: EdgeInsets.zero,
itemCount: options.length,
separatorBuilder: (_, __) => const Divider(height: 1),
itemBuilder: (BuildContext context, int index) {
final Customer option = options.elementAt(index);
return ListTile(
title: Text(option.name),
onTap: () {
onSelected(option);
},
);
},
),
),
);
},
);
},
),
],
);
}
}

View File

@ -1,8 +1,13 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import '../../../application/customer/customer_loader/customer_loader_bloc.dart';
import '../../../common/extension/extension.dart';
import '../../../common/theme/theme.dart'; import '../../../common/theme/theme.dart';
import '../../../domain/customer/customer.dart';
import '../spaces/space.dart'; import '../spaces/space.dart';
part 'password_text_field.dart'; part 'password_text_field.dart';
part 'text_field.dart'; part 'text_field.dart';
part 'search_text_field.dart'; part 'search_text_field.dart';
part 'customer_autocomplete.dart';

View File

@ -0,0 +1,52 @@
import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import '../../../application/checkout/checkout_form/checkout_form_bloc.dart';
import '../../../common/theme/theme.dart';
import '../../components/spaces/space.dart';
import 'widgets/checkout_left_panel.dart';
import 'widgets/checkout_right_panel.dart';
@RoutePage()
class CheckoutPage extends StatelessWidget {
const CheckoutPage({super.key});
@override
Widget build(BuildContext context) {
return SafeArea(
child: Hero(
tag: 'checkout_screen',
child: BlocBuilder<CheckoutFormBloc, CheckoutFormState>(
builder: (context, state) {
final price = state.items.fold(
0,
(previousValue, element) =>
previousValue +
(element.product.price.toInt() +
(element.variant?.priceModifier.toInt() ?? 0)) *
element.quantity,
);
return Scaffold(
backgroundColor: AppColor.white,
body: Row(
children: [
Expanded(
flex: 3,
child: CheckoutLeftPanel(
checkoutState: state,
price: price,
),
),
SpaceWidth(2),
Expanded(flex: 3, child: CheckoutRightPanel()),
],
),
);
},
),
),
);
}
}

View File

@ -0,0 +1,199 @@
import 'package:flutter/widgets.dart';
import '../../../../application/checkout/checkout_form/checkout_form_bloc.dart';
import '../../../../common/extension/extension.dart';
import '../../../../common/theme/theme.dart';
import '../../../components/border/dashed_border.dart';
import '../../../components/card/order_menu.dart';
import '../../../components/page/page_title.dart';
import '../../../components/spaces/space.dart';
class CheckoutLeftPanel extends StatelessWidget {
final CheckoutFormState checkoutState;
final int price;
const CheckoutLeftPanel({
super.key,
required this.checkoutState,
required this.price,
});
@override
Widget build(BuildContext context) {
return Column(
mainAxisAlignment: MainAxisAlignment.start,
children: [
PageTitle(
title: 'Konfirmasi',
subtitle: checkoutState.table != null
? 'Pesanan Meja ${checkoutState.table?.tableName ?? "-"}'
: 'Pesanan',
actionWidget: [
Container(
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(8.0),
border: Border.all(color: AppColor.primary, width: 1.0),
),
padding: const EdgeInsets.all(8.0),
child: Text(
checkoutState.orderType.value,
style: TextStyle(
color: AppColor.primary,
fontSize: 16,
fontWeight: FontWeight.w600,
),
),
),
],
),
Container(
padding: const EdgeInsets.all(16.0).copyWith(bottom: 8),
decoration: const BoxDecoration(
border: Border(
bottom: BorderSide(color: AppColor.border, width: 1.0),
),
),
child: Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Item',
style: AppStyle.lg.copyWith(
color: AppColor.primary,
fontWeight: FontWeight.w600,
),
),
SpaceWidth(160),
SizedBox(
width: 50.0,
child: Text(
'Qty',
style: AppStyle.lg.copyWith(
color: AppColor.primary,
fontWeight: FontWeight.w600,
),
),
),
SizedBox(
child: Text(
'Price',
style: AppStyle.lg.copyWith(
color: AppColor.primary,
fontWeight: FontWeight.w600,
),
),
),
],
),
),
Expanded(
child: ListView.separated(
shrinkWrap: true,
padding: const EdgeInsets.all(16.0).copyWith(top: 8.0),
itemBuilder: (context, index) =>
OrderMenu(data: checkoutState.items[index]),
separatorBuilder: (context, index) => const SpaceHeight(12.0),
itemCount: checkoutState.items.length,
),
),
DashedDivider(color: AppColor.border),
Container(
padding: const EdgeInsets.all(16.0),
child: Column(
children: [
if (checkoutState.delivery != null) ...[
Column(
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Pengiriman',
style: AppStyle.md.copyWith(
color: AppColor.black,
fontWeight: FontWeight.w600,
),
),
Text(
checkoutState.delivery?.name ?? '-',
style: AppStyle.md.copyWith(
color: AppColor.black,
fontWeight: FontWeight.w600,
),
),
],
),
const SpaceHeight(8.0),
DashedDivider(color: AppColor.border),
const SpaceHeight(8.0),
],
),
],
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Sub total',
style: AppStyle.md.copyWith(
color: AppColor.black,
fontWeight: FontWeight.w600,
),
),
Text(
price.currencyFormatRp,
style: AppStyle.md.copyWith(
color: AppColor.black,
fontWeight: FontWeight.w600,
),
),
],
),
const SpaceHeight(8.0),
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Pajak PB1',
style: AppStyle.md.copyWith(
color: AppColor.black,
fontWeight: FontWeight.w400,
),
),
Text(
'${checkoutState.tax}% (Rp. 0)',
style: AppStyle.md.copyWith(
color: AppColor.black,
fontWeight: FontWeight.w500,
),
),
],
),
const SpaceHeight(8.0),
DashedDivider(color: AppColor.border),
const SpaceHeight(8.0),
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Text(
'Total',
style: AppStyle.xl.copyWith(
color: AppColor.black,
fontWeight: FontWeight.bold,
),
),
Text(
price.currencyFormatRp,
style: const TextStyle(
color: AppColor.primary,
fontWeight: FontWeight.bold,
fontSize: 18,
),
),
],
),
],
),
),
],
);
}
}

View File

@ -0,0 +1,48 @@
import 'package:flutter/widgets.dart';
import '../../../../common/theme/theme.dart';
import '../../../components/field/field.dart';
import '../../../components/page/page_title.dart';
class CheckoutRightPanel extends StatelessWidget {
const CheckoutRightPanel({super.key});
@override
Widget build(BuildContext context) {
return Column(
children: [
PageTitle(
title: 'Pembayaran',
isBack: false,
subtitle: 'Silahkan lakukan pembayaran',
),
Expanded(
child: SingleChildScrollView(
child: Column(
children: [
// Container(
// padding: const EdgeInsets.all(16),
// decoration: BoxDecoration(
// color: AppColor.white,
// border: Border(
// bottom: BorderSide(color: AppColor.border, width: 1.0),
// ),
// ),
// child: CustomerAutocomplete(
// controller: customerController,
// selectedCustomer: selectedCustomer,
// onSelected: (customer) {
// setState(() {
// selectedCustomer = customer;
// });
// },
// ),
// ),
],
),
),
),
],
);
}
}

View File

@ -1,5 +1,6 @@
import 'dart:developer'; import 'dart:developer';
import 'package:auto_route/auto_route.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
@ -10,6 +11,7 @@ import '../../../../../components/button/button.dart';
import '../../../../../components/card/order_menu.dart'; import '../../../../../components/card/order_menu.dart';
import '../../../../../components/spaces/space.dart'; import '../../../../../components/spaces/space.dart';
import '../../../../../components/toast/flushbar.dart'; import '../../../../../components/toast/flushbar.dart';
import '../../../../../router/app_router.gr.dart';
import 'home_right_title.dart'; import 'home_right_title.dart';
class HomeRightPanel extends StatelessWidget { class HomeRightPanel extends StatelessWidget {
@ -146,7 +148,8 @@ class HomeRightPanel extends StatelessWidget {
elevation: 1, elevation: 1,
disabled: state.items.isEmpty, disabled: state.items.isEmpty,
onPressed: () { onPressed: () {
if (state.orderType.name == 'dineIn') { if (state.orderType.name == 'dineIn' &&
state.table == null) {
AppFlushbar.showError( AppFlushbar.showError(
context, context,
'Mohon pilih meja terlebih dahulu', 'Mohon pilih meja terlebih dahulu',
@ -161,12 +164,7 @@ class HomeRightPanel extends StatelessWidget {
); );
return; return;
} }
// context.push( context.router.push(CheckoutRoute());
// ConfirmPaymentPage(
// isTable: widget.table == null ? false : true,
// table: widget.table,
// ),
// );
}, },
label: 'Lanjutkan Pembayaran', label: 'Lanjutkan Pembayaran',
), ),

View File

@ -25,5 +25,8 @@ class AppRouter extends RootStackRouter {
// Sync // Sync
AutoRoute(page: SyncRoute.page), AutoRoute(page: SyncRoute.page),
// Checkout
AutoRoute(page: CheckoutRoute.page),
]; ];
} }

View File

@ -10,165 +10,183 @@
// ignore_for_file: no_leading_underscores_for_library_prefixes // ignore_for_file: no_leading_underscores_for_library_prefixes
import 'package:apskel_pos_flutter_v2/presentation/pages/auth/login/login_page.dart' import 'package:apskel_pos_flutter_v2/presentation/pages/auth/login/login_page.dart'
as _i3;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/main_page.dart'
as _i4; as _i4;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/customer/customer_page.dart' import 'package:apskel_pos_flutter_v2/presentation/pages/checkout/checkout_page.dart'
as _i1; as _i1;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/home/home_page.dart' import 'package:apskel_pos_flutter_v2/presentation/pages/main/main_page.dart'
as _i2;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/report/report_page.dart'
as _i5; as _i5;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/setting/setting_page.dart' import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/customer/customer_page.dart'
as _i2;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/home/home_page.dart'
as _i3;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/report/report_page.dart'
as _i6; as _i6;
import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/table/table_page.dart' import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/setting/setting_page.dart'
as _i9;
import 'package:apskel_pos_flutter_v2/presentation/pages/splash/splash_page.dart'
as _i7; as _i7;
import 'package:apskel_pos_flutter_v2/presentation/pages/sync/sync_page.dart' import 'package:apskel_pos_flutter_v2/presentation/pages/main/pages/table/table_page.dart'
as _i10;
import 'package:apskel_pos_flutter_v2/presentation/pages/splash/splash_page.dart'
as _i8; as _i8;
import 'package:auto_route/auto_route.dart' as _i10; import 'package:apskel_pos_flutter_v2/presentation/pages/sync/sync_page.dart'
as _i9;
import 'package:auto_route/auto_route.dart' as _i11;
/// generated route for /// generated route for
/// [_i1.CustomerPage] /// [_i1.CheckoutPage]
class CustomerRoute extends _i10.PageRouteInfo<void> { class CheckoutRoute extends _i11.PageRouteInfo<void> {
const CustomerRoute({List<_i10.PageRouteInfo>? children}) const CheckoutRoute({List<_i11.PageRouteInfo>? children})
: super(CheckoutRoute.name, initialChildren: children);
static const String name = 'CheckoutRoute';
static _i11.PageInfo page = _i11.PageInfo(
name,
builder: (data) {
return const _i1.CheckoutPage();
},
);
}
/// generated route for
/// [_i2.CustomerPage]
class CustomerRoute extends _i11.PageRouteInfo<void> {
const CustomerRoute({List<_i11.PageRouteInfo>? children})
: super(CustomerRoute.name, initialChildren: children); : super(CustomerRoute.name, initialChildren: children);
static const String name = 'CustomerRoute'; static const String name = 'CustomerRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return _i10.WrappedRoute(child: const _i1.CustomerPage()); return _i11.WrappedRoute(child: const _i2.CustomerPage());
}, },
); );
} }
/// generated route for /// generated route for
/// [_i2.HomePage] /// [_i3.HomePage]
class HomeRoute extends _i10.PageRouteInfo<void> { class HomeRoute extends _i11.PageRouteInfo<void> {
const HomeRoute({List<_i10.PageRouteInfo>? children}) const HomeRoute({List<_i11.PageRouteInfo>? children})
: super(HomeRoute.name, initialChildren: children); : super(HomeRoute.name, initialChildren: children);
static const String name = 'HomeRoute'; static const String name = 'HomeRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return _i10.WrappedRoute(child: const _i2.HomePage()); return _i11.WrappedRoute(child: const _i3.HomePage());
}, },
); );
} }
/// generated route for /// generated route for
/// [_i3.LoginPage] /// [_i4.LoginPage]
class LoginRoute extends _i10.PageRouteInfo<void> { class LoginRoute extends _i11.PageRouteInfo<void> {
const LoginRoute({List<_i10.PageRouteInfo>? children}) const LoginRoute({List<_i11.PageRouteInfo>? children})
: super(LoginRoute.name, initialChildren: children); : super(LoginRoute.name, initialChildren: children);
static const String name = 'LoginRoute'; static const String name = 'LoginRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return _i10.WrappedRoute(child: const _i3.LoginPage()); return _i11.WrappedRoute(child: const _i4.LoginPage());
}, },
); );
} }
/// generated route for /// generated route for
/// [_i4.MainPage] /// [_i5.MainPage]
class MainRoute extends _i10.PageRouteInfo<void> { class MainRoute extends _i11.PageRouteInfo<void> {
const MainRoute({List<_i10.PageRouteInfo>? children}) const MainRoute({List<_i11.PageRouteInfo>? children})
: super(MainRoute.name, initialChildren: children); : super(MainRoute.name, initialChildren: children);
static const String name = 'MainRoute'; static const String name = 'MainRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return const _i4.MainPage(); return const _i5.MainPage();
}, },
); );
} }
/// generated route for /// generated route for
/// [_i5.ReportPage] /// [_i6.ReportPage]
class ReportRoute extends _i10.PageRouteInfo<void> { class ReportRoute extends _i11.PageRouteInfo<void> {
const ReportRoute({List<_i10.PageRouteInfo>? children}) const ReportRoute({List<_i11.PageRouteInfo>? children})
: super(ReportRoute.name, initialChildren: children); : super(ReportRoute.name, initialChildren: children);
static const String name = 'ReportRoute'; static const String name = 'ReportRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return const _i5.ReportPage(); return const _i6.ReportPage();
}, },
); );
} }
/// generated route for /// generated route for
/// [_i6.SettingPage] /// [_i7.SettingPage]
class SettingRoute extends _i10.PageRouteInfo<void> { class SettingRoute extends _i11.PageRouteInfo<void> {
const SettingRoute({List<_i10.PageRouteInfo>? children}) const SettingRoute({List<_i11.PageRouteInfo>? children})
: super(SettingRoute.name, initialChildren: children); : super(SettingRoute.name, initialChildren: children);
static const String name = 'SettingRoute'; static const String name = 'SettingRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return const _i6.SettingPage(); return const _i7.SettingPage();
}, },
); );
} }
/// generated route for /// generated route for
/// [_i7.SplashPage] /// [_i8.SplashPage]
class SplashRoute extends _i10.PageRouteInfo<void> { class SplashRoute extends _i11.PageRouteInfo<void> {
const SplashRoute({List<_i10.PageRouteInfo>? children}) const SplashRoute({List<_i11.PageRouteInfo>? children})
: super(SplashRoute.name, initialChildren: children); : super(SplashRoute.name, initialChildren: children);
static const String name = 'SplashRoute'; static const String name = 'SplashRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return const _i7.SplashPage(); return const _i8.SplashPage();
}, },
); );
} }
/// generated route for /// generated route for
/// [_i8.SyncPage] /// [_i9.SyncPage]
class SyncRoute extends _i10.PageRouteInfo<void> { class SyncRoute extends _i11.PageRouteInfo<void> {
const SyncRoute({List<_i10.PageRouteInfo>? children}) const SyncRoute({List<_i11.PageRouteInfo>? children})
: super(SyncRoute.name, initialChildren: children); : super(SyncRoute.name, initialChildren: children);
static const String name = 'SyncRoute'; static const String name = 'SyncRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return _i10.WrappedRoute(child: const _i8.SyncPage()); return _i11.WrappedRoute(child: const _i9.SyncPage());
}, },
); );
} }
/// generated route for /// generated route for
/// [_i9.TablePage] /// [_i10.TablePage]
class TableRoute extends _i10.PageRouteInfo<void> { class TableRoute extends _i11.PageRouteInfo<void> {
const TableRoute({List<_i10.PageRouteInfo>? children}) const TableRoute({List<_i11.PageRouteInfo>? children})
: super(TableRoute.name, initialChildren: children); : super(TableRoute.name, initialChildren: children);
static const String name = 'TableRoute'; static const String name = 'TableRoute';
static _i10.PageInfo page = _i10.PageInfo( static _i11.PageInfo page = _i11.PageInfo(
name, name,
builder: (data) { builder: (data) {
return _i10.WrappedRoute(child: const _i9.TablePage()); return _i11.WrappedRoute(child: const _i10.TablePage());
}, },
); );
} }