// Package imports: import 'package:built_collection/built_collection.dart'; import 'package:built_value/built_value.dart'; import 'package:built_value/serializer.dart'; import 'package:collection/collection.dart'; import 'package:diacritic/diacritic.dart'; import 'package:flutter_redux/flutter_redux.dart'; // Project imports: import 'package:invoiceninja_flutter/data/models/models.dart'; import 'package:invoiceninja_flutter/main_app.dart'; import 'package:invoiceninja_flutter/redux/app/app_state.dart'; import 'package:invoiceninja_flutter/redux/design/design_selectors.dart'; import 'package:invoiceninja_flutter/utils/formatting.dart'; import 'package:invoiceninja_flutter/utils/strings.dart'; part 'project_model.g.dart'; abstract class ProjectListResponse implements Built { factory ProjectListResponse([void updates(ProjectListResponseBuilder b)]) = _$ProjectListResponse; ProjectListResponse._(); @override @memoized int get hashCode; BuiltList get data; static Serializer get serializer => _$projectListResponseSerializer; } abstract class ProjectItemResponse implements Built { factory ProjectItemResponse([void updates(ProjectItemResponseBuilder b)]) = _$ProjectItemResponse; ProjectItemResponse._(); @override @memoized int get hashCode; ProjectEntity get data; static Serializer get serializer => _$projectItemResponseSerializer; } class ProjectFields { static const String number = 'number'; static const String name = 'name'; static const String color = 'color'; static const String clientNumber = 'client_number'; static const String clientIdNumber = 'client_id_number'; static const String client = 'client'; static const String taskRate = 'task_rate'; static const String dueDate = 'due_date'; static const String privateNotes = 'private_notes'; static const String publicNotes = 'public_notes'; static const String budgetedHours = 'budgeted_hours'; static const String customValue1 = 'custom1'; static const String customValue2 = 'custom2'; static const String customValue3 = 'custom3'; static const String customValue4 = 'custom4'; static const String updatedAt = 'updated_at'; static const String archivedAt = 'archived_at'; static const String isDeleted = 'is_deleted'; static const String documents = 'documents'; static const String totalHours = 'total_hours'; } abstract class ProjectEntity extends Object with BaseEntity, SelectableEntity, BelongsToClient implements Built { factory ProjectEntity({ String? id, AppState? state, ClientEntity? client, UserEntity? user, }) { return _$ProjectEntity._( id: id ?? BaseEntity.nextId, number: '', isChanged: false, name: '', color: '', clientId: client?.id ?? '', taskRate: 0.0, dueDate: '', privateNotes: '', publicNotes: '', budgetedHours: 0.0, customValue1: '', customValue2: '', customValue3: '', customValue4: '', updatedAt: 0, archivedAt: 0, isDeleted: false, createdUserId: '', createdAt: 0, assignedUserId: user?.id ?? '', totalHours: 0.0, documents: BuiltList(), ); } ProjectEntity._(); @override @memoized int get hashCode; ProjectEntity get clone => rebuild((b) => b ..id = BaseEntity.nextId ..number = '' ..documents.clear() ..isChanged = false ..isDeleted = false); @override EntityType get entityType { return EntityType.project; } String get name; String get color; @override @BuiltValueField(wireName: 'client_id') String get clientId; @BuiltValueField(wireName: 'task_rate') double get taskRate; @BuiltValueField(wireName: 'due_date') String get dueDate; @BuiltValueField(wireName: 'private_notes') String get privateNotes; @BuiltValueField(wireName: 'public_notes') String get publicNotes; @BuiltValueField(wireName: 'budgeted_hours') double get budgetedHours; @BuiltValueField(wireName: 'custom_value1') String get customValue1; @BuiltValueField(wireName: 'custom_value2') String get customValue2; @BuiltValueField(wireName: 'custom_value3') String get customValue3; @BuiltValueField(wireName: 'custom_value4') String get customValue4; String get number; @BuiltValueField(wireName: 'current_hours') double get totalHours; BuiltList get documents; @override List getActions( {UserCompanyEntity? userCompany, ClientEntity? client, bool includeEdit = false, bool includePreview = false, bool multiselect = false}) { final actions = []; if (!multiselect && !isDeleted!) { if (includeEdit && userCompany!.canEditEntity(this)) { actions.add(EntityAction.edit); } if (isActive) { if (userCompany!.canCreate(EntityType.task)) { actions.add(EntityAction.newTask); } if (userCompany.canCreate(EntityType.expense)) { actions.add(EntityAction.newExpense); } } } if (userCompany!.canCreate(EntityType.invoice) && !isDeleted!) { actions.add(EntityAction.invoiceProject); } if (userCompany.canCreate(EntityType.project)) { actions.add(EntityAction.clone); } if (!isDeleted! && multiselect) { actions.add(EntityAction.documents); } if (!isDeleted!) { final store = StoreProvider.of(navigatorKey.currentContext!); if (hasDesignTemplatesForEntityType( store.state.designState.map, entityType)) { actions.add(EntityAction.runTemplate); } } if (actions.isNotEmpty && actions.last != null) { actions.add(null); } return actions..addAll(super.getActions(userCompany: userCompany)); } int compareTo( ProjectEntity project, String sortField, bool sortAscending, BuiltMap userMap, BuiltMap clientMap) { int response = 0; final ProjectEntity projectA = sortAscending ? this : project; final ProjectEntity projectB = sortAscending ? project : this; switch (sortField) { case ProjectFields.name: response = projectA.name.toLowerCase().compareTo(projectB.name.toLowerCase()); break; case ProjectFields.taskRate: response = projectA.taskRate.compareTo(projectB.taskRate); break; case ProjectFields.client: final clientA = clientMap[projectA.clientId] ?? ClientEntity(); final clientB = clientMap[projectB.clientId] ?? ClientEntity(); response = removeDiacritics(clientA.listDisplayName) .toLowerCase() .compareTo(removeDiacritics(clientB.listDisplayName).toLowerCase()); break; case ProjectFields.clientNumber: final clientA = clientMap[projectA.clientId] ?? ClientEntity(); final clientB = clientMap[projectB.clientId] ?? ClientEntity(); response = clientA.number .toLowerCase() .compareTo(clientB.number.toLowerCase()); break; case ProjectFields.clientIdNumber: final clientA = clientMap[projectA.clientId] ?? ClientEntity(); final clientB = clientMap[projectB.clientId] ?? ClientEntity(); response = clientA.idNumber .toLowerCase() .compareTo(clientB.idNumber.toLowerCase()); break; case ProjectFields.dueDate: response = projectA.dueDate.compareTo(projectB.dueDate); break; case ProjectFields.privateNotes: response = projectA.privateNotes.compareTo(projectB.privateNotes); break; case ProjectFields.publicNotes: response = projectA.publicNotes.compareTo(projectB.publicNotes); break; case ProjectFields.budgetedHours: response = projectA.budgetedHours.compareTo(projectB.budgetedHours); break; case ProjectFields.totalHours: response = projectA.totalHours.compareTo(projectB.totalHours); break; case EntityFields.state: final stateA = EntityState.valueOf(projectA.entityState); final stateB = EntityState.valueOf(projectB.entityState); response = stateA.name.toLowerCase().compareTo(stateB.name.toLowerCase()); break; case EntityFields.createdAt: response = projectA.createdAt.compareTo(projectB.createdAt); break; case ProjectFields.archivedAt: response = projectA.archivedAt.compareTo(projectB.archivedAt); break; case ProjectFields.updatedAt: response = projectA.updatedAt.compareTo(projectB.updatedAt); break; case EntityFields.assignedTo: final userA = userMap[projectA.assignedUserId] ?? UserEntity(); final userB = userMap[projectB.assignedUserId] ?? UserEntity(); response = userA.listDisplayName .toLowerCase() .compareTo(userB.listDisplayName.toLowerCase()); break; case EntityFields.createdBy: final userA = userMap[projectA.createdUserId] ?? UserEntity(); final userB = userMap[projectB.createdUserId] ?? UserEntity(); response = userA.listDisplayName .toLowerCase() .compareTo(userB.listDisplayName.toLowerCase()); break; case ProjectFields.documents: response = projectA.documents.length.compareTo(projectB.documents.length); break; case ProjectFields.number: response = compareNatural(projectA.number, projectB.number); break; case ProjectFields.customValue1: response = projectA.customValue1.compareTo(projectB.customValue1); break; case ProjectFields.customValue2: response = projectA.customValue2.compareTo(projectB.customValue2); break; case ProjectFields.customValue3: response = projectA.customValue3.compareTo(projectB.customValue3); break; case ProjectFields.customValue4: response = projectA.customValue4.compareTo(projectB.customValue4); break; default: print('## ERROR: sort by project.$sortField is not implemented'); break; } if (response == 0) { response = project.number.toLowerCase().compareTo(number.toLowerCase()); } return response; } bool matchesName(String filter) => name.toLowerCase().contains(filter.toLowerCase()); @override bool matchesFilter(String? filter) { return matchesStrings( haystacks: [ name, number, publicNotes, privateNotes, customValue1, customValue2, customValue3, customValue4, ], needle: filter, ); } @override String? matchesFilterValue(String? filter) { return matchesStringsValue( haystacks: [ name, number, publicNotes, privateNotes, customValue1, customValue2, customValue3, customValue4, ], needle: filter, ); } @override String get listDisplayName { return name; } @override double? get listDisplayAmount => null; bool get hasClient => clientId.isNotEmpty; @override FormatNumberType get listDisplayAmountType => FormatNumberType.money; // ignore: unused_element static void _initializeBuilder(ProjectEntityBuilder builder) => builder ..color = '' ..totalHours = 0; static Serializer get serializer => _$projectEntitySerializer; }