invoice/lib/redux/invoice/invoice_actions.dart

708 lines
18 KiB
Dart

import 'dart:async';
import 'package:built_collection/built_collection.dart';
import 'package:flutter/material.dart';
import 'package:flutter/widgets.dart';
import 'package:flutter_redux/flutter_redux.dart';
import 'package:http/http.dart';
import 'package:invoiceninja_flutter/data/models/models.dart';
import 'package:invoiceninja_flutter/redux/app/app_actions.dart';
import 'package:invoiceninja_flutter/redux/app/app_state.dart';
import 'package:invoiceninja_flutter/redux/design/design_selectors.dart';
import 'package:invoiceninja_flutter/ui/app/entities/entity_actions_dialog.dart';
import 'package:invoiceninja_flutter/utils/completers.dart';
import 'package:invoiceninja_flutter/utils/dialogs.dart';
import 'package:invoiceninja_flutter/utils/localization.dart';
import 'package:url_launcher/url_launcher.dart';
class ViewInvoiceList implements StopLoading {
ViewInvoiceList({this.force = false});
final bool force;
}
class ViewInvoice implements PersistUI, PersistPrefs {
ViewInvoice({this.invoiceId, this.force = false});
final String invoiceId;
final bool force;
}
class EditInvoice implements PersistUI, PersistPrefs {
EditInvoice({
this.invoice,
this.completer,
this.invoiceItemIndex,
this.force = false,
});
final InvoiceEntity invoice;
final int invoiceItemIndex;
final Completer completer;
final bool force;
}
class ShowEmailInvoice {
ShowEmailInvoice({this.invoice, this.context, this.completer});
final InvoiceEntity invoice;
final BuildContext context;
final Completer completer;
}
class ShowPdfInvoice {
ShowPdfInvoice({this.invoice, this.context, this.activityId});
final InvoiceEntity invoice;
final BuildContext context;
final String activityId;
}
class EditInvoiceItem implements PersistUI {
EditInvoiceItem([this.invoiceItemIndex]);
final int invoiceItemIndex;
}
class UpdateInvoice implements PersistUI {
UpdateInvoice(this.invoice);
final InvoiceEntity invoice;
}
class UpdateInvoiceClient implements PersistUI {
UpdateInvoiceClient({this.client});
final ClientEntity client;
}
class LoadInvoice {
LoadInvoice({this.completer, this.invoiceId});
final Completer completer;
final String invoiceId;
}
class LoadInvoices {
LoadInvoices({this.completer});
final Completer completer;
}
class LoadInvoiceRequest implements StartLoading {}
class LoadInvoiceFailure implements StopLoading {
LoadInvoiceFailure(this.error);
final dynamic error;
@override
String toString() {
return 'LoadInvoiceFailure{error: $error}';
}
}
class LoadInvoiceSuccess implements StopLoading, PersistData {
LoadInvoiceSuccess(this.invoice);
final InvoiceEntity invoice;
@override
String toString() {
return 'LoadInvoiceSuccess{invoice: $invoice}';
}
}
class LoadInvoicesRequest implements StartLoading {}
class LoadInvoicesFailure implements StopLoading {
LoadInvoicesFailure(this.error);
final dynamic error;
@override
String toString() {
return 'LoadInvoicesFailure{error: $error}';
}
}
class LoadInvoicesSuccess implements StopLoading {
LoadInvoicesSuccess(this.invoices);
final BuiltList<InvoiceEntity> invoices;
@override
String toString() {
return 'LoadInvoicesSuccess{invoices: $invoices}';
}
}
class AddInvoiceContact implements PersistUI {
AddInvoiceContact({this.contact, this.invitation});
final ContactEntity contact;
final InvitationEntity invitation;
}
class RemoveInvoiceContact implements PersistUI {
RemoveInvoiceContact({this.invitation});
final InvitationEntity invitation;
}
class AddInvoiceItem implements PersistUI {
AddInvoiceItem({this.invoiceItem});
final InvoiceItemEntity invoiceItem;
}
class MoveInvoiceItem implements PersistUI {
MoveInvoiceItem({
this.oldIndex,
this.newIndex,
});
final int oldIndex;
final int newIndex;
}
class AddInvoiceItems implements PersistUI {
AddInvoiceItems(this.lineItems);
final List<InvoiceItemEntity> lineItems;
}
class UpdateInvoiceItem implements PersistUI {
UpdateInvoiceItem({this.index, this.invoiceItem});
final int index;
final InvoiceItemEntity invoiceItem;
}
class DeleteInvoiceItem implements PersistUI {
DeleteInvoiceItem(this.index);
final int index;
}
class SaveInvoiceRequest implements StartSaving {
SaveInvoiceRequest({
this.completer,
this.invoice,
this.action,
});
final Completer completer;
final InvoiceEntity invoice;
final EntityAction action;
}
class SaveInvoiceSuccess implements StopSaving, PersistUI {
SaveInvoiceSuccess(this.invoice);
final InvoiceEntity invoice;
}
class AddInvoiceSuccess implements StopSaving, PersistUI {
AddInvoiceSuccess(this.invoice);
final InvoiceEntity invoice;
}
class SaveInvoiceFailure implements StopSaving {
SaveInvoiceFailure(this.error);
final Object error;
}
class EmailInvoiceRequest implements StartSaving {
EmailInvoiceRequest(
{this.completer, this.invoiceId, this.template, this.subject, this.body});
final Completer completer;
final String invoiceId;
final EmailTemplate template;
final String subject;
final String body;
}
class EmailInvoiceSuccess implements StopSaving, PersistData {
EmailInvoiceSuccess({@required this.invoice});
final InvoiceEntity invoice;
}
class EmailInvoiceFailure implements StopSaving {
EmailInvoiceFailure(this.error);
final dynamic error;
}
class MarkInvoicesSentRequest implements StartSaving {
MarkInvoicesSentRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class MarkInvoicesSentSuccess implements StopSaving, PersistData {
MarkInvoicesSentSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class MarkInvoicesSentFailure implements StopSaving {
MarkInvoicesSentFailure(this.error);
final dynamic error;
}
class BulkEmailInvoicesRequest implements StartSaving {
BulkEmailInvoicesRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class BulkEmailInvoicesSuccess implements StopSaving, PersistData {
BulkEmailInvoicesSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class BulkEmailInvoicesFailure implements StopSaving {
BulkEmailInvoicesFailure(this.error);
final dynamic error;
}
class MarkInvoicesPaidRequest implements StartSaving {
MarkInvoicesPaidRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class MarkInvoicesPaidSuccess implements StopSaving {
MarkInvoicesPaidSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class MarkInvoicesPaidFailure implements StopSaving {
MarkInvoicesPaidFailure(this.error);
final dynamic error;
}
class ReverseInvoicesRequest implements StartSaving {
ReverseInvoicesRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class ReverseInvoicesSuccess implements StopSaving {
ReverseInvoicesSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class ReverseInvoicesFailure implements StopSaving {
ReverseInvoicesFailure(this.error);
final Object error;
}
class CancelInvoicesRequest implements StartSaving {
CancelInvoicesRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class CancelInvoicesSuccess implements StopSaving {
CancelInvoicesSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class CancelInvoicesFailure implements StopSaving {
CancelInvoicesFailure(this.error);
final Object error;
}
class ArchiveInvoicesRequest implements StartSaving {
ArchiveInvoicesRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class ArchiveInvoicesSuccess implements StopSaving, PersistData {
ArchiveInvoicesSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class ArchiveInvoicesFailure implements StopSaving {
ArchiveInvoicesFailure(this.invoices);
final List<InvoiceEntity> invoices;
}
class DeleteInvoicesRequest implements StartSaving {
DeleteInvoicesRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class DeleteInvoicesSuccess implements StopSaving, PersistData {
DeleteInvoicesSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class DeleteInvoicesFailure implements StopSaving {
DeleteInvoicesFailure(this.invoices);
final List<InvoiceEntity> invoices;
}
class DownloadInvoicesRequest implements StartSaving {
DownloadInvoicesRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class DownloadInvoicesSuccess implements StopSaving {}
class DownloadInvoicesFailure implements StopSaving {
DownloadInvoicesFailure(this.error);
final Object error;
}
class RestoreInvoicesRequest implements StartSaving {
RestoreInvoicesRequest(this.completer, this.invoiceIds);
final Completer completer;
final List<String> invoiceIds;
}
class RestoreInvoicesSuccess implements StopSaving, PersistData {
RestoreInvoicesSuccess(this.invoices);
final List<InvoiceEntity> invoices;
}
class RestoreInvoicesFailure implements StopSaving {
RestoreInvoicesFailure(this.invoices);
final List<InvoiceEntity> invoices;
}
class FilterInvoices implements PersistUI {
FilterInvoices(this.filter);
final String filter;
}
class SortInvoices implements PersistUI, PersistPrefs {
SortInvoices(this.field);
final String field;
}
class FilterInvoicesByState implements PersistUI {
FilterInvoicesByState(this.state);
final EntityState state;
}
class FilterInvoicesByStatus implements PersistUI {
FilterInvoicesByStatus(this.status);
final EntityStatus status;
}
class FilterInvoiceDropdown {
FilterInvoiceDropdown(this.filter);
final String filter;
}
class FilterInvoicesByCustom1 implements PersistUI {
FilterInvoicesByCustom1(this.value);
final String value;
}
class FilterInvoicesByCustom2 implements PersistUI {
FilterInvoicesByCustom2(this.value);
final String value;
}
class FilterInvoicesByCustom3 implements PersistUI {
FilterInvoicesByCustom3(this.value);
final String value;
}
class FilterInvoicesByCustom4 implements PersistUI {
FilterInvoicesByCustom4(this.value);
final String value;
}
class StartInvoiceMultiselect {}
class AddToInvoiceMultiselect {
AddToInvoiceMultiselect({@required this.entity});
final BaseEntity entity;
}
class RemoveFromInvoiceMultiselect {
RemoveFromInvoiceMultiselect({@required this.entity});
final BaseEntity entity;
}
class ClearInvoiceMultiselect {}
class SaveInvoiceDocumentRequest implements StartSaving {
SaveInvoiceDocumentRequest({
@required this.completer,
@required this.multipartFile,
@required this.invoice,
});
final Completer completer;
final MultipartFile multipartFile;
final InvoiceEntity invoice;
}
class SaveInvoiceDocumentSuccess implements StopSaving, PersistData, PersistUI {
SaveInvoiceDocumentSuccess(this.document);
final DocumentEntity document;
}
class SaveInvoiceDocumentFailure implements StopSaving {
SaveInvoiceDocumentFailure(this.error);
final Object error;
}
class UpdateInvoiceTab implements PersistUI {
UpdateInvoiceTab({this.tabIndex});
final int tabIndex;
}
void handleInvoiceAction(BuildContext context, List<BaseEntity> invoices,
EntityAction action) async {
if (invoices.isEmpty) {
return;
}
final store = StoreProvider.of<AppState>(context);
final state = store.state;
final localization = AppLocalization.of(context);
final invoice = invoices.first as InvoiceEntity;
final invoiceIds = invoices.map((invoice) => invoice.id).toList();
switch (action) {
case EntityAction.edit:
editEntity(context: context, entity: invoice);
break;
case EntityAction.view:
store.dispatch(UpdateUserPreferences(isPreviewEnabled: true));
viewEntity(entity: invoice);
break;
case EntityAction.viewPdf:
store.dispatch(ShowPdfInvoice(invoice: invoice, context: context));
break;
case EntityAction.clientPortal:
if (await canLaunch(invoice.invitationSilentLink)) {
await launch(invoice.invitationSilentLink,
forceSafariVC: false, forceWebView: false);
}
break;
case EntityAction.markSent:
store.dispatch(MarkInvoicesSentRequest(
snackBarCompleter<Null>(
context,
invoiceIds.length == 1
? localization.markedInvoiceAsSent
: localization.markedInvoicesAsSent),
invoiceIds));
break;
case EntityAction.reverse:
store.dispatch(ReverseInvoicesRequest(
snackBarCompleter<Null>(
context,
invoiceIds.length == 1
? localization.reversedInvoice
: localization.reversedInvoices),
invoiceIds));
break;
case EntityAction.cancel:
store.dispatch(CancelInvoicesRequest(
snackBarCompleter<Null>(
context,
invoiceIds.length == 1
? localization.cancelledInvoice
: localization.cancelledInvoices),
invoiceIds));
break;
case EntityAction.markPaid:
store.dispatch(MarkInvoicesPaidRequest(
snackBarCompleter<Null>(
context,
invoiceIds.length == 1
? localization.markedInvoiceAsPaid
: localization.markedInvoicesAsPaid),
invoiceIds));
break;
case EntityAction.emailInvoice:
case EntityAction.bulkEmailInvoice:
bool emailValid = true;
invoiceIds.forEach((element) {
final client = state.clientState.get(invoice.clientId);
if (!client.hasEmailAddress) {
emailValid = false;
}
});
if (!emailValid) {
showMessageDialog(
context: context,
message: localization.clientEmailNotSet,
secondaryActions: [
TextButton(
onPressed: () {
Navigator.of(context).pop();
editEntity(
context: context,
entity: state.clientState.get(invoice.clientId));
},
child: Text(localization.editClient.toUpperCase()))
]);
return;
}
if (action == EntityAction.emailInvoice) {
store.dispatch(ShowEmailInvoice(
completer:
snackBarCompleter<Null>(context, localization.emailedInvoice),
invoice: invoice,
context: context));
} else {
store.dispatch(BulkEmailInvoicesRequest(
snackBarCompleter<Null>(
context,
invoiceIds.length == 1
? localization.emailedInvoice
: localization.emailedInvoices),
invoiceIds));
}
break;
case EntityAction.cloneToOther:
cloneToDialog(context: context, invoice: invoice);
break;
case EntityAction.cloneToInvoice:
createEntity(context: context, entity: invoice.clone);
break;
case EntityAction.cloneToQuote:
final designId = getDesignIdForClientByEntity(
state: state,
clientId: invoice.clientId,
entityType: EntityType.quote);
createEntity(
context: context,
entity: invoice.clone.rebuild((b) => b
..entityType = EntityType.quote
..designId = designId));
break;
case EntityAction.cloneToCredit:
final designId = getDesignIdForClientByEntity(
state: state,
clientId: invoice.clientId,
entityType: EntityType.credit);
createEntity(
context: context,
entity: invoice.clone.rebuild((b) => b
..entityType = EntityType.credit
..designId = designId));
break;
case EntityAction.cloneToRecurring:
createEntity(
context: context,
entity: invoice.clone
.rebuild((b) => b..entityType = EntityType.recurringInvoice));
break;
case EntityAction.newPayment:
createEntity(
context: context,
entity: PaymentEntity(state: state).rebuild((b) => b
..isForInvoice = true
..clientId = invoice.clientId
..invoices.addAll(invoices
.where((invoice) => !(invoice as InvoiceEntity).isPaid)
.map((invoice) => PaymentableEntity.fromInvoice(invoice))
.toList())),
filterEntity: state.clientState.map[invoice.clientId],
);
break;
case EntityAction.download:
store.dispatch(DownloadInvoicesRequest(
snackBarCompleter<Null>(context, localization.exportedData),
invoiceIds));
break;
case EntityAction.restore:
final message = invoiceIds.length > 1
? localization.restoredInvoices
.replaceFirst(':value', invoiceIds.length.toString())
: localization.restoredInvoice;
store.dispatch(RestoreInvoicesRequest(
snackBarCompleter<Null>(context, message), invoiceIds));
break;
case EntityAction.archive:
final message = invoiceIds.length > 1
? localization.archivedInvoices
.replaceFirst(':value', invoiceIds.length.toString())
: localization.archivedInvoice;
store.dispatch(ArchiveInvoicesRequest(
snackBarCompleter<Null>(context, message), invoiceIds));
break;
case EntityAction.delete:
final message = invoiceIds.length > 1
? localization.deletedInvoices
.replaceFirst(':value', invoiceIds.length.toString())
: localization.deletedInvoice;
store.dispatch(DeleteInvoicesRequest(
snackBarCompleter<Null>(context, message), invoiceIds));
break;
case EntityAction.toggleMultiselect:
if (!store.state.invoiceListState.isInMultiselect()) {
store.dispatch(StartInvoiceMultiselect());
}
for (final invoice in invoices) {
if (!store.state.invoiceListState.isSelected(invoice.id)) {
store.dispatch(AddToInvoiceMultiselect(entity: invoice));
} else {
store.dispatch(RemoveFromInvoiceMultiselect(entity: invoice));
}
}
break;
case EntityAction.more:
showEntityActionsDialog(
entities: [invoice],
);
break;
}
}