invoice/lib/ui/reports/client_report.dart

296 lines
11 KiB
Dart

import 'package:built_collection/built_collection.dart';
import 'package:invoiceninja_flutter/constants.dart';
import 'package:invoiceninja_flutter/data/models/client_model.dart';
import 'package:invoiceninja_flutter/data/models/company_model.dart';
import 'package:invoiceninja_flutter/data/models/dashboard_model.dart';
import 'package:invoiceninja_flutter/data/models/static/industry_model.dart';
import 'package:invoiceninja_flutter/data/models/static/size_model.dart';
import 'package:invoiceninja_flutter/redux/reports/reports_state.dart';
import 'package:invoiceninja_flutter/ui/reports/reports_screen.dart';
import 'package:invoiceninja_flutter/utils/dates.dart';
import 'package:invoiceninja_flutter/utils/formatting.dart';
import 'package:memoize/memoize.dart';
class ClientReportFields {
static const String name = 'name';
static const String website = 'website';
static const String privateNotes = 'private_notes';
static const String publicNotes = 'public_notes';
static const String industry = 'industry';
static const String size = 'size';
static const String address1 = 'address1';
static const String address2 = 'address2';
static const String city = 'city';
static const String state = 'state';
static const String postCode = 'postal_code';
static const String phone = 'phone';
static const String country = 'country';
static const String shippingAddress1 = 'shipping_address1';
static const String shippingAddress2 = 'shipping_address2';
static const String shippingCity = 'shipping_city';
static const String shippingState = 'shipping_state';
static const String shippingPostalCode = 'shipping_postal_code';
static const String shippingCountry = 'shipping_country';
static const String customValue1 = 'custom_value1';
static const String customValue2 = 'custom_value2';
static const String customValue3 = 'custom_value3';
static const String customValue4 = 'custom_value4';
static const String createdBy = 'created_by';
static const String assignedTo = 'assigned_to';
static const String balance = 'balance';
static const String creditBalance = 'credit_balance';
static const String paidToDate = 'paid_to_date';
static const String idNumber = 'id_number';
static const String vatNumber = 'vat_number';
static const String createdAt = 'created_at';
static const String updatedAt = 'updated_at';
static const String contactFirstName = 'contact_first_name';
static const String contactLastName = 'contact_last_name';
static const String contactEmail = 'contact_email';
static const String contactPhone = 'contact_phone';
static const String contactCustomValue1 = 'contact_custom_value1';
static const String contactCustomValue2 = 'contact_custom_value2';
static const String contactCustomValue3 = 'contact_custom_value3';
static const String contactCustomValue4 = 'contact_custom_value4';
static const String contactLastLogin = 'contact_last_login';
}
var memoizedClientReport = memo5((
UserCompanyEntity userCompany,
ReportsUIState reportsUIState,
BuiltMap<String, ClientEntity> clientMap,
BuiltMap<String, IndustryEntity> industryMap,
BuiltMap<String, SizeEntity> sizeMap,
) =>
clientReport(userCompany, reportsUIState, clientMap, industryMap, sizeMap));
ReportResult clientReport(
UserCompanyEntity userCompany,
ReportsUIState reportsUIState,
BuiltMap<String, ClientEntity> clientMap,
BuiltMap<String, IndustryEntity> industryMap,
BuiltMap<String, SizeEntity> sizeMap,
) {
final List<List<ReportElement>> data = [];
BuiltList<String> columns;
final reportSettings = userCompany.settings.reportSettings;
final clientReportSettings =
reportSettings != null && reportSettings.containsKey(kReportClient)
? reportSettings[kReportClient]
: ReportSettingsEntity();
if (clientReportSettings.columns.isNotEmpty) {
columns = clientReportSettings.columns;
} else {
columns = BuiltList(<String>[
ClientReportFields.name,
ClientReportFields.idNumber,
]);
}
for (var clientId in clientMap.keys) {
final client = clientMap[clientId];
final contact = client.primaryContact;
if (client.isDeleted) {
continue;
}
bool skip = false;
final List<ReportElement> row = [];
for (var column in columns) {
String value = '';
double amount;
switch (column) {
case ClientReportFields.name:
value = client.name;
break;
case ClientReportFields.website:
value = client.website;
break;
case ClientReportFields.privateNotes:
value = client.privateNotes;
break;
case ClientReportFields.publicNotes:
value = client.publicNotes;
break;
case ClientReportFields.industry:
value = industryMap[client.industryId].listDisplayName;
break;
case ClientReportFields.size:
value = sizeMap[client.sizeId].listDisplayName;
break;
case ClientReportFields.address1:
value = client.address1;
break;
case ClientReportFields.address2:
value = client.address2;
break;
case ClientReportFields.city:
value = client.city;
break;
case ClientReportFields.state:
value = client.state;
break;
case ClientReportFields.postCode:
value = client.postalCode;
break;
case ClientReportFields.phone:
value = client.phone;
break;
case ClientReportFields.idNumber:
value = client.idNumber;
break;
case ClientReportFields.vatNumber:
value = client.vatNumber;
break;
case ClientReportFields.createdAt:
value = convertTimestampToDateString(client.createdAt);
break;
case ClientReportFields.assignedTo:
value = userCompany
.company.userMap[client.assignedUserId]?.listDisplayName ??
'';
break;
case ClientReportFields.createdBy:
value = userCompany
.company.userMap[client.createdUserId]?.listDisplayName ??
'';
break;
case ClientReportFields.updatedAt:
value = convertTimestampToDateString(client.updatedAt);
break;
case ClientReportFields.contactLastLogin:
value = convertTimestampToDateString(contact.lastLogin);
break;
case ClientReportFields.balance:
amount = client.balance;
break;
case ClientReportFields.paidToDate:
amount = client.paidToDate;
break;
}
if (reportsUIState.filters.containsKey(column)) {
final filter = reportsUIState.filters[column];
if (filter.isNotEmpty) {
if (getReportColumnType(column) == ReportColumnType.number) {
final String range = filter.replaceAll(',', '-') + '-';
final List<String> parts = range.split('-');
final min = parseDouble(parts[0]);
final max = parseDouble(parts[1]);
if (amount < min || (max > 0 && amount > max)) {
skip = true;
}
} else if (getReportColumnType(column) == ReportColumnType.dateTime) {
final startDate = calculateStartDate(
dateRange: DateRange.valueOf(filter),
company: userCompany.company,
customStartDate: reportsUIState.customStartDate,
customEndDate: reportsUIState.customEndDate,
);
final endDate = calculateEndDate(
dateRange: DateRange.valueOf(filter),
company: userCompany.company,
customStartDate: reportsUIState.customStartDate,
customEndDate: reportsUIState.customEndDate,
);
if (reportsUIState.customStartDate.isNotEmpty &&
reportsUIState.customEndDate.isNotEmpty) {
if (!(startDate.compareTo(value) <= 0 &&
endDate.compareTo(value) >= 0)) {
skip = true;
}
} else if (reportsUIState.customStartDate.isNotEmpty) {
if (!(startDate.compareTo(value) <= 0)) {
skip = true;
}
} else if (reportsUIState.customEndDate.isNotEmpty) {
if (!(endDate.compareTo(value) >= 0)) {
skip = true;
}
}
} else if (!value.toLowerCase().contains(filter.toLowerCase())) {
skip = true;
}
}
}
if (amount != null) {
row.add(client.getReportAmount(value: amount));
} else {
row.add(client.getReportValue(value: value));
}
}
if (!skip) {
data.add(row);
}
}
data.sort((rowA, rowB) {
if (rowA.length <= clientReportSettings.sortIndex ||
rowB.length <= clientReportSettings.sortIndex) {
return 0;
}
final valueA = rowA[clientReportSettings.sortIndex].sortString();
final valueB = rowB[clientReportSettings.sortIndex].sortString();
if (clientReportSettings.sortAscending) {
return valueA.compareTo(valueB);
} else {
return valueB.compareTo(valueA);
}
});
return ReportResult(
allColumns: [
ClientReportFields.name,
ClientReportFields.website,
ClientReportFields.privateNotes,
ClientReportFields.publicNotes,
ClientReportFields.industry,
ClientReportFields.size,
ClientReportFields.address1,
ClientReportFields.address2,
ClientReportFields.city,
ClientReportFields.state,
ClientReportFields.postCode,
ClientReportFields.phone,
ClientReportFields.country,
ClientReportFields.shippingAddress1,
ClientReportFields.shippingAddress2,
ClientReportFields.shippingCity,
ClientReportFields.shippingState,
ClientReportFields.shippingPostalCode,
ClientReportFields.shippingCountry,
ClientReportFields.customValue1,
ClientReportFields.customValue2,
ClientReportFields.customValue3,
ClientReportFields.customValue4,
ClientReportFields.createdBy,
ClientReportFields.assignedTo,
ClientReportFields.balance,
ClientReportFields.creditBalance,
ClientReportFields.paidToDate,
ClientReportFields.idNumber,
ClientReportFields.vatNumber,
ClientReportFields.createdAt,
ClientReportFields.updatedAt,
ClientReportFields.contactFirstName,
ClientReportFields.contactLastName,
ClientReportFields.contactEmail,
ClientReportFields.contactPhone,
ClientReportFields.contactCustomValue1,
ClientReportFields.contactCustomValue2,
ClientReportFields.contactCustomValue3,
ClientReportFields.contactCustomValue4,
ClientReportFields.contactLastLogin,
],
columns: columns.toList(),
data: data,
);
}