diff --git a/lib/redux/recurring_invoice/recurring_invoice_actions.dart b/lib/redux/recurring_invoice/recurring_invoice_actions.dart index 55b62c09c..81cc9a9e0 100644 --- a/lib/redux/recurring_invoice/recurring_invoice_actions.dart +++ b/lib/redux/recurring_invoice/recurring_invoice_actions.dart @@ -38,15 +38,25 @@ class EditRecurringInvoice extends AbstractNavigatorAction @required NavigatorState navigator, this.completer, this.cancelCompleter, + this.itemIndex, this.force = false}) : super(navigator: navigator); final InvoiceEntity recurringInvoice; + final int itemIndex; final Completer completer; final Completer cancelCompleter; final bool force; } +class ShowEmailRecurringInvoice { + ShowEmailRecurringInvoice({this.invoice, this.context, this.completer}); + + final InvoiceEntity invoice; + final BuildContext context; + final Completer completer; +} + class EditRecurringInvoiceItem implements PersistUI { EditRecurringInvoiceItem([this.itemIndex]); @@ -65,7 +75,6 @@ class UpdateRecurringInvoiceClient implements PersistUI { final ClientEntity client; } - class LoadRecurringInvoice { LoadRecurringInvoice({this.completer, this.recurringInvoiceId}); @@ -134,6 +143,20 @@ class LoadRecurringInvoicesSuccess implements StopLoading { } } +class AddRecurringInvoiceContact implements PersistUI { + AddRecurringInvoiceContact({this.contact, this.invitation}); + + final ContactEntity contact; + final InvitationEntity invitation; +} + +class RemoveRecurringInvoiceContact implements PersistUI { + RemoveRecurringInvoiceContact({this.invitation}); + + final InvitationEntity invitation; +} + + class SaveRecurringInvoiceRequest implements StartSaving { SaveRecurringInvoiceRequest({this.completer, this.recurringInvoice}); @@ -266,6 +289,12 @@ class FilterRecurringInvoicesByState implements PersistUI { final EntityState state; } +class FilterRecurringInvoiceDropdown { + FilterRecurringInvoiceDropdown(this.filter); + + final String filter; +} + class FilterRecurringInvoicesByCustom1 implements PersistUI { FilterRecurringInvoicesByCustom1(this.value); diff --git a/lib/redux/recurring_invoice/recurring_invoice_reducer.dart b/lib/redux/recurring_invoice/recurring_invoice_reducer.dart index ecd71b5bd..b69589a94 100644 --- a/lib/redux/recurring_invoice/recurring_invoice_reducer.dart +++ b/lib/redux/recurring_invoice/recurring_invoice_reducer.dart @@ -9,20 +9,38 @@ import 'package:invoiceninja_flutter/redux/ui/list_ui_state.dart'; import 'package:invoiceninja_flutter/redux/recurring_invoice/recurring_invoice_state.dart'; import 'package:invoiceninja_flutter/data/models/entities.dart'; -EntityUIState recurringInvoiceUIReducer( - RecurringInvoiceUIState state, dynamic action) { +EntityUIState recurringInvoiceUIReducer(RecurringInvoiceUIState state, dynamic action) { return state.rebuild((b) => b - ..listUIState - .replace(recurringInvoiceListReducer(state.listUIState, action)) + ..listUIState.replace(recurringInvoiceListReducer(state.listUIState, action)) ..editing.replace(editingReducer(state.editing, action)) + ..editingItemIndex = editingItemIndexReducer(state.editingItemIndex, action) ..selectedId = selectedIdReducer(state.selectedId, action)); } +final editingItemIndexReducer = combineReducers([ + TypedReducer( + (index, action) => action.itemIndex), + TypedReducer( + (index, action) => action.itemIndex), +]); + +Reducer dropdownFilterReducer = combineReducers([ + TypedReducer( + filterRecurringInvoiceDropdownReducer), +]); + +String filterRecurringInvoiceDropdownReducer( + String dropdownFilter, FilterRecurringInvoiceDropdown action) { + return action.filter; +} + Reducer selectedIdReducer = combineReducers([ TypedReducer( - (String selectedId, dynamic action) => action.recurringInvoiceId), + (selectedId, action) => action.recurringInvoiceId), TypedReducer( - (String selectedId, dynamic action) => action.recurringInvoice.id), + (selectedId, action) => action.recurringInvoice.id), + TypedReducer( + (selectedId, action) => action.invoice.id), TypedReducer( (selectedId, action) => action.clearSelection ? '' : selectedId), TypedReducer( @@ -30,17 +48,40 @@ Reducer selectedIdReducer = combineReducers([ TypedReducer( (selectedId, action) => ''), TypedReducer((selectedId, action) => ''), - TypedReducer((selectedId, action) => - action.clearSelection - ? '' - : action.entityType == EntityType.recurringInvoice - ? action.entityId - : selectedId), + TypedReducer((selectedId, action) => + action.entityType == EntityType.recurringInvoice ? '' : selectedId), + TypedReducer((selectedId, action) => action + .clearSelection + ? '' + : action.entityType == EntityType.recurringInvoice ? action.entityId : selectedId), ]); final editingReducer = combineReducers([ TypedReducer(_updateEditing), TypedReducer(_updateEditing), + TypedReducer(_updateEditing), + TypedReducer((recurringInvoice, action) { + return action.recurringInvoice.rebuild((b) => b..isChanged = true); + }), + TypedReducer((recurringInvoice, action) { + return recurringInvoice.rebuild((b) => b..isChanged = true); + }), + TypedReducer((recurringInvoice, action) { + return recurringInvoice.rebuild((b) => b..isChanged = true); + }), + TypedReducer((recurringInvoice, action) { + return recurringInvoice.rebuild((b) => b..isChanged = true); + }), + TypedReducer((recurringInvoice, action) { + final client = action.client; + return recurringInvoice.rebuild((b) => b + ..isChanged = true + ..clientId = client?.id ?? '' + ..invitations.replace((client?.contacts ?? []) + .where((contact) => contact.sendEmail) + .map((contact) => InvitationEntity(contactId: contact.id)) + .toList())); + }), TypedReducer( (recurringInvoices, action) { return action.recurringInvoices[0]; @@ -53,12 +94,23 @@ final editingReducer = combineReducers([ (recurringInvoices, action) { return action.recurringInvoices[0]; }), - TypedReducer(_updateEditing), - TypedReducer( - (recurringInvoice, action) { - return action.recurringInvoice.rebuild((b) => b..isChanged = true); - }), + TypedReducer( + _addRecurringInvoiceItem), + TypedReducer( + _addRecurringInvoiceItems), + TypedReducer( + _removeRecurringInvoiceItem), + TypedReducer( + _updateRecurringInvoiceItem), TypedReducer(_clearEditing), + TypedReducer((recurringInvoice, action) { + return recurringInvoice.rebuild((b) => b + ..invitations.add( + action.invitation ?? InvitationEntity(contactId: action.contact.id))); + }), + TypedReducer((recurringInvoice, action) { + return recurringInvoice.rebuild((b) => b..invitations.remove(action.invitation)); + }), ]); InvoiceEntity _clearEditing(InvoiceEntity recurringInvoice, dynamic action) { @@ -69,15 +121,55 @@ InvoiceEntity _updateEditing(InvoiceEntity recurringInvoice, dynamic action) { return action.recurringInvoice; } +InvoiceEntity _addRecurringInvoiceItem( + InvoiceEntity recurringInvoice, AddRecurringInvoiceItem action) { + final item = action.recurringInvoiceItem ?? RecurringInvoiceItemEntity(); + return recurringInvoice.rebuild((b) => b + ..hasTasks = b.hasTasks || item.isTask + ..hasExpenses = b.hasExpenses || item.isExpense + ..lineItems.add(item)); +} + +InvoiceEntity _addRecurringInvoiceItems( + InvoiceEntity recurringInvoice, AddRecurringInvoiceItems action) { + return recurringInvoice.rebuild((b) => b + ..hasTasks = action.lineItems.where((item) => item.isTask).isNotEmpty + ..hasExpenses = action.lineItems.where((item) => item.isExpense).isNotEmpty + ..lineItems.addAll(action.lineItems)); +} + +InvoiceEntity _removeRecurringInvoiceItem( + InvoiceEntity recurringInvoice, DeleteRecurringInvoiceItem action) { + if (recurringInvoice.lineItems.length <= action.index) { + return recurringInvoice; + } + return recurringInvoice.rebuild((b) => b..lineItems.removeAt(action.index)); +} + +InvoiceEntity _updateRecurringInvoiceItem( + InvoiceEntity recurringInvoice, UpdateRecurringInvoiceItem action) { + if (recurringInvoice.lineItems.length <= action.index) { + return recurringInvoice; + } + return recurringInvoice + .rebuild((b) => b..lineItems[action.index] = action.recurringInvoiceItem); +} + final recurringInvoiceListReducer = combineReducers([ TypedReducer(_sortRecurringInvoices), TypedReducer( _filterRecurringInvoicesByState), + TypedReducer( + _filterRecurringInvoicesByStatus), TypedReducer(_filterRecurringInvoices), TypedReducer( _filterRecurringInvoicesByCustom1), TypedReducer( _filterRecurringInvoicesByCustom2), + TypedReducer( + _filterRecurringInvoicesByCustom3), + TypedReducer( + _filterRecurringInvoicesByCustom4), TypedReducer( _startListMultiselect), TypedReducer( @@ -89,38 +181,62 @@ final recurringInvoiceListReducer = combineReducers([ ]); ListUIState _filterRecurringInvoicesByCustom1( - ListUIState recurringInvoiceListState, - FilterRecurringInvoicesByCustom1 action) { + ListUIState recurringInvoiceListState, FilterRecurringInvoicesByCustom1 action) { if (recurringInvoiceListState.custom1Filters.contains(action.value)) { return recurringInvoiceListState .rebuild((b) => b..custom1Filters.remove(action.value)); } else { - return recurringInvoiceListState - .rebuild((b) => b..custom1Filters.add(action.value)); + return recurringInvoiceListState.rebuild((b) => b..custom1Filters.add(action.value)); } } ListUIState _filterRecurringInvoicesByCustom2( - ListUIState recurringInvoiceListState, - FilterRecurringInvoicesByCustom2 action) { + ListUIState recurringInvoiceListState, FilterRecurringInvoicesByCustom2 action) { if (recurringInvoiceListState.custom2Filters.contains(action.value)) { return recurringInvoiceListState .rebuild((b) => b..custom2Filters.remove(action.value)); } else { + return recurringInvoiceListState.rebuild((b) => b..custom2Filters.add(action.value)); + } +} + +ListUIState _filterRecurringInvoicesByCustom3( + ListUIState recurringInvoiceListState, FilterRecurringInvoicesByCustom3 action) { + if (recurringInvoiceListState.custom3Filters.contains(action.value)) { return recurringInvoiceListState - .rebuild((b) => b..custom2Filters.add(action.value)); + .rebuild((b) => b..custom3Filters.remove(action.value)); + } else { + return recurringInvoiceListState.rebuild((b) => b..custom3Filters.add(action.value)); + } +} + +ListUIState _filterRecurringInvoicesByCustom4( + ListUIState recurringInvoiceListState, FilterRecurringInvoicesByCustom4 action) { + if (recurringInvoiceListState.custom4Filters.contains(action.value)) { + return recurringInvoiceListState + .rebuild((b) => b..custom4Filters.remove(action.value)); + } else { + return recurringInvoiceListState.rebuild((b) => b..custom4Filters.add(action.value)); } } ListUIState _filterRecurringInvoicesByState( - ListUIState recurringInvoiceListState, - FilterRecurringInvoicesByState action) { + ListUIState recurringInvoiceListState, FilterRecurringInvoicesByState action) { if (recurringInvoiceListState.stateFilters.contains(action.state)) { return recurringInvoiceListState .rebuild((b) => b..stateFilters.remove(action.state)); } else { + return recurringInvoiceListState.rebuild((b) => b..stateFilters.add(action.state)); + } +} + +ListUIState _filterRecurringInvoicesByStatus( + ListUIState recurringInvoiceListState, FilterRecurringInvoicesByStatus action) { + if (recurringInvoiceListState.statusFilters.contains(action.status)) { return recurringInvoiceListState - .rebuild((b) => b..stateFilters.add(action.state)); + .rebuild((b) => b..statusFilters.remove(action.status)); + } else { + return recurringInvoiceListState.rebuild((b) => b..statusFilters.add(action.status)); } } @@ -141,24 +257,24 @@ ListUIState _sortRecurringInvoices( } ListUIState _startListMultiselect( - ListUIState productListState, StartRecurringInvoiceMultiselect action) { - return productListState.rebuild((b) => b..selectedIds = ListBuilder()); + ListUIState recurringInvoiceListState, StartRecurringInvoiceMultiselect action) { + return recurringInvoiceListState.rebuild((b) => b..selectedIds = ListBuilder()); } ListUIState _addToListMultiselect( - ListUIState productListState, AddToRecurringInvoiceMultiselect action) { - return productListState.rebuild((b) => b..selectedIds.add(action.entity.id)); + ListUIState recurringInvoiceListState, AddToRecurringInvoiceMultiselect action) { + return recurringInvoiceListState.rebuild((b) => b..selectedIds.add(action.entity.id)); } -ListUIState _removeFromListMultiselect(ListUIState productListState, +ListUIState _removeFromListMultiselect(ListUIState recurringInvoiceListState, RemoveFromRecurringInvoiceMultiselect action) { - return productListState + return recurringInvoiceListState .rebuild((b) => b..selectedIds.remove(action.entity.id)); } ListUIState _clearListMultiselect( - ListUIState productListState, ClearRecurringInvoiceMultiselect action) { - return productListState.rebuild((b) => b..selectedIds = null); + ListUIState recurringInvoiceListState, ClearRecurringInvoiceMultiselect action) { + return recurringInvoiceListState.rebuild((b) => b..selectedIds = null); } final recurringInvoicesReducer = combineReducers([ @@ -168,9 +284,9 @@ final recurringInvoicesReducer = combineReducers([ _addRecurringInvoice), TypedReducer( _setLoadedRecurringInvoices), - TypedReducer( - _setLoadedRecurringInvoice), TypedReducer(_setLoadedCompany), + TypedReducer( + _updateRecurringInvoice), TypedReducer( _archiveRecurringInvoiceSuccess), TypedReducer( @@ -190,8 +306,7 @@ RecurringInvoiceState _archiveRecurringInvoiceSuccess( } RecurringInvoiceState _deleteRecurringInvoiceSuccess( - RecurringInvoiceState recurringInvoiceState, - DeleteRecurringInvoicesSuccess action) { + RecurringInvoiceState recurringInvoiceState, DeleteRecurringInvoicesSuccess action) { return recurringInvoiceState.rebuild((b) { for (final recurringInvoice in action.recurringInvoices) { b.map[recurringInvoice.id] = recurringInvoice; @@ -199,6 +314,12 @@ RecurringInvoiceState _deleteRecurringInvoiceSuccess( }); } +RecurringInvoiceState _emailRecurringInvoiceSuccess( + RecurringInvoiceState recurringInvoiceState, EmailRecurringInvoiceSuccess action) { + return recurringInvoiceState + .rebuild((b) => b..map[action.recurringInvoice.id] = action.recurringInvoice); +} + RecurringInvoiceState _restoreRecurringInvoiceSuccess( RecurringInvoiceState recurringInvoiceState, RestoreRecurringInvoicesSuccess action) { @@ -210,25 +331,18 @@ RecurringInvoiceState _restoreRecurringInvoiceSuccess( } RecurringInvoiceState _addRecurringInvoice( - RecurringInvoiceState recurringInvoiceState, - AddRecurringInvoiceSuccess action) { + RecurringInvoiceState recurringInvoiceState, AddRecurringInvoiceSuccess action) { return recurringInvoiceState.rebuild((b) => b ..map[action.recurringInvoice.id] = action.recurringInvoice ..list.add(action.recurringInvoice.id)); } RecurringInvoiceState _updateRecurringInvoice( - RecurringInvoiceState recurringInvoiceState, - SaveRecurringInvoiceSuccess action) { - return recurringInvoiceState.rebuild( - (b) => b..map[action.recurringInvoice.id] = action.recurringInvoice); -} - -RecurringInvoiceState _setLoadedRecurringInvoice( - RecurringInvoiceState recurringInvoiceState, - LoadRecurringInvoiceSuccess action) { - return recurringInvoiceState.rebuild( - (b) => b..map[action.recurringInvoice.id] = action.recurringInvoice); + RecurringInvoiceState recurringInvoiceState, dynamic action) { + final InvoiceEntity recurringInvoice = action.recurringInvoice; + return recurringInvoiceState.rebuild((b) => b + ..map[action.recurringInvoice.id] = recurringInvoice + .rebuild((b) => b..loadedAt = DateTime.now().millisecondsSinceEpoch)); } RecurringInvoiceState _setLoadedRecurringInvoices(