import 'dart:async'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_redux/flutter_redux.dart'; import 'package:flutter_styled_toast/flutter_styled_toast.dart'; import 'package:invoiceninja_flutter/redux/app/app_actions.dart'; import 'package:invoiceninja_flutter/redux/ui/ui_actions.dart'; import 'package:invoiceninja_flutter/utils/completers.dart'; import 'package:invoiceninja_flutter/main_app.dart'; import 'package:redux/redux.dart'; import 'package:invoiceninja_flutter/data/models/models.dart'; import 'package:invoiceninja_flutter/ui/app/dialogs/error_dialog.dart'; import 'package:invoiceninja_flutter/ui/expense/view/expense_view_vm.dart'; import 'package:invoiceninja_flutter/redux/expense/expense_actions.dart'; import 'package:invoiceninja_flutter/data/models/expense_model.dart'; import 'package:invoiceninja_flutter/ui/expense/edit/expense_edit.dart'; import 'package:invoiceninja_flutter/redux/app/app_state.dart'; class ExpenseEditScreen extends StatelessWidget { const ExpenseEditScreen({Key key}) : super(key: key); static const String route = '/expense/edit'; @override Widget build(BuildContext context) { return StoreConnector( converter: (Store store) { return ExpenseEditVM.fromStore(store); }, builder: (context, viewModel) { return ExpenseEdit( viewModel: viewModel, key: ValueKey(viewModel.expense.id), ); }, ); } } abstract class AbstractExpenseEditVM { AbstractExpenseEditVM({ @required this.state, @required this.expense, @required this.onChanged, @required this.origExpense, @required this.onSavePressed, @required this.onCancelPressed, @required this.onAddClientPressed, @required this.onAddVendorPressed, }); final ExpenseEntity expense; final Function(ExpenseEntity) onChanged; final Function(BuildContext) onSavePressed; final Function(BuildContext) onCancelPressed; final ExpenseEntity origExpense; final AppState state; final Function(BuildContext context, Completer completer) onAddClientPressed; final Function(BuildContext context, Completer completer) onAddVendorPressed; } class ExpenseEditVM extends AbstractExpenseEditVM { ExpenseEditVM({ AppState state, ExpenseEntity expense, Function(ExpenseEntity) onChanged, Function(BuildContext) onSavePressed, Function(BuildContext) onCancelPressed, bool isLoading, bool isSaving, ExpenseEntity origExpense, Function(BuildContext context, Completer completer) onAddClientPressed, Function(BuildContext context, Completer completer) onAddVendorPressed, }) : super( state: state, expense: expense, onChanged: onChanged, onSavePressed: onSavePressed, onCancelPressed: onCancelPressed, origExpense: origExpense, onAddClientPressed: onAddClientPressed, onAddVendorPressed: onAddVendorPressed, ); factory ExpenseEditVM.fromStore(Store store) { final expense = store.state.expenseUIState.editing; final state = store.state; return ExpenseEditVM( state: state, isLoading: state.isLoading, isSaving: state.isSaving, origExpense: state.expenseState.map[expense.id], expense: expense, onChanged: (ExpenseEntity expense) { store.dispatch(UpdateExpense(expense)); }, onCancelPressed: (BuildContext context) { createEntity(context: context, entity: ExpenseEntity(), force: true); store.dispatch(UpdateCurrentRoute(state.uiState.previousRoute)); }, onAddClientPressed: (context, completer) { createEntity( context: context, entity: ClientEntity(), force: true, completer: completer, cancelCompleter: Completer() ..future.then((_) { store.dispatch(UpdateCurrentRoute(ExpenseEditScreen.route)); })); completer.future.then((SelectableEntity client) { store.dispatch(UpdateCurrentRoute(ExpenseEditScreen.route)); }); }, onAddVendorPressed: (context, completer) { createEntity( context: context, entity: VendorEntity(), force: true, completer: completer, cancelCompleter: Completer() ..future.then((_) { store.dispatch(UpdateCurrentRoute(ExpenseEditScreen.route)); })); completer.future.then((SelectableEntity expense) { store.dispatch(UpdateCurrentRoute(ExpenseEditScreen.route)); }); }, onSavePressed: (BuildContext context) { Debouncer.runOnComplete(() { final expense = store.state.expenseUIState.editing; final localization = navigatorKey.localization; final navigator = navigatorKey.currentState; final Completer completer = new Completer(); store.dispatch( SaveExpenseRequest(completer: completer, expense: expense)); return completer.future.then((savedExpense) { showToast(expense.isNew ? localization.createdExpense : localization.updatedExpense); if (state.prefState.isMobile) { store.dispatch(UpdateCurrentRoute(ExpenseViewScreen.route)); if (expense.isNew) { navigator.pushReplacementNamed(ExpenseViewScreen.route); } else { navigator.pop(savedExpense); } } else { if (state.prefState.isPreviewEnabled) { viewEntity(entity: savedExpense, force: true); } else { editEntity( context: navigatorKey.currentContext, entity: savedExpense, force: true); } } }).catchError((Object error) { showDialog( context: navigatorKey.currentContext, builder: (BuildContext context) { return ErrorDialog(error); }); }); }); }, ); } }