import 'dart:async'; import 'package:flutter/foundation.dart'; import 'package:flutter/material.dart'; import 'package:flutter_redux/flutter_redux.dart'; import 'package:invoiceninja_flutter/data/models/expense_model.dart'; import 'package:invoiceninja_flutter/data/models/models.dart'; import 'package:invoiceninja_flutter/redux/app/app_state.dart'; import 'package:invoiceninja_flutter/redux/client/client_actions.dart'; import 'package:invoiceninja_flutter/redux/document/document_actions.dart'; import 'package:invoiceninja_flutter/redux/expense/expense_actions.dart'; import 'package:invoiceninja_flutter/redux/invoice/invoice_actions.dart'; import 'package:invoiceninja_flutter/redux/ui/ui_actions.dart'; import 'package:invoiceninja_flutter/redux/vendor/vendor_actions.dart'; import 'package:invoiceninja_flutter/ui/app/dialogs/error_dialog.dart'; import 'package:invoiceninja_flutter/ui/app/entities/entity_actions_dialog.dart'; import 'package:invoiceninja_flutter/ui/app/snackbar_row.dart'; import 'package:invoiceninja_flutter/ui/expense/expense_screen.dart'; import 'package:invoiceninja_flutter/ui/expense/view/expense_view.dart'; import 'package:invoiceninja_flutter/utils/completers.dart'; import 'package:invoiceninja_flutter/utils/localization.dart'; import 'package:redux/redux.dart'; class ExpenseViewScreen extends StatelessWidget { const ExpenseViewScreen({Key key}) : super(key: key); static const String route = '/expense/view'; @override Widget build(BuildContext context) { return StoreConnector( converter: (Store store) { return ExpenseViewVM.fromStore(store); }, builder: (context, vm) { return ExpenseView( viewModel: vm, ); }, ); } } class ExpenseViewVM { ExpenseViewVM({ @required this.state, @required this.expense, @required this.company, @required this.onEntityAction, @required this.onEntityPressed, @required this.onEditPressed, @required this.onBackPressed, @required this.onRefreshed, @required this.onUploadDocument, @required this.onDeleteDocument, @required this.isSaving, @required this.isLoading, @required this.isDirty, }); factory ExpenseViewVM.fromStore(Store store) { final state = store.state; final userCompany = state.userCompany; final expense = state.expenseState.map[state.expenseUIState.selectedId] ?? ExpenseEntity(id: state.expenseUIState.selectedId); final vendor = state.vendorState.map[expense.vendorId]; final client = state.clientState.map[expense.clientId]; final invoice = state.invoiceState.map[expense.invoiceId]; Future _handleRefresh(BuildContext context) { final completer = snackBarCompleter( context, AppLocalization.of(context).refreshComplete); store.dispatch(LoadExpense(completer: completer, expenseId: expense.id)); return completer.future; } return ExpenseViewVM( state: state, company: state.selectedCompany, isSaving: state.isSaving, isLoading: state.isLoading, isDirty: expense.isNew, expense: expense, onEditPressed: (BuildContext context) { final Completer completer = Completer(); store.dispatch(EditExpense( expense: expense, context: context, completer: completer)); completer.future.then((expense) { Scaffold.of(context).showSnackBar(SnackBar( content: SnackBarRow( message: AppLocalization.of(context).updatedExpense, ))); }); }, onRefreshed: (context) => _handleRefresh(context), onBackPressed: () { if (state.uiState.currentRoute.contains(ExpenseScreen.route)) { store.dispatch(UpdateCurrentRoute(ExpenseScreen.route)); } }, onEntityPressed: (BuildContext context, EntityType entityType, [longPress = false]) { switch (entityType) { case EntityType.vendor: if (longPress) { showEntityActionsDialog( userCompany: userCompany, context: context, entities: [vendor], onEntityAction: (BuildContext context, List vendors, EntityAction action) => handleVendorAction(context, vendors, action)); } else { store.dispatch( ViewVendor(vendorId: vendor.id, context: context)); } break; case EntityType.client: if (longPress) { showEntityActionsDialog( userCompany: userCompany, context: context, entities: [client], onEntityAction: (BuildContext context, List clients, EntityAction action) => handleClientAction(context, clients, action)); } else { store.dispatch( ViewClient(clientId: client.id, context: context)); } break; case EntityType.invoice: if (longPress) { showEntityActionsDialog( userCompany: userCompany, context: context, entities: [invoice], client: client, onEntityAction: (BuildContext context, List invoices, EntityAction action) => handleInvoiceAction(context, invoices, action)); } else { store.dispatch( ViewInvoice(invoiceId: invoice.id, context: context)); } break; } }, onEntityAction: (BuildContext context, EntityAction action) => handleExpenseAction(context, expense, action), onUploadDocument: (BuildContext context, String path) { final Completer completer = Completer(); final document = DocumentEntity().rebuild((b) => b ..expenseId = expense.id ..path = path); store.dispatch( SaveDocumentRequest(document: document, completer: completer)); completer.future.then((client) { Scaffold.of(context).showSnackBar(SnackBar( content: SnackBarRow( message: AppLocalization.of(context).uploadedDocument, ))); }).catchError((Object error) { showDialog( context: context, builder: (BuildContext context) { return ErrorDialog(error); }); }); }, onDeleteDocument: (BuildContext context, DocumentEntity document) { store.dispatch(DeleteDocumentRequest( snackBarCompleter( context, AppLocalization.of(context).deletedDocument), document.id)); }); } final AppState state; final ExpenseEntity expense; final CompanyEntity company; final Function(BuildContext, EntityAction) onEntityAction; final Function(BuildContext, EntityType, [bool]) onEntityPressed; final Function(BuildContext) onEditPressed; final Function onBackPressed; final Function(BuildContext) onRefreshed; final Function(BuildContext, String) onUploadDocument; final Function(BuildContext, DocumentEntity) onDeleteDocument; final bool isSaving; final bool isLoading; final bool isDirty; }