import 'dart:async'; import 'package:redux/redux.dart'; import 'package:flutter/material.dart'; import 'package:flutter/foundation.dart'; import 'package:flutter/widgets.dart'; import 'package:flutter_redux/flutter_redux.dart'; import 'package:built_collection/built_collection.dart'; import 'package:invoiceninja_flutter/redux/client/client_actions.dart'; import 'package:invoiceninja_flutter/redux/ui/list_ui_state.dart'; import 'package:invoiceninja_flutter/utils/completers.dart'; import 'package:invoiceninja_flutter/utils/localization.dart'; import 'package:invoiceninja_flutter/redux/expense/expense_selectors.dart'; import 'package:invoiceninja_flutter/data/models/models.dart'; import 'package:invoiceninja_flutter/ui/expense/expense_list.dart'; import 'package:invoiceninja_flutter/redux/app/app_state.dart'; import 'package:invoiceninja_flutter/redux/expense/expense_actions.dart'; class ExpenseListBuilder extends StatelessWidget { const ExpenseListBuilder({Key key}) : super(key: key); @override Widget build(BuildContext context) { return StoreConnector( converter: ExpenseListVM.fromStore, builder: (context, viewModel) { return ExpenseList( viewModel: viewModel, ); }, ); } } class ExpenseListVM { ExpenseListVM({ @required this.user, @required this.expenseList, @required this.expenseMap, @required this.filter, @required this.isLoading, @required this.isLoaded, @required this.onExpenseTap, @required this.listState, @required this.onRefreshed, @required this.onEntityAction, @required this.onClearEntityFilterPressed, @required this.onViewEntityFilterPressed, }); static ExpenseListVM fromStore(Store store) { Future _handleRefresh(BuildContext context) { if (store.state.isLoading) { return Future(null); } final completer = snackBarCompleter( context, AppLocalization.of(context).refreshComplete); store.dispatch(LoadExpenses(completer: completer, force: true)); return completer.future; } final state = store.state; return ExpenseListVM( user: state.user, listState: state.expenseListState, expenseList: memoizedFilteredExpenseList(state.expenseState.map, state.expenseState.list, state.expenseListState), expenseMap: state.expenseState.map, isLoading: state.isLoading, isLoaded: state.expenseState.isLoaded, filter: state.expenseUIState.listUIState.filter, onClearEntityFilterPressed: () => store.dispatch(FilterExpensesByEntity()), onViewEntityFilterPressed: (BuildContext context) => store.dispatch( ViewClient( clientId: state.expenseListState.filterEntityId, context: context)), onExpenseTap: (context, expense) { store.dispatch(ViewExpense(expenseId: expense.id, context: context)); }, onEntityAction: (context, expense, action) { switch (action) { case EntityAction.edit: store.dispatch(EditExpense(context: context, expense: expense)); break; case EntityAction.clone: Navigator.of(context).pop(); store.dispatch( EditExpense(context: context, expense: expense.clone)); break; case EntityAction.restore: store.dispatch(RestoreExpenseRequest( snackBarCompleter( context, AppLocalization.of(context).restoredExpense), expense.id)); break; case EntityAction.archive: store.dispatch(ArchiveExpenseRequest( snackBarCompleter( context, AppLocalization.of(context).archivedExpense), expense.id)); break; case EntityAction.delete: store.dispatch(DeleteExpenseRequest( snackBarCompleter( context, AppLocalization.of(context).deletedExpense), expense.id)); break; } }, onRefreshed: (context) => _handleRefresh(context), ); } final UserEntity user; final List expenseList; final BuiltMap expenseMap; final ListUIState listState; final String filter; final bool isLoading; final bool isLoaded; final Function(BuildContext, ExpenseEntity) onExpenseTap; final Function(BuildContext) onRefreshed; final Function(BuildContext, ExpenseEntity, EntityAction) onEntityAction; final Function onClearEntityFilterPressed; final Function(BuildContext) onViewEntityFilterPressed; }