From c5673a147c40d84a53479d263b47b301e1cf2734 Mon Sep 17 00:00:00 2001 From: efrilm Date: Mon, 10 Nov 2025 13:06:06 +0700 Subject: [PATCH] void struck --- .../print_struck/print_struck_bloc.dart | 16 ++ .../print_struck_bloc.freezed.dart | 185 ++++++++++++++++++ .../print_struck/print_struck_event.dart | 1 + .../repositories/i_printer_repository.dart | 1 + .../repositories/printer_repository.dart | 57 ++++++ .../components/print/print_ui.dart | 66 +++++++ .../pages/void_success/void_success_page.dart | 58 ++++-- .../widgets/void_success_left_panel.dart | 32 +-- 8 files changed, 381 insertions(+), 35 deletions(-) diff --git a/lib/application/printer/print_struck/print_struck_bloc.dart b/lib/application/printer/print_struck/print_struck_bloc.dart index 111aa31..5b8d0ad 100644 --- a/lib/application/printer/print_struck/print_struck_bloc.dart +++ b/lib/application/printer/print_struck/print_struck_bloc.dart @@ -64,6 +64,22 @@ class PrintStruckBloc extends Bloc { order: e.order, ); + emit( + state.copyWith( + isPrinting: false, + failureOrPrintStruck: optionOf(failureOrSuccess), + ), + ); + }, + voided: (e) async { + Either failureOrSuccess; + + emit(state.copyWith(isPrinting: true, failureOrPrintStruck: none())); + + failureOrSuccess = await _printerRepository.printStruckVoid( + order: e.order, + ); + emit( state.copyWith( isPrinting: false, diff --git a/lib/application/printer/print_struck/print_struck_bloc.freezed.dart b/lib/application/printer/print_struck/print_struck_bloc.freezed.dart index d0ee0ff..79f0686 100644 --- a/lib/application/printer/print_struck/print_struck_bloc.freezed.dart +++ b/lib/application/printer/print_struck/print_struck_bloc.freezed.dart @@ -23,18 +23,21 @@ mixin _$PrintStruckEvent { required TResult Function(Order order) order, required TResult Function(Order order) cashier, required TResult Function(Order order) payment, + required TResult Function(Order order) voided, }) => throw _privateConstructorUsedError; @optionalTypeArgs TResult? whenOrNull({ TResult? Function(Order order)? order, TResult? Function(Order order)? cashier, TResult? Function(Order order)? payment, + TResult? Function(Order order)? voided, }) => throw _privateConstructorUsedError; @optionalTypeArgs TResult maybeWhen({ TResult Function(Order order)? order, TResult Function(Order order)? cashier, TResult Function(Order order)? payment, + TResult Function(Order order)? voided, required TResult orElse(), }) => throw _privateConstructorUsedError; @optionalTypeArgs @@ -42,18 +45,21 @@ mixin _$PrintStruckEvent { required TResult Function(_Order value) order, required TResult Function(_Cashier value) cashier, required TResult Function(_Payment value) payment, + required TResult Function(_Voided value) voided, }) => throw _privateConstructorUsedError; @optionalTypeArgs TResult? mapOrNull({ TResult? Function(_Order value)? order, TResult? Function(_Cashier value)? cashier, TResult? Function(_Payment value)? payment, + TResult? Function(_Voided value)? voided, }) => throw _privateConstructorUsedError; @optionalTypeArgs TResult maybeMap({ TResult Function(_Order value)? order, TResult Function(_Cashier value)? cashier, TResult Function(_Payment value)? payment, + TResult Function(_Voided value)? voided, required TResult orElse(), }) => throw _privateConstructorUsedError; @@ -191,6 +197,7 @@ class _$OrderImpl implements _Order { required TResult Function(Order order) order, required TResult Function(Order order) cashier, required TResult Function(Order order) payment, + required TResult Function(Order order) voided, }) { return order(this.order); } @@ -201,6 +208,7 @@ class _$OrderImpl implements _Order { TResult? Function(Order order)? order, TResult? Function(Order order)? cashier, TResult? Function(Order order)? payment, + TResult? Function(Order order)? voided, }) { return order?.call(this.order); } @@ -211,6 +219,7 @@ class _$OrderImpl implements _Order { TResult Function(Order order)? order, TResult Function(Order order)? cashier, TResult Function(Order order)? payment, + TResult Function(Order order)? voided, required TResult orElse(), }) { if (order != null) { @@ -225,6 +234,7 @@ class _$OrderImpl implements _Order { required TResult Function(_Order value) order, required TResult Function(_Cashier value) cashier, required TResult Function(_Payment value) payment, + required TResult Function(_Voided value) voided, }) { return order(this); } @@ -235,6 +245,7 @@ class _$OrderImpl implements _Order { TResult? Function(_Order value)? order, TResult? Function(_Cashier value)? cashier, TResult? Function(_Payment value)? payment, + TResult? Function(_Voided value)? voided, }) { return order?.call(this); } @@ -245,6 +256,7 @@ class _$OrderImpl implements _Order { TResult Function(_Order value)? order, TResult Function(_Cashier value)? cashier, TResult Function(_Payment value)? payment, + TResult Function(_Voided value)? voided, required TResult orElse(), }) { if (order != null) { @@ -346,6 +358,7 @@ class _$CashierImpl implements _Cashier { required TResult Function(Order order) order, required TResult Function(Order order) cashier, required TResult Function(Order order) payment, + required TResult Function(Order order) voided, }) { return cashier(this.order); } @@ -356,6 +369,7 @@ class _$CashierImpl implements _Cashier { TResult? Function(Order order)? order, TResult? Function(Order order)? cashier, TResult? Function(Order order)? payment, + TResult? Function(Order order)? voided, }) { return cashier?.call(this.order); } @@ -366,6 +380,7 @@ class _$CashierImpl implements _Cashier { TResult Function(Order order)? order, TResult Function(Order order)? cashier, TResult Function(Order order)? payment, + TResult Function(Order order)? voided, required TResult orElse(), }) { if (cashier != null) { @@ -380,6 +395,7 @@ class _$CashierImpl implements _Cashier { required TResult Function(_Order value) order, required TResult Function(_Cashier value) cashier, required TResult Function(_Payment value) payment, + required TResult Function(_Voided value) voided, }) { return cashier(this); } @@ -390,6 +406,7 @@ class _$CashierImpl implements _Cashier { TResult? Function(_Order value)? order, TResult? Function(_Cashier value)? cashier, TResult? Function(_Payment value)? payment, + TResult? Function(_Voided value)? voided, }) { return cashier?.call(this); } @@ -400,6 +417,7 @@ class _$CashierImpl implements _Cashier { TResult Function(_Order value)? order, TResult Function(_Cashier value)? cashier, TResult Function(_Payment value)? payment, + TResult Function(_Voided value)? voided, required TResult orElse(), }) { if (cashier != null) { @@ -501,6 +519,7 @@ class _$PaymentImpl implements _Payment { required TResult Function(Order order) order, required TResult Function(Order order) cashier, required TResult Function(Order order) payment, + required TResult Function(Order order) voided, }) { return payment(this.order); } @@ -511,6 +530,7 @@ class _$PaymentImpl implements _Payment { TResult? Function(Order order)? order, TResult? Function(Order order)? cashier, TResult? Function(Order order)? payment, + TResult? Function(Order order)? voided, }) { return payment?.call(this.order); } @@ -521,6 +541,7 @@ class _$PaymentImpl implements _Payment { TResult Function(Order order)? order, TResult Function(Order order)? cashier, TResult Function(Order order)? payment, + TResult Function(Order order)? voided, required TResult orElse(), }) { if (payment != null) { @@ -535,6 +556,7 @@ class _$PaymentImpl implements _Payment { required TResult Function(_Order value) order, required TResult Function(_Cashier value) cashier, required TResult Function(_Payment value) payment, + required TResult Function(_Voided value) voided, }) { return payment(this); } @@ -545,6 +567,7 @@ class _$PaymentImpl implements _Payment { TResult? Function(_Order value)? order, TResult? Function(_Cashier value)? cashier, TResult? Function(_Payment value)? payment, + TResult? Function(_Voided value)? voided, }) { return payment?.call(this); } @@ -555,6 +578,7 @@ class _$PaymentImpl implements _Payment { TResult Function(_Order value)? order, TResult Function(_Cashier value)? cashier, TResult Function(_Payment value)? payment, + TResult Function(_Voided value)? voided, required TResult orElse(), }) { if (payment != null) { @@ -578,6 +602,167 @@ abstract class _Payment implements PrintStruckEvent { throw _privateConstructorUsedError; } +/// @nodoc +abstract class _$$VoidedImplCopyWith<$Res> + implements $PrintStruckEventCopyWith<$Res> { + factory _$$VoidedImplCopyWith( + _$VoidedImpl value, + $Res Function(_$VoidedImpl) then, + ) = __$$VoidedImplCopyWithImpl<$Res>; + @override + @useResult + $Res call({Order order}); + + @override + $OrderCopyWith<$Res> get order; +} + +/// @nodoc +class __$$VoidedImplCopyWithImpl<$Res> + extends _$PrintStruckEventCopyWithImpl<$Res, _$VoidedImpl> + implements _$$VoidedImplCopyWith<$Res> { + __$$VoidedImplCopyWithImpl( + _$VoidedImpl _value, + $Res Function(_$VoidedImpl) _then, + ) : super(_value, _then); + + /// Create a copy of PrintStruckEvent + /// with the given fields replaced by the non-null parameter values. + @pragma('vm:prefer-inline') + @override + $Res call({Object? order = null}) { + return _then( + _$VoidedImpl( + null == order + ? _value.order + : order // ignore: cast_nullable_to_non_nullable + as Order, + ), + ); + } +} + +/// @nodoc + +class _$VoidedImpl implements _Voided { + const _$VoidedImpl(this.order); + + @override + final Order order; + + @override + String toString() { + return 'PrintStruckEvent.voided(order: $order)'; + } + + @override + bool operator ==(Object other) { + return identical(this, other) || + (other.runtimeType == runtimeType && + other is _$VoidedImpl && + (identical(other.order, order) || other.order == order)); + } + + @override + int get hashCode => Object.hash(runtimeType, order); + + /// Create a copy of PrintStruckEvent + /// with the given fields replaced by the non-null parameter values. + @JsonKey(includeFromJson: false, includeToJson: false) + @override + @pragma('vm:prefer-inline') + _$$VoidedImplCopyWith<_$VoidedImpl> get copyWith => + __$$VoidedImplCopyWithImpl<_$VoidedImpl>(this, _$identity); + + @override + @optionalTypeArgs + TResult when({ + required TResult Function(Order order) order, + required TResult Function(Order order) cashier, + required TResult Function(Order order) payment, + required TResult Function(Order order) voided, + }) { + return voided(this.order); + } + + @override + @optionalTypeArgs + TResult? whenOrNull({ + TResult? Function(Order order)? order, + TResult? Function(Order order)? cashier, + TResult? Function(Order order)? payment, + TResult? Function(Order order)? voided, + }) { + return voided?.call(this.order); + } + + @override + @optionalTypeArgs + TResult maybeWhen({ + TResult Function(Order order)? order, + TResult Function(Order order)? cashier, + TResult Function(Order order)? payment, + TResult Function(Order order)? voided, + required TResult orElse(), + }) { + if (voided != null) { + return voided(this.order); + } + return orElse(); + } + + @override + @optionalTypeArgs + TResult map({ + required TResult Function(_Order value) order, + required TResult Function(_Cashier value) cashier, + required TResult Function(_Payment value) payment, + required TResult Function(_Voided value) voided, + }) { + return voided(this); + } + + @override + @optionalTypeArgs + TResult? mapOrNull({ + TResult? Function(_Order value)? order, + TResult? Function(_Cashier value)? cashier, + TResult? Function(_Payment value)? payment, + TResult? Function(_Voided value)? voided, + }) { + return voided?.call(this); + } + + @override + @optionalTypeArgs + TResult maybeMap({ + TResult Function(_Order value)? order, + TResult Function(_Cashier value)? cashier, + TResult Function(_Payment value)? payment, + TResult Function(_Voided value)? voided, + required TResult orElse(), + }) { + if (voided != null) { + return voided(this); + } + return orElse(); + } +} + +abstract class _Voided implements PrintStruckEvent { + const factory _Voided(final Order order) = _$VoidedImpl; + + @override + Order get order; + + /// Create a copy of PrintStruckEvent + /// with the given fields replaced by the non-null parameter values. + @override + @JsonKey(includeFromJson: false, includeToJson: false) + _$$VoidedImplCopyWith<_$VoidedImpl> get copyWith => + throw _privateConstructorUsedError; +} + /// @nodoc mixin _$PrintStruckState { Option> get failureOrPrintStruck => diff --git a/lib/application/printer/print_struck/print_struck_event.dart b/lib/application/printer/print_struck/print_struck_event.dart index a9747bb..a2d8368 100644 --- a/lib/application/printer/print_struck/print_struck_event.dart +++ b/lib/application/printer/print_struck/print_struck_event.dart @@ -5,4 +5,5 @@ class PrintStruckEvent with _$PrintStruckEvent { const factory PrintStruckEvent.order(Order order) = _Order; const factory PrintStruckEvent.cashier(Order order) = _Cashier; const factory PrintStruckEvent.payment(Order order) = _Payment; + const factory PrintStruckEvent.voided(Order order) = _Voided; } diff --git a/lib/domain/printer/repositories/i_printer_repository.dart b/lib/domain/printer/repositories/i_printer_repository.dart index 99d66b1..f8375a0 100644 --- a/lib/domain/printer/repositories/i_printer_repository.dart +++ b/lib/domain/printer/repositories/i_printer_repository.dart @@ -25,4 +25,5 @@ abstract class IPrinterRepository { Future> printStruckPayment({ required Order order, }); + Future> printStruckVoid({required Order order}); } diff --git a/lib/infrastructure/printer/repositories/printer_repository.dart b/lib/infrastructure/printer/repositories/printer_repository.dart index 8ea4815..8276c12 100644 --- a/lib/infrastructure/printer/repositories/printer_repository.dart +++ b/lib/infrastructure/printer/repositories/printer_repository.dart @@ -489,6 +489,15 @@ class PrinterRepository implements IPrinterRepository { return _printCashier(order: order, outlet: outlet, cashieName: user.name); } + @override + Future> printStruckVoid({ + required Order order, + }) async { + final outlet = await _outletLocalDatasource.currentOutlet(); + final user = await _authLocalDataProvider.currentUser(); + return _printVoid(order: order, outlet: outlet, cashieName: user.name); + } + @override Future> printStruckPayment({ required Order order, @@ -766,4 +775,52 @@ class PrinterRepository implements IPrinterRepository { ); } } + + Future> _printVoid({ + required Order order, + required Outlet outlet, + required String cashieName, + }) async { + log('Starting to print void', name: _logName); + final voidPrinter = await _localDataProvider.findPrinterByCode('receipt'); + if (voidPrinter.hasData) { + try { + final printer = voidPrinter.data!.toDomain(); + + final printValue = await PrintUi().printVoid( + order: order, + outlet: outlet, + cashierName: cashieName, + ); + + await printStruct(printer, printValue); + + log('Finished printed void', name: _logName); + + return right(unit); + } catch (e, stackTrace) { + FirebaseCrashlytics.instance.recordError( + 'Print Struck Void Error: $e', + stackTrace, + reason: + 'Print struck void error / Printer not setting in printer page', + information: ['Order ID: ${order.id}'], + ); + log("Error printing void", name: _logName, error: e); + return left( + const PrinterFailure.dynamicErrorMessage('Error printing void order'), + ); + } + } else { + FirebaseCrashlytics.instance.recordError( + 'Void printer not found', + null, + reason: 'Void printer not found / Printer not setting in printer page', + information: ['Order ID: ${order.id}'], + ); + return left( + const PrinterFailure.dynamicErrorMessage('Void printer not found'), + ); + } + } } diff --git a/lib/presentation/components/print/print_ui.dart b/lib/presentation/components/print/print_ui.dart index 1bba6ab..ee292f5 100644 --- a/lib/presentation/components/print/print_ui.dart +++ b/lib/presentation/components/print/print_ui.dart @@ -348,4 +348,70 @@ class PrintUi { return bytes; } + + Future> printVoid({ + required Order order, + required Outlet outlet, + required String cashierName, + int paper = 58, + }) async { + List bytes = []; + + final profile = await CapabilityProfile.load(); + final generator = Generator( + paper == 58 ? PaperSize.mm58 : PaperSize.mm80, + profile, + ); + final builder = ReceiptComponentBuilder( + generator: generator, + paperSize: 58, + ); + + bytes += generator.reset(); + + bytes += builder.header( + outletName: outlet.name, + address: outlet.address, + phoneNumber: outlet.phoneNumber, + ); + + bytes += builder.dateTime(DateTime.now()); + + bytes += builder.orderInfo( + orderNumber: order.orderNumber, + customerName: order.metadata['customer_name'] ?? 'John Doe', + cashierName: cashierName, + paymentMethod: order.payments.isEmpty + ? null + : order.payments.last.paymentMethodName, + tableNumber: order.tableNumber, + ); + + bytes += builder.orderType('Void'); + + bytes += builder.emptyLines(1); + + for (final item in order.orderItems) { + bytes += builder.orderItem( + productName: item.productName, + quantity: item.quantity, + unitPrice: item.unitPrice.currencyFormatRpV2, + totalPrice: item.totalPrice.currencyFormatRpV2, + variantName: item.productVariantName, + notes: item.notes, + ); + } + + bytes += builder.summary( + totalItems: order.orderItems.length, + subtotal: order.subtotal.currencyFormatRpV2, + discount: order.discountAmount.currencyFormatRpV2, + total: order.totalAmount.currencyFormatRpV2, + paid: order.totalPaid.currencyFormatRpV2, + ); + + bytes += builder.footer(message: 'Kasir'); + + return bytes; + } } diff --git a/lib/presentation/pages/void/pages/void_success/void_success_page.dart b/lib/presentation/pages/void/pages/void_success/void_success_page.dart index 7b69949..c30e8cd 100644 --- a/lib/presentation/pages/void/pages/void_success/void_success_page.dart +++ b/lib/presentation/pages/void/pages/void_success/void_success_page.dart @@ -2,9 +2,11 @@ import 'package:auto_route/auto_route.dart'; import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; +import '../../../../../application/printer/print_struck/print_struck_bloc.dart'; import '../../../../../application/void/void_form/void_form_bloc.dart'; import '../../../../../common/theme/theme.dart'; import '../../../../components/spaces/space.dart'; +import '../../../../components/toast/flushbar.dart'; import 'widgets/void_success_left_panel.dart'; import 'widgets/void_success_right_panel.dart'; @@ -14,26 +16,44 @@ class VoidSuccessPage extends StatelessWidget { @override Widget build(BuildContext context) { - return Scaffold( - backgroundColor: AppColor.background, - body: BlocBuilder( - builder: (context, state) { - return SafeArea( - child: Padding( - padding: const EdgeInsets.all(24.0), - child: Row( - children: [ - Expanded(flex: 35, child: VoidSuccessLeftPanel(state: state)), - SpaceWidth(16), - Expanded( - flex: 65, - child: VoidSuccessRightPanel(state: state), - ), - ], + return BlocListener( + listenWhen: (previous, current) => + previous.failureOrPrintStruck != current.failureOrPrintStruck, + listener: (context, state) { + state.failureOrPrintStruck.fold( + () {}, + (either) => either.fold( + (f) => AppFlushbar.showPrinterFailureToast(context, f), + (success) { + AppFlushbar.showSuccess(context, "Struck berhasil dicetak"); + }, + ), + ); + }, + child: Scaffold( + backgroundColor: AppColor.background, + body: BlocBuilder( + builder: (context, state) { + return SafeArea( + child: Padding( + padding: const EdgeInsets.all(24.0), + child: Row( + children: [ + Expanded( + flex: 35, + child: VoidSuccessLeftPanel(state: state), + ), + SpaceWidth(16), + Expanded( + flex: 65, + child: VoidSuccessRightPanel(state: state), + ), + ], + ), ), - ), - ); - }, + ); + }, + ), ), ); } diff --git a/lib/presentation/pages/void/pages/void_success/widgets/void_success_left_panel.dart b/lib/presentation/pages/void/pages/void_success/widgets/void_success_left_panel.dart index 3cb2f97..3239008 100644 --- a/lib/presentation/pages/void/pages/void_success/widgets/void_success_left_panel.dart +++ b/lib/presentation/pages/void/pages/void_success/widgets/void_success_left_panel.dart @@ -2,6 +2,7 @@ import 'package:auto_route/auto_route.dart'; import 'package:flutter/material.dart'; import 'package:flutter_bloc/flutter_bloc.dart'; +import '../../../../../../application/printer/print_struck/print_struck_bloc.dart'; import '../../../../../../application/void/void_form/void_form_bloc.dart'; import '../../../../../../common/extension/extension.dart'; import '../../../../../../common/theme/theme.dart'; @@ -149,22 +150,21 @@ class VoidSuccessLeftPanel extends StatelessWidget { Expanded( child: AppElevatedButton.filled( onPressed: () { - // onPrintRecipt( - // context, - // order: widget.order, - // paymentMethod: widget.paymentMethod, - // nominalBayar: widget.paymentMethod == "Cash" - // ? widget.nominalBayar - // : widget.order.totalAmount ?? 0, - // kembalian: widget.nominalBayar - - // (widget.order.totalAmount ?? 0), - // productQuantity: widget.productQuantity, - // ); - // onPrint( - // context, - // productQuantity: widget.productQuantity, - // order: widget.order, - // ); + context.read().add( + PrintStruckEvent.voided( + state.order.copyWith( + orderItems: state.voidType.isAll + ? state.order.orderItems + : state.voidItems, + subtotal: state.voidType.isAll + ? state.order.totalAmount + : state.totalPriceVoid, + totalAmount: state.voidType.isAll + ? state.order.totalAmount + : state.totalPriceVoid, + ), + ), + ); }, label: 'Cetak Struk', icon: Icon(Icons.print_rounded, color: AppColor.white),