Added Streams to TourRepository

This commit is contained in:
Dennis Nemec
2026-01-03 01:29:21 +01:00
parent edb8676f5a
commit 9111dc92db
43 changed files with 1232 additions and 931 deletions

68
ios/Podfile.lock Normal file
View File

@ -0,0 +1,68 @@
PODS:
- app_links (6.4.1):
- Flutter
- Flutter (1.0.0)
- geolocator_apple (1.2.0):
- Flutter
- FlutterMacOS
- image_picker_ios (0.0.1):
- Flutter
- mobile_scanner (7.0.0):
- Flutter
- FlutterMacOS
- package_info_plus (0.4.5):
- Flutter
- path_provider_foundation (0.0.1):
- Flutter
- FlutterMacOS
- shared_preferences_foundation (0.0.1):
- Flutter
- FlutterMacOS
- url_launcher_ios (0.0.1):
- Flutter
DEPENDENCIES:
- app_links (from `.symlinks/plugins/app_links/ios`)
- Flutter (from `Flutter`)
- geolocator_apple (from `.symlinks/plugins/geolocator_apple/darwin`)
- image_picker_ios (from `.symlinks/plugins/image_picker_ios/ios`)
- mobile_scanner (from `.symlinks/plugins/mobile_scanner/darwin`)
- package_info_plus (from `.symlinks/plugins/package_info_plus/ios`)
- path_provider_foundation (from `.symlinks/plugins/path_provider_foundation/darwin`)
- shared_preferences_foundation (from `.symlinks/plugins/shared_preferences_foundation/darwin`)
- url_launcher_ios (from `.symlinks/plugins/url_launcher_ios/ios`)
EXTERNAL SOURCES:
app_links:
:path: ".symlinks/plugins/app_links/ios"
Flutter:
:path: Flutter
geolocator_apple:
:path: ".symlinks/plugins/geolocator_apple/darwin"
image_picker_ios:
:path: ".symlinks/plugins/image_picker_ios/ios"
mobile_scanner:
:path: ".symlinks/plugins/mobile_scanner/darwin"
package_info_plus:
:path: ".symlinks/plugins/package_info_plus/ios"
path_provider_foundation:
:path: ".symlinks/plugins/path_provider_foundation/darwin"
shared_preferences_foundation:
:path: ".symlinks/plugins/shared_preferences_foundation/darwin"
url_launcher_ios:
:path: ".symlinks/plugins/url_launcher_ios/ios"
SPEC CHECKSUMS:
app_links: 3dbc685f76b1693c66a6d9dd1e9ab6f73d97dc0a
Flutter: cabc95a1d2626b1b06e7179b784ebcf0c0cde467
geolocator_apple: ab36aa0e8b7d7a2d7639b3b4e48308394e8cef5e
image_picker_ios: 7fe1ff8e34c1790d6fff70a32484959f563a928a
mobile_scanner: 9157936403f5a0644ca3779a38ff8404c5434a93
package_info_plus: af8e2ca6888548050f16fa2f1938db7b5a5df499
path_provider_foundation: 080d55be775b7414fd5a5ef3ac137b97b097e564
shared_preferences_foundation: 7036424c3d8ec98dfe75ff1667cb0cd531ec82bb
url_launcher_ios: 694010445543906933d732453a59da0a173ae33d
PODFILE CHECKSUM: 3c63482e143d1b91d2d2560aee9fb04ecc74ac7e
COCOAPODS: 1.16.2

View File

@ -1,3 +1,5 @@
import 'package:hl_lieferservice/dto/image_note_response.dart';
import 'article.dart'; import 'article.dart';
import 'contact_person.dart'; import 'contact_person.dart';
import 'customer.dart'; import 'customer.dart';
@ -73,7 +75,7 @@ class DeliveryDTO {
DiscountDTO? discount; DiscountDTO? discount;
PaymentMethodDTO payment; PaymentMethodDTO payment;
List<NoteDTO> notes; List<NoteDTO> notes;
List<ImageDTO> images; List<ImageNoteDTO> images;
List<DeliveryOptionDTO> options; List<DeliveryOptionDTO> options;
factory DeliveryDTO.fromJson(Map<String, dynamic> json) => factory DeliveryDTO.fromJson(Map<String, dynamic> json) =>

View File

@ -45,7 +45,7 @@ DeliveryDTO _$DeliveryDTOFromJson(Map<String, dynamic> json) => DeliveryDTO(
totalGrossValue: json['total_gross_value'] as String, totalGrossValue: json['total_gross_value'] as String,
images: images:
(json['images'] as List<dynamic>) (json['images'] as List<dynamic>)
.map((e) => ImageDTO.fromJson(e as Map<String, dynamic>)) .map((e) => ImageNoteDTO.fromJson(e as Map<String, dynamic>))
.toList(), .toList(),
customer: CustomerDTO.fromJson(json['customer'] as Map<String, dynamic>), customer: CustomerDTO.fromJson(json['customer'] as Map<String, dynamic>),
finishedTime: json['finished_time'] as String, finishedTime: json['finished_time'] as String,

View File

@ -0,0 +1,18 @@
import 'note.dart';
import 'package:json_annotation/json_annotation.dart';
part 'image_note_response.g.dart';
@JsonSerializable(fieldRename: FieldRename.snake)
class ImageNoteDTO {
final String url;
final String oid;
final String name;
ImageNoteDTO({required this.url, required this.oid, required this.name});
factory ImageNoteDTO.fromJson(Map<String, dynamic> json) =>
_$ImageNoteDTOFromJson(json);
Map<dynamic, dynamic> toJson() => _$ImageNoteDTOToJson(this);
}

View File

@ -0,0 +1,20 @@
// GENERATED CODE - DO NOT MODIFY BY HAND
part of 'image_note_response.dart';
// **************************************************************************
// JsonSerializableGenerator
// **************************************************************************
ImageNoteDTO _$ImageNoteDTOFromJson(Map<String, dynamic> json) => ImageNoteDTO(
url: json['url'] as String,
oid: json['oid'] as String,
name: json['name'] as String,
);
Map<String, dynamic> _$ImageNoteDTOToJson(ImageNoteDTO instance) =>
<String, dynamic>{
'url': instance.url,
'oid': instance.oid,
'name': instance.name,
};

View File

@ -1,3 +1,5 @@
import 'package:hl_lieferservice/dto/image_note_response.dart';
import 'note.dart'; import 'note.dart';
import 'package:json_annotation/json_annotation.dart'; import 'package:json_annotation/json_annotation.dart';
@ -6,9 +8,10 @@ part 'note_get_response.g.dart';
@JsonSerializable(fieldRename: FieldRename.snake) @JsonSerializable(fieldRename: FieldRename.snake)
class NoteGetResponseDTO { class NoteGetResponseDTO {
NoteGetResponseDTO( NoteGetResponseDTO(
{required this.notes, required this.succeeded, required this.message}); {required this.notes, required this.succeeded, required this.message, required this.images});
final List<NoteDTO> notes; final List<NoteDTO> notes;
final List<ImageNoteDTO> images;
final bool succeeded; final bool succeeded;
final String message; final String message;

View File

@ -14,11 +14,16 @@ NoteGetResponseDTO _$NoteGetResponseDTOFromJson(Map<String, dynamic> json) =>
.toList(), .toList(),
succeeded: json['succeeded'] as bool, succeeded: json['succeeded'] as bool,
message: json['message'] as String, message: json['message'] as String,
images:
(json['images'] as List<dynamic>)
.map((e) => ImageNoteDTO.fromJson(e as Map<String, dynamic>))
.toList(),
); );
Map<String, dynamic> _$NoteGetResponseDTOToJson(NoteGetResponseDTO instance) => Map<String, dynamic> _$NoteGetResponseDTOToJson(NoteGetResponseDTO instance) =>
<String, dynamic>{ <String, dynamic>{
'notes': instance.notes, 'notes': instance.notes,
'images': instance.images,
'succeeded': instance.succeeded, 'succeeded': instance.succeeded,
'message': instance.message, 'message': instance.message,
}; };

View File

@ -56,10 +56,14 @@ class _CarManagementPageState extends State<CarManagementPage> {
return Scaffold( return Scaffold(
body: BlocConsumer<CarsBloc, CarsState>( body: BlocConsumer<CarsBloc, CarsState>(
listener: (context, state) { listener: (context, state) {
if (state is CarsLoaded && context.read<TourBloc>().state is TourLoaded) { if (state is CarsLoaded &&
var tour = (context.read<TourBloc>().state as TourLoaded).tour.copyWith(); context.read<TourBloc>().state is TourLoaded) {
var tour =
(context.read<TourBloc>().state as TourLoaded).tour.copyWith();
tour.driver.cars = state.cars; tour.driver.cars = state.cars;
context.read<TourBloc>().add(UpdateTour(tour: tour)); context.read<TourBloc>().add(
TourUpdated(tour: tour, payments: tour.paymentMethods),
);
} }
}, },
builder: (context, state) { builder: (context, state) {

View File

@ -1,353 +0,0 @@
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/dto/discount_add_response.dart';
import 'package:hl_lieferservice/dto/discount_update_response.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_event.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_state.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_event.dart';
import 'package:hl_lieferservice/feature/delivery/detail/repository/delivery_repository.dart';
import 'package:hl_lieferservice/feature/delivery/detail/repository/note_repository.dart';
import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart';
import 'package:hl_lieferservice/widget/operations/bloc/operation_event.dart';
import '../../../../dto/discount_remove_response.dart';
import '../../../../model/article.dart';
import '../../../../model/delivery.dart' as model;
class DeliveryBloc extends Bloc<DeliveryEvent, DeliveryState> {
OperationBloc opBloc;
NoteBloc noteBloc;
DeliveryRepository repository;
NoteRepository noteRepository;
DeliveryBloc({
required this.opBloc,
required this.repository,
required this.noteRepository,
required this.noteBloc
}) : super(DeliveryInitial()) {
on<UnscanArticleEvent>(_unscan);
on<ResetScanAmountEvent>(_resetAmount);
on<LoadDeliveryEvent>(_load);
on<AddDiscountEvent>(_addDiscount);
on<RemoveDiscountEvent>(_removeDiscount);
on<UpdateDiscountEvent>(_updateDiscount);
on<UpdateDeliveryOptionEvent>(_updateDeliveryOptions);
on<UpdateSelectedPaymentMethodEvent>(_updatePayment);
on<FinishDeliveryEvent>(_finishDelivery);
}
void _finishDelivery(FinishDeliveryEvent event,
Emitter<DeliveryState> emit,) async {
final currentState = state;
opBloc.add(LoadOperation());
if (currentState is DeliveryLoaded) {
try {
model.Delivery newDelivery = event.delivery.copyWith();
newDelivery.state = model.DeliveryState.finished;
for (final option in event.delivery.options) {
debugPrint("VALUE=${option.value};KEY=${option.key}");
}
await repository.updateDelivery(newDelivery);
await noteRepository.addNamedImage(
event.delivery.id,
event.driverSignature,
"delivery_${event.delivery.id}_signature_driver.jpg",
);
await noteRepository.addNamedImage(
event.delivery.id,
event.customerSignature,
"delivery_${event.delivery.id}_signature_customer.jpg",
);
emit(DeliveryFinished(delivery: newDelivery));
opBloc.add(FinishOperation());
} catch (e, st) {
opBloc.add(FailOperation(message: "Failed to update delivery"));
debugPrint(st.toString());
}
}
}
void _updatePayment(UpdateSelectedPaymentMethodEvent event,
Emitter<DeliveryState> emit,) {
final currentState = state;
if (currentState is DeliveryLoaded) {
emit(
DeliveryLoaded(
delivery: currentState.delivery.copyWith(payment: event.payment),
),
);
}
}
void _updateDeliveryOptions(UpdateDeliveryOptionEvent event,
Emitter<DeliveryState> emit,) {
final currentState = state;
if (currentState is DeliveryLoaded) {
List<model.DeliveryOption> options =
currentState.delivery.options.map((option) {
if (option.key == event.key) {
if (option.numerical) {
return option.copyWith(value: event.value);
} else {
return option.copyWith(value: event.value == true ? "1" : "0");
}
}
return option;
}).toList();
emit(
DeliveryLoaded(
delivery: currentState.delivery.copyWith(options: options),
),
);
}
}
void _updateDiscount(UpdateDiscountEvent event,
Emitter<DeliveryState> emit,) async {
opBloc.add(LoadOperation());
try {
final currentState = state;
if (currentState is DeliveryLoaded) {
DiscountUpdateResponseDTO response = await repository.updateDiscount(
event.deliveryId,
event.reason,
event.value,
);
model.Delivery delivery = currentState.delivery;
if (response.values?.receipt != null) {
delivery.totalNetValue = response.values!.receipt.net;
delivery.totalGrossValue = response.values!.receipt.gross;
}
String discountArticleNumber = delivery.discount!.article.articleNumber;
delivery.discount = model.Discount(
article:
response.values?.article != null
? Article.fromDTO(response.values!.article)
: delivery.discount!.article,
note:
response.values?.note != null
? response.values!.note.noteDescription
: delivery.discount!.note,
noteId:
response.values?.note != null
? response.values!.note.rowId
: delivery.discount!.noteId,
);
delivery.articles = [
...delivery.articles.where(
(article) => article.articleNumber != discountArticleNumber,
),
delivery.discount!.article,
];
emit(currentState.copyWith(delivery));
opBloc.add(FinishOperation());
}
} catch (e, st) {
debugPrint(
"Fehler beim Hinzufügen eins Discounts zur Lieferung: ${event
.deliveryId}:",
);
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Hinzufügen des Discounts: $e"),
);
}
}
void _removeDiscount(RemoveDiscountEvent event,
Emitter<DeliveryState> emit,) async {
opBloc.add(LoadOperation());
try {
final currentState = state;
if (currentState is DeliveryLoaded) {
model.Delivery delivery = currentState.delivery;
DiscountRemoveResponseDTO response = await repository.removeDiscount(
event.deliveryId,
);
delivery.articles =
delivery.articles
.where(
(article) =>
article.internalId !=
delivery.discount?.article.internalId,
)
.toList();
delivery.discount = null;
delivery.totalGrossValue = response.receipt.gross;
delivery.totalNetValue = response.receipt.net;
emit(currentState.copyWith(delivery));
opBloc.add(FinishOperation());
}
} catch (e, st) {
debugPrint(
"Fehler beim Löschen des Discounts der Lieferung: ${event.deliveryId}:",
);
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Löschen des Discounts: $e"),
);
}
}
void _addDiscount(AddDiscountEvent event, Emitter<DeliveryState> emit) async {
opBloc.add(LoadOperation());
try {
final currentState = state;
if (currentState is DeliveryLoaded) {
DiscountAddResponseDTO response = await repository.addDiscount(
event.deliveryId,
event.reason,
event.value,
);
model.Delivery delivery = currentState.delivery;
delivery.totalNetValue = response.values.receipt.net;
delivery.totalGrossValue = response.values.receipt.gross;
delivery.discount = model.Discount(
article: Article.fromDTO(response.values.article),
note: response.values.note.noteDescription,
noteId: response.values.note.rowId,
);
noteBloc.add(AddNoteOffline(note: response.values.note.noteDescription,
deliveryId: delivery.id,
noteId: response.values.note.rowId));
delivery.articles = [...delivery.articles, delivery.discount!.article];
emit(currentState.copyWith(delivery));
opBloc.add(FinishOperation());
}
} catch (e, st) {
debugPrint(
"Fehler beim Hinzufügen eins Discounts zur Lieferung: ${event
.deliveryId}:",
);
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Hinzufügen des Discounts: $e"),
);
}
}
void _load(LoadDeliveryEvent event, Emitter<DeliveryState> emit) async {
debugPrint("Discount; ${event.delivery.discount?.note}");
emit(DeliveryLoaded(delivery: event.delivery));
}
void _unscan(UnscanArticleEvent event, Emitter<DeliveryState> emit) async {
opBloc.add(LoadOperation());
try {
String? noteId = await repository.unscan(
event.articleId,
event.newAmount,
event.reason,
);
if (noteId != null) {
final currentState = state;
if (currentState is DeliveryLoaded) {
Article article = currentState.delivery.articles.firstWhere(
(article) => article.internalId == int.parse(event.articleId),
);
article.removeNoteId = noteId;
article.scannedRemovedAmount += event.newAmount;
article.scannedAmount -= event.newAmount;
List<Article> articles = [
...currentState.delivery.articles.where(
(article) => article.internalId != int.parse(event.articleId),
),
article,
];
currentState.delivery.articles = articles;
emit.call(currentState.copyWith(currentState.delivery));
}
}
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint("Fehler beim Unscan des Artikels: ${event.articleId}:");
debugPrint("$e");
debugPrint("$st");
opBloc.add(FailOperation(message: "Fehler beim Unscan des Artikels: $e"));
}
}
void _resetAmount(ResetScanAmountEvent event,
Emitter<DeliveryState> emit,) async {
opBloc.add(LoadOperation());
try {
final currentState = state;
await repository.resetScan(event.articleId);
if (currentState is DeliveryLoaded) {
Article article = currentState.delivery.articles.firstWhere(
(article) => article.internalId == int.parse(event.articleId),
);
article.removeNoteId = null;
article.scannedRemovedAmount = 0;
article.scannedAmount = article.amount;
List<Article> articles = [
...currentState.delivery.articles.where(
(article) => article.internalId != int.parse(event.articleId),
),
article,
];
currentState.delivery.articles = articles;
emit.call(currentState.copyWith(currentState.delivery));
}
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint("Fehler beim Unscan des Artikels: ${event.articleId}:");
debugPrint("$e");
debugPrint("$st");
opBloc.add(FailOperation(message: "Fehler beim Zurücksetzen: $e"));
}
}
}

View File

@ -1,85 +0,0 @@
import 'dart:typed_data';
import 'package:hl_lieferservice/model/delivery.dart';
import 'package:hl_lieferservice/model/tour.dart';
abstract class DeliveryEvent {}
class LoadDeliveryEvent extends DeliveryEvent {
LoadDeliveryEvent({required this.delivery});
Delivery delivery;
}
class UnscanArticleEvent extends DeliveryEvent {
UnscanArticleEvent({
required this.articleId,
required this.newAmount,
required this.reason,
});
String articleId;
String reason;
int newAmount;
}
class ResetScanAmountEvent extends DeliveryEvent {
ResetScanAmountEvent({required this.articleId});
String articleId;
}
class AddDiscountEvent extends DeliveryEvent {
AddDiscountEvent({
required this.deliveryId,
required this.value,
required this.reason,
});
String deliveryId;
String reason;
int value;
}
class RemoveDiscountEvent extends DeliveryEvent {
RemoveDiscountEvent({required this.deliveryId});
String deliveryId;
}
class UpdateDiscountEvent extends DeliveryEvent {
UpdateDiscountEvent({
required this.deliveryId,
required this.value,
required this.reason,
});
String deliveryId;
String? reason;
int? value;
}
class UpdateDeliveryOptionEvent extends DeliveryEvent {
UpdateDeliveryOptionEvent({required this.key, required this.value});
String key;
dynamic value;
}
class UpdateSelectedPaymentMethodEvent extends DeliveryEvent {
UpdateSelectedPaymentMethodEvent({required this.payment});
Payment payment;
}
class FinishDeliveryEvent extends DeliveryEvent {
FinishDeliveryEvent({
required this.delivery,
required this.driverSignature,
required this.customerSignature,
});
Delivery delivery;
Uint8List customerSignature;
Uint8List driverSignature;
}

View File

@ -1,25 +0,0 @@
import 'package:hl_lieferservice/model/delivery.dart';
abstract class DeliveryState {}
class DeliveryInitial extends DeliveryState {}
class DeliveryLoaded extends DeliveryState {
DeliveryLoaded({required this.delivery});
Delivery delivery;
DeliveryLoaded copyWith(Delivery? delivery) {
return DeliveryLoaded(delivery: delivery ?? this.delivery);
}
}
class DeliveryFinished extends DeliveryState {
DeliveryFinished({required this.delivery});
Delivery delivery;
DeliveryFinished copyWith(Delivery? delivery) {
return DeliveryFinished(delivery: delivery ?? this.delivery);
}
}

View File

@ -1,8 +1,8 @@
import 'dart:async';
import 'dart:typed_data'; import 'dart:typed_data';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/model/delivery.dart';
import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart'; import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart';
import 'package:hl_lieferservice/widget/operations/bloc/operation_event.dart'; import 'package:hl_lieferservice/widget/operations/bloc/operation_event.dart';
@ -13,9 +13,23 @@ import 'package:hl_lieferservice/feature/delivery/detail/repository/note_reposit
class NoteBloc extends Bloc<NoteEvent, NoteState> { class NoteBloc extends Bloc<NoteEvent, NoteState> {
final NoteRepository repository; final NoteRepository repository;
final OperationBloc opBloc; final OperationBloc opBloc;
final String deliveryId;
NoteBloc({required this.repository, required this.opBloc}) StreamSubscription? _noteSubscription;
StreamSubscription? _imageSubscription;
NoteBloc({required this.repository, required this.opBloc, required this.deliveryId})
: super(NoteInitial()) { : super(NoteInitial()) {
repository.loadNotes(deliveryId);
_noteSubscription = repository.notes.listen((notes) {
add(NotesUpdated(notes: notes));
});
_imageSubscription = repository.images.listen((images) {
add(ImageUpdated(images: images));
});
on<LoadNote>(_load); on<LoadNote>(_load);
on<AddNote>(_add); on<AddNote>(_add);
on<EditNote>(_edit); on<EditNote>(_edit);
@ -23,61 +37,41 @@ class NoteBloc extends Bloc<NoteEvent, NoteState> {
on<AddImageNote>(_upload); on<AddImageNote>(_upload);
on<RemoveImageNote>(_removeImage); on<RemoveImageNote>(_removeImage);
on<ResetNotes>(_reset); on<ResetNotes>(_reset);
on<AddNoteOffline>(_addOffline); on<NotesUpdated>(_noteUpdated);
on<ImageUpdated>(_imageUpdated);
}
@override
Future<void> close() {
_noteSubscription?.cancel();
_imageSubscription?.cancel();
return super.close();
}
Future<void> _imageUpdated(ImageUpdated event, Emitter<NoteState> emit) async {
final currentState = state;
if (currentState is NoteLoaded) {
emit.call(currentState.copyWith(images: event.images));
}
}
Future<void> _noteUpdated(NotesUpdated event, Emitter<NoteState> emit) async {
emit(NoteLoaded(notes: event.notes));
} }
Future<void> _reset(ResetNotes event, Emitter<NoteState> emit) async { Future<void> _reset(ResetNotes event, Emitter<NoteState> emit) async {
emit.call(NoteInitial()); emit.call(NoteInitial());
} }
Future<void> _addOffline(AddNoteOffline event,
Emitter<NoteState> emit,) async {
if (state is NoteInitial) {
emit(
NoteLoadedBase(
notes: [Note(content: event.note, id: int.parse(event.noteId))],
),
);
}
if (state is NoteLoadedBase) {
emit(
NoteLoadedBase(
notes: [
...(state as NoteLoadedBase).notes,
Note(content: event.note, id: int.parse(event.noteId)),
],
),
);
}
if (state is NoteLoaded) {
final current = state as NoteLoaded;
emit(NoteLoaded(notes: [...current.notes, Note(content: event.note, id: int.parse(event.noteId))],
templates: [...current.templates],
images: [...current.images]));
}
}
Future<void> _removeImage(RemoveImageNote event, Future<void> _removeImage(RemoveImageNote event,
Emitter<NoteState> emit,) async { Emitter<NoteState> emit,) async {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
final currentState = state;
await repository.deleteImage(event.deliveryId, event.objectId); await repository.deleteImage(event.deliveryId, event.objectId);
if (currentState is NoteLoaded) {
emit.call(
currentState.copyWith(
images:
currentState.images
.where((image) => image.$1.objectId != event.objectId)
.toList(),
),
);
}
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e, st) { } catch (e, st) {
debugPrint("Fehler beim Löschen des Bildes: $e"); debugPrint("Fehler beim Löschen des Bildes: $e");
@ -91,18 +85,8 @@ class NoteBloc extends Bloc<NoteEvent, NoteState> {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
final currentState = state;
Uint8List imageBytes = await event.file.readAsBytes(); Uint8List imageBytes = await event.file.readAsBytes();
ImageNote note = await repository.addImage(event.deliveryId, imageBytes); await repository.addImage(event.deliveryId, imageBytes);
if (currentState is NoteLoaded) {
emit.call(
currentState.copyWith(
images: [...currentState.images, (note, imageBytes)],
),
);
}
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e, st) { } catch (e, st) {
debugPrint("Fehler beim Hinzufügen des Bildes: $e"); debugPrint("Fehler beim Hinzufügen des Bildes: $e");
@ -118,20 +102,12 @@ class NoteBloc extends Bloc<NoteEvent, NoteState> {
try { try {
List<String> urls = List<String> urls =
event.delivery.images.map((image) => image.url).toList(); event.delivery.images.map((image) => image.url).toList();
List<Note> notes = await repository.loadNotes(event.delivery.id);
List<NoteTemplate> templates = await repository.loadTemplates();
List<Uint8List> images = await repository.loadImages(urls);
emit.call( debugPrint("IMAGE URLS : $urls");
NoteLoaded(
notes: notes, await repository.loadNotes(event.delivery.id);
templates: templates, await repository.loadTemplates();
images: List.generate(
images.length,
(index) => (event.delivery.images[index], images[index]),
),
),
);
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e, st) { } catch (e, st) {
debugPrint("Fehler beim Herunterladen der Notizen: $e"); debugPrint("Fehler beim Herunterladen der Notizen: $e");
@ -149,14 +125,7 @@ class NoteBloc extends Bloc<NoteEvent, NoteState> {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
final currentState = state; await repository.addNote(event.deliveryId, event.note);
Note note = await repository.addNote(event.deliveryId, event.note);
if (currentState is NoteLoaded) {
List<Note> refreshedNotes = [...currentState.notes, note];
emit.call(currentState.copyWith(notes: refreshedNotes));
}
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e, st) { } catch (e, st) {
debugPrint("Fehler beim Hinzufügen der Notiz: $e"); debugPrint("Fehler beim Hinzufügen der Notiz: $e");
@ -170,19 +139,7 @@ class NoteBloc extends Bloc<NoteEvent, NoteState> {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
final currentState = state;
await repository.editNote(event.noteId, event.content); await repository.editNote(event.noteId, event.content);
if (currentState is NoteLoaded) {
List<Note> refreshedNotes = [
...currentState.notes.where(
(note) => note.id != int.parse(event.noteId),
),
Note(content: event.content, id: int.parse(event.noteId)),
];
emit.call(currentState.copyWith(notes: refreshedNotes));
}
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e, st) { } catch (e, st) {
debugPrint("Fehler beim Hinzufügen der Notiz: $e"); debugPrint("Fehler beim Hinzufügen der Notiz: $e");
@ -196,18 +153,7 @@ class NoteBloc extends Bloc<NoteEvent, NoteState> {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
final currentState = state;
await repository.deleteNote(event.noteId); await repository.deleteNote(event.noteId);
if (currentState is NoteLoaded) {
List<Note> refreshedNotes =
currentState.notes
.where((note) => note.id != int.parse(event.noteId))
.toList();
emit.call(currentState.copyWith(notes: refreshedNotes));
}
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e, st) { } catch (e, st) {
debugPrint("Fehler beim Hinzufügen der Notiz: $e"); debugPrint("Fehler beim Hinzufügen der Notiz: $e");

View File

@ -1,3 +1,5 @@
import 'dart:typed_data';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
import 'package:image_picker/image_picker.dart'; import 'package:image_picker/image_picker.dart';
@ -52,3 +54,15 @@ class RemoveImageNote extends NoteEvent {
final String objectId; final String objectId;
final String deliveryId; final String deliveryId;
} }
class NotesUpdated extends NoteEvent {
final List<Note> notes;
NotesUpdated({required this.notes});
}
class ImageUpdated extends NoteEvent {
final List<ImageNote> images;
ImageUpdated({required this.images});
}

View File

@ -20,18 +20,18 @@ class NoteLoadedBase extends NoteState {
class NoteLoaded extends NoteLoadedBase { class NoteLoaded extends NoteLoadedBase {
NoteLoaded({ NoteLoaded({
required this.templates, this.templates,
required this.images, this.images,
required super.notes, required super.notes,
}); });
List<NoteTemplate> templates; List<NoteTemplate>? templates;
List<(ImageNote, Uint8List)> images; List<ImageNote>? images;
NoteLoaded copyWith({ NoteLoaded copyWith({
List<Note>? notes, List<Note>? notes,
List<NoteTemplate>? templates, List<NoteTemplate>? templates,
List<(ImageNote, Uint8List)>? images, List<ImageNote>? images,
}) { }) {
return NoteLoaded( return NoteLoaded(
notes: notes ?? this.notes, notes: notes ?? this.notes,

View File

@ -1,3 +1,5 @@
import 'dart:typed_data';
import 'package:hl_lieferservice/model/article.dart'; import 'package:hl_lieferservice/model/article.dart';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';

View File

@ -43,7 +43,11 @@ class _ArticleListItem extends State<ArticleListItem> {
onPressed: () { onPressed: () {
showDialog( showDialog(
context: context, context: context,
builder: (context) => ArticleUnscanDialog(article: widget.article), builder:
(context) => ArticleUnscanDialog(
article: widget.article,
deliveryId: widget.deliveryId,
),
); );
}, },
icon: Icon( icon: Icon(
@ -61,7 +65,10 @@ class _ArticleListItem extends State<ArticleListItem> {
showDialog( showDialog(
context: context, context: context,
builder: builder:
(context) => ResetArticleAmountDialog(article: widget.article), (context) => ResetArticleAmountDialog(
article: widget.article,
deliveryId: widget.deliveryId,
),
); );
}, },
icon: Icon( icon: Icon(

View File

@ -1,14 +1,19 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_event.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_event.dart';
import '../../../../../model/article.dart'; import '../../../../../model/article.dart';
class ResetArticleAmountDialog extends StatefulWidget { class ResetArticleAmountDialog extends StatefulWidget {
const ResetArticleAmountDialog({super.key, required this.article}); const ResetArticleAmountDialog({
super.key,
required this.article,
required this.deliveryId,
});
final Article article; final Article article;
final String deliveryId;
@override @override
State<StatefulWidget> createState() => _ResetArticleAmountDialogState(); State<StatefulWidget> createState() => _ResetArticleAmountDialogState();
@ -16,8 +21,11 @@ class ResetArticleAmountDialog extends StatefulWidget {
class _ResetArticleAmountDialogState extends State<ResetArticleAmountDialog> { class _ResetArticleAmountDialogState extends State<ResetArticleAmountDialog> {
void _reset() { void _reset() {
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
ResetScanAmountEvent(articleId: widget.article.internalId.toString()), ResetScanAmountEvent(
articleId: widget.article.internalId.toString(),
deliveryId: widget.deliveryId,
),
); );
Navigator.pop(context); Navigator.pop(context);

View File

@ -1,14 +1,15 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter/services.dart'; import 'package:flutter/services.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_event.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_event.dart';
import '../../../../../model/article.dart'; import '../../../../../model/article.dart';
class ArticleUnscanDialog extends StatefulWidget { class ArticleUnscanDialog extends StatefulWidget {
const ArticleUnscanDialog({super.key, required this.article}); const ArticleUnscanDialog({super.key, required this.article, required this.deliveryId});
final String deliveryId;
final Article article; final Article article;
@override @override
@ -22,8 +23,9 @@ class _ArticleUnscanDialogState extends State<ArticleUnscanDialog> {
final _formKey = GlobalKey<FormState>(); final _formKey = GlobalKey<FormState>();
void _unscan() { void _unscan() {
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
UnscanArticleEvent( UnscanArticleEvent(
deliveryId: widget.deliveryId,
articleId: widget.article.internalId.toString(), articleId: widget.article.internalId.toString(),
newAmount: int.parse(unscanAmountController.text), newAmount: int.parse(unscanAmountController.text),
reason: unscanNoteController.text, reason: unscanNoteController.text,

View File

@ -3,23 +3,19 @@ import 'dart:typed_data';
import 'package:easy_stepper/easy_stepper.dart'; import 'package:easy_stepper/easy_stepper.dart';
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_event.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_state.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_event.dart'; import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_event.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_state.dart';
import 'package:hl_lieferservice/feature/delivery/detail/presentation/delivery_sign.dart'; import 'package:hl_lieferservice/feature/delivery/detail/presentation/delivery_sign.dart';
import 'package:hl_lieferservice/feature/delivery/detail/presentation/steps/step.dart'; import 'package:hl_lieferservice/feature/delivery/detail/presentation/steps/step.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_event.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_event.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_state.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_state.dart';
import 'package:hl_lieferservice/model/delivery.dart' as model; import 'package:hl_lieferservice/model/delivery.dart';
class DeliveryDetail extends StatefulWidget { class DeliveryDetail extends StatefulWidget {
final model.Delivery delivery; final String deliveryId;
const DeliveryDetail({super.key, required this.delivery}); const DeliveryDetail({super.key, required this.deliveryId});
@override @override
State<StatefulWidget> createState() => _DeliveryDetailState(); State<StatefulWidget> createState() => _DeliveryDetailState();
@ -33,11 +29,6 @@ class _DeliveryDetailState extends State<DeliveryDetail> {
void initState() { void initState() {
super.initState(); super.initState();
// Initialize BLOC
context.read<DeliveryBloc>().add(
LoadDeliveryEvent(delivery: widget.delivery),
);
// Reset Note BLOC // Reset Note BLOC
// otherwise the notes of the previously // otherwise the notes of the previously
// opened delivery would be loaded // opened delivery would be loaded
@ -124,28 +115,27 @@ class _DeliveryDetailState extends State<DeliveryDetail> {
} }
} }
void _openSignatureView() { void _openSignatureView(Delivery delivery) {
Navigator.of(context).push( Navigator.of(context).push(
MaterialPageRoute( MaterialPageRoute(
builder: builder:
(context) => (context) =>
SignatureView(onSigned: _onSign, delivery: widget.delivery), SignatureView(onSigned: _onSign, delivery: delivery),
), ),
); );
} }
void _onSign(Uint8List customer, Uint8List driver) async { void _onSign(Uint8List customer, Uint8List driver) async {
final currentState = context.read<DeliveryBloc>().state as DeliveryLoaded; context.read<TourBloc>().add(
context.read<DeliveryBloc>().add(
FinishDeliveryEvent( FinishDeliveryEvent(
delivery: currentState.delivery, deliveryId: widget.deliveryId,
customerSignature: customer, customerSignature: customer,
driverSignature: driver, driverSignature: driver,
), ),
); );
} }
Widget _stepsNavigation() { Widget _stepsNavigation(Delivery delivery) {
return SizedBox( return SizedBox(
width: double.infinity, width: double.infinity,
height: 90, height: 90,
@ -160,8 +150,8 @@ class _DeliveryDetailState extends State<DeliveryDetail> {
padding: const EdgeInsets.only(left: 20), padding: const EdgeInsets.only(left: 20),
child: FilledButton( child: FilledButton(
onPressed: onPressed:
_step == _steps.length - 1 () => _step == _steps.length - 1
? _openSignatureView ? _openSignatureView(delivery)
: _clickForward, : _clickForward,
child: child:
_step == _steps.length - 1 _step == _steps.length - 1
@ -178,48 +168,27 @@ class _DeliveryDetailState extends State<DeliveryDetail> {
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Scaffold( return Scaffold(
appBar: AppBar(title: const Text("Auslieferungsdetails")), appBar: AppBar(title: const Text("Auslieferungsdetails")),
body: BlocConsumer<DeliveryBloc, DeliveryState>( body: BlocBuilder<TourBloc, TourState>(builder: (context, state) {
listener: (context, state) {
if (state is DeliveryFinished) {
final tourState = context.read<TourBloc>().state as TourLoaded;
final newTour = tourState.tour.copyWith(
deliveries:
tourState.tour.deliveries.map((delivery) {
if (delivery.id == state.delivery.id) {
return state.delivery;
}
return delivery;
}).toList(),
);
context.read<TourBloc>().add(UpdateTour(tour: newTour));
Navigator.pop(context);
Navigator.pop(context);
}
},
builder: (context, state) {
final currentState = state; final currentState = state;
if (currentState is DeliveryLoaded) { if (currentState is TourLoaded) {
Delivery delivery = currentState.tour.deliveries.firstWhere((delivery) => delivery.id == widget.deliveryId);
return Column( return Column(
children: [ children: [
_stepInfo(), _stepInfo(),
const Divider(), const Divider(),
Expanded( Expanded(
child: child:
StepFactory().make(_step, currentState.delivery) ?? StepFactory().make(_step, delivery) ??
_stepMissingWarning(), _stepMissingWarning(),
), ),
_stepsNavigation(), _stepsNavigation(delivery),
], ],
); );
} }
return Container(); return const Center(child: CircularProgressIndicator(),);
}, }),
),
); );
} }
} }

View File

@ -1,9 +1,10 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_event.dart';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
import '../../overview/bloc/tour_event.dart';
class DeliveryDiscount extends StatefulWidget { class DeliveryDiscount extends StatefulWidget {
const DeliveryDiscount({ const DeliveryDiscount({
super.key, super.key,
@ -126,14 +127,14 @@ class _DeliveryDiscountState extends State<DeliveryDiscount> {
_isUpdated = false; _isUpdated = false;
}); });
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
RemoveDiscountEvent(deliveryId: widget.deliveryId), RemoveDiscountEvent(deliveryId: widget.deliveryId),
); );
} }
void _updateValues() async { void _updateValues() async {
if (_isUpdated) { if (_isUpdated) {
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
UpdateDiscountEvent( UpdateDiscountEvent(
deliveryId: widget.deliveryId, deliveryId: widget.deliveryId,
value: _discountValue, value: _discountValue,
@ -141,7 +142,7 @@ class _DeliveryDiscountState extends State<DeliveryDiscount> {
), ),
); );
} else { } else {
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
AddDiscountEvent( AddDiscountEvent(
deliveryId: widget.deliveryId, deliveryId: widget.deliveryId,
value: _discountValue, value: _discountValue,

View File

@ -1,13 +1,15 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_event.dart';
import 'package:hl_lieferservice/model/delivery.dart' as model; import 'package:hl_lieferservice/model/delivery.dart' as model;
import '../../overview/bloc/tour_event.dart';
class DeliveryOptionsView extends StatefulWidget { class DeliveryOptionsView extends StatefulWidget {
const DeliveryOptionsView({super.key, required this.options}); const DeliveryOptionsView({super.key, required this.options, required this.deliveryId});
final List<model.DeliveryOption> options; final List<model.DeliveryOption> options;
final String deliveryId;
@override @override
State<StatefulWidget> createState() => _DeliveryOptionsViewState(); State<StatefulWidget> createState() => _DeliveryOptionsViewState();
@ -28,17 +30,16 @@ class _DeliveryOptionsViewState extends State<DeliveryOptionsView> {
debugPrint(option.key); debugPrint(option.key);
if (value is bool) { if (value is bool) {
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
UpdateDeliveryOptionEvent(key: option.key, value: !value), UpdateDeliveryOptionEvent(key: option.key, value: !value, deliveryId: widget.deliveryId),
); );
return; return;
} }
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
UpdateDeliveryOptionEvent(key: option.key, value: value), UpdateDeliveryOptionEvent(key: option.key, value: value, deliveryId: widget.deliveryId),
); );
} }
List<Widget> _options() { List<Widget> _options() {

View File

@ -1,8 +1,7 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_event.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_event.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_state.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_state.dart';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
@ -97,11 +96,10 @@ class _DeliverySummaryState extends State<DeliverySummary> {
dropdownMenuEntries: entries, dropdownMenuEntries: entries,
initialSelection: widget.delivery.payment.id, initialSelection: widget.delivery.payment.id,
onSelected: (id) { onSelected: (id) {
context.read<DeliveryBloc>().add( context.read<TourBloc>().add(
UpdateSelectedPaymentMethodEvent( UpdateSelectedPaymentMethodEvent(
payment: _paymentMethods.firstWhere( deliveryId: widget.delivery.id,
(payment) => payment.id == id, payment: _paymentMethods.firstWhere((payment) => payment.id == id),
),
), ),
); );
}, },

View File

@ -9,7 +9,7 @@ import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_event.dart';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
class NoteImageOverview extends StatefulWidget { class NoteImageOverview extends StatefulWidget {
final List<(ImageNote, Uint8List)> images; final List<ImageNote> images;
final String deliveryId; final String deliveryId;
const NoteImageOverview({ const NoteImageOverview({
@ -26,7 +26,7 @@ class _NoteImageOverviewState extends State<NoteImageOverview> {
int? _imageDeleting; int? _imageDeleting;
void _onRemoveImage(int index) { void _onRemoveImage(int index) {
ImageNote note = widget.images[index].$1; ImageNote note = widget.images[index];
context.read<NoteBloc>().add( context.read<NoteBloc>().add(
RemoveImageNote(objectId: note.objectId, deliveryId: widget.deliveryId), RemoveImageNote(objectId: note.objectId, deliveryId: widget.deliveryId),
@ -42,8 +42,7 @@ class _NoteImageOverviewState extends State<NoteImageOverview> {
), ),
items: items:
widget.images.mapIndexed((index, data) { widget.images.mapIndexed((index, data) {
ImageNote note = data.$1; Uint8List bytes = data.data!;
Uint8List bytes = data.$2;
return Builder( return Builder(
builder: (BuildContext context) { builder: (BuildContext context) {

View File

@ -16,7 +16,7 @@ import 'package:image_picker/image_picker.dart';
class NoteOverview extends StatefulWidget { class NoteOverview extends StatefulWidget {
final List<NoteInformation> notes; final List<NoteInformation> notes;
final List<NoteTemplate> templates; final List<NoteTemplate> templates;
final List<(ImageNote, Uint8List)> images; final List<ImageNote> images;
final String deliveryId; final String deliveryId;
const NoteOverview({ const NoteOverview({

View File

@ -17,6 +17,9 @@ class _DeliveryStepInfo extends State<DeliveryStepOptions> {
debugPrint( debugPrint(
"${widget.delivery.options.map((option) => "${option.display}, ${option.value}")}", "${widget.delivery.options.map((option) => "${option.display}, ${option.value}")}",
); );
return DeliveryOptionsView(options: widget.delivery.options); return DeliveryOptionsView(
options: widget.delivery.options,
deliveryId: widget.delivery.id,
);
} }
} }

View File

@ -41,7 +41,7 @@ class _DeliveryStepInfo extends State<DeliveryStepNote> {
BuildContext context, BuildContext context,
List<Note> notes, List<Note> notes,
List<NoteTemplate> templates, List<NoteTemplate> templates,
List<(ImageNote, Uint8List)> images, List<ImageNote> images,
) { ) {
List<NoteInformation> hydratedNotes = List<NoteInformation> hydratedNotes =
notes notes
@ -75,8 +75,8 @@ class _DeliveryStepInfo extends State<DeliveryStepNote> {
return _notesOverview( return _notesOverview(
context, context,
state.notes, state.notes,
state.templates, (state.templates ?? []),
state.images, (state.images ?? []),
); );
} }

View File

@ -1,13 +1,17 @@
import 'dart:typed_data';
import 'package:hl_lieferservice/dto/discount_add_response.dart'; import 'package:hl_lieferservice/dto/discount_add_response.dart';
import 'package:hl_lieferservice/dto/discount_remove_response.dart'; import 'package:hl_lieferservice/dto/discount_remove_response.dart';
import 'package:hl_lieferservice/dto/discount_update_response.dart'; import 'package:hl_lieferservice/dto/discount_update_response.dart';
import 'package:hl_lieferservice/feature/delivery/detail/repository/note_repository.dart';
import 'package:hl_lieferservice/feature/delivery/detail/service/notes_service.dart';
import 'package:hl_lieferservice/feature/delivery/overview/service/delivery_info_service.dart'; import 'package:hl_lieferservice/feature/delivery/overview/service/delivery_info_service.dart';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
class DeliveryRepository { class DeliveryRepository {
DeliveryRepository({required this.service}); DeliveryRepository({required this.service});
DeliveryInfoService service; TourService service;
Future<String?> unscan(String articleId, int newAmount, String reason) async { Future<String?> unscan(String articleId, int newAmount, String reason) async {
return await service.unscanArticle(articleId, newAmount, reason); return await service.unscanArticle(articleId, newAmount, reason);
@ -17,6 +21,16 @@ class DeliveryRepository {
return await service.resetScannedArticleAmount(articleId); return await service.resetScannedArticleAmount(articleId);
} }
Future<void> uploadDriverSignature(String deliveryId, Uint8List signature) async {
NoteRepository noteRepository = NoteRepository(service: NoteService());
await noteRepository.addNamedImage(deliveryId, signature, "delivery_${deliveryId}_signature_driver.jpg");
}
Future<void> uploadCustomerSignature(String deliveryId, Uint8List signature) async {
NoteRepository noteRepository = NoteRepository(service: NoteService());
await noteRepository.addNamedImage(deliveryId, signature, "delivery_${deliveryId}_signature_customer.jpg");
}
Future<DiscountAddResponseDTO> addDiscount( Future<DiscountAddResponseDTO> addDiscount(
String deliveryId, String deliveryId,
String reason, String reason,

View File

@ -2,43 +2,61 @@ import 'dart:typed_data';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
import 'package:hl_lieferservice/feature/delivery/detail/service/notes_service.dart'; import 'package:hl_lieferservice/feature/delivery/detail/service/notes_service.dart';
import 'package:rxdart/rxdart.dart';
class NoteRepository { class NoteRepository {
final NoteService service; final NoteService service;
final _notesStream = BehaviorSubject<List<Note>>.seeded([]);
final _imageNoteStream = BehaviorSubject<List<ImageNote>>.seeded([]);
final _noteTemplateStream = BehaviorSubject<List<NoteTemplate>>.seeded([]);
Stream<List<Note>> get notes => _notesStream.stream;
Stream<List<ImageNote>> get images => _imageNoteStream.stream;
Stream<List<NoteTemplate>> get templates => _noteTemplateStream.stream;
NoteRepository({required this.service}); NoteRepository({required this.service});
Future<Note> addNote(String deliveryId, String content) async { Future<void> addNote(String deliveryId, String content) async {
return service.addNote(content, int.parse(deliveryId)); final note = await service.addNote(content, int.parse(deliveryId));
final currentNotes = _notesStream.value;
currentNotes.add(note);
_notesStream.add(currentNotes);
} }
Future<void> editNote(String noteId, String content) async { Future<void> editNote(String noteId, String content) async {
return service.editNote(Note(content: content, id: int.parse(noteId))); final newNote = Note(content: content, id: int.parse(noteId));
await service.editNote(newNote);
final currentNotes = _notesStream.value;
final index = currentNotes.indexWhere((note) => note.id == int.parse(noteId));
currentNotes[index] = newNote;
_notesStream.add(currentNotes);
} }
Future<void> deleteNote(String noteId) async { Future<void> deleteNote(String noteId) async {
return service.deleteNote(int.parse(noteId)); await service.deleteNote(int.parse(noteId));
final currentNotes = _notesStream.value;
final index = currentNotes.indexWhere((note) => note.id == int.parse(noteId));
currentNotes.removeAt(index);
_notesStream.add(currentNotes);
} }
Future<List<Note>> loadNotes(String deliveryId) async { Future<void> loadNotes(String deliveryId) async {
return service.getNotes(deliveryId); var (notes, images) = await service.getNotes(deliveryId);
_notesStream.add(notes);
_imageNoteStream.add(images);
} }
Future<List<Uint8List>> loadImages(List<String> urls) async { Future<void> loadTemplates() async {
List<Uint8List> images = []; _noteTemplateStream.add(await service.getNoteTemplates());
for (final image in await service.downloadImages(urls)) {
images.add(await image);
} }
return images; Future<void> addImage(String deliveryId, Uint8List bytes) async {
}
Future<List<NoteTemplate>> loadTemplates() async {
return service.getNoteTemplates();
}
Future<ImageNote> addImage(String deliveryId, Uint8List bytes) async {
final fileName = final fileName =
"delivery_note_${deliveryId}_${DateTime.timestamp().microsecondsSinceEpoch}.jpg"; "delivery_note_${deliveryId}_${DateTime.timestamp().microsecondsSinceEpoch}.jpg";
@ -48,11 +66,12 @@ class NoteRepository {
bytes, bytes,
"image/png", "image/png",
); );
final imageNotes = _imageNoteStream.value;
return ImageNote.make(objectId, fileName); imageNotes.add(ImageNote.make(objectId, fileName, bytes));
_imageNoteStream.add(imageNotes);
} }
Future<ImageNote> addNamedImage(String deliveryId, Uint8List bytes, String filename) async { Future<void> addNamedImage(String deliveryId, Uint8List bytes, String filename) async {
String objectId = await service.uploadImage( String objectId = await service.uploadImage(
deliveryId, deliveryId,
filename, filename,
@ -60,10 +79,18 @@ class NoteRepository {
"image/png", "image/png",
); );
return ImageNote.make(objectId, filename); final imageNotes = _imageNoteStream.value;
imageNotes.add(ImageNote.make(objectId, filename, bytes));
_imageNoteStream.add(imageNotes);
} }
Future<void> deleteImage(String deliveryId, String objectId) async { Future<void> deleteImage(String deliveryId, String objectId) async {
await service.removeImage(objectId); await service.removeImage(objectId);
final images = _imageNoteStream.value;
final index = images.indexWhere((imageNote) => imageNote.objectId == objectId);
images.removeAt(index);
_imageNoteStream.add(images);
} }
} }

View File

@ -17,9 +17,7 @@ import '../../../../model/delivery.dart';
import '../../../../util.dart'; import '../../../../util.dart';
import '../../../authentication/exceptions.dart'; import '../../../authentication/exceptions.dart';
class NoteService extends ErpFrameService { class NoteService {
NoteService({required super.backendUrl});
Future<void> deleteNote(int noteId) async { Future<void> deleteNote(int noteId) async {
try { try {
var response = await http.post( var response = await http.post(
@ -110,7 +108,7 @@ class NoteService extends ErpFrameService {
} }
} }
Future<List<Note>> getNotes(String deliveryId) async { Future<(List<Note>, List<ImageNote>)> getNotes(String deliveryId) async {
try { try {
var response = await http.post( var response = await http.post(
urlBuilder("_web_getNotes"), urlBuilder("_web_getNotes"),
@ -129,9 +127,22 @@ class NoteService extends ErpFrameService {
); );
if (responseDto.succeeded == true) { if (responseDto.succeeded == true) {
return responseDto.notes List<ImageNote> imageNotes =
.map((noteDto) => Note.fromDto(noteDto)) responseDto.images
.map((imageNoteDto) => ImageNote.fromDTO(imageNoteDto))
.toList(); .toList();
final images = await downloadImages(imageNotes.map((note) => note.url).toList());
for (var (index, note) in imageNotes.indexed) {
note.data = await images[index];
}
return (
responseDto.notes
.map((noteDto) => Note.fromDto(noteDto))
.toList(),
imageNotes
);
} else { } else {
throw responseDto.message; throw responseDto.message;
} }
@ -262,9 +273,11 @@ class NoteService extends ErpFrameService {
Future<List<Future<Uint8List>>> downloadImages(List<String> urls) async { Future<List<Future<Uint8List>>> downloadImages(List<String> urls) async {
try { try {
LocalDocuFrameConfiguration config = getConfig();
return urls.map((url) async { return urls.map((url) async {
return (await http.get( return (await http.get(
Uri.parse("$backendUrl$url"), Uri.parse("${config.backendUrl}$url"),
headers: getSessionOrThrow(), headers: getSessionOrThrow(),
)).bodyBytes; )).bodyBytes;
}).toList(); }).toList();

View File

@ -1,31 +1,120 @@
import 'package:flutter/cupertino.dart'; import 'dart:async';
import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart'; import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_event.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_event.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_state.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_state.dart';
import 'package:hl_lieferservice/feature/delivery/overview/repository/tour_repository.dart'; import 'package:hl_lieferservice/feature/delivery/overview/repository/tour_repository.dart';
import 'package:hl_lieferservice/feature/delivery/overview/service/distance_service.dart'; import 'package:hl_lieferservice/feature/delivery/overview/service/distance_service.dart';
import 'package:hl_lieferservice/model/delivery.dart';
import 'package:hl_lieferservice/model/tour.dart'; import 'package:hl_lieferservice/model/tour.dart';
import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart'; import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart';
import 'package:hl_lieferservice/widget/operations/bloc/operation_event.dart'; import 'package:hl_lieferservice/widget/operations/bloc/operation_event.dart';
import 'package:rxdart/rxdart.dart';
class TourBloc extends Bloc<TourEvent, TourState> { class TourBloc extends Bloc<TourEvent, TourState> {
OperationBloc opBloc; OperationBloc opBloc;
TourRepository tourRepository; TourRepository tourRepository;
StreamSubscription? _combinedSubscription;
TourBloc({required this.opBloc, required this.tourRepository}) TourBloc({required this.opBloc, required this.tourRepository})
: super(TourInitial()) { : super(TourInitial()) {
_combinedSubscription = CombineLatestStream.combine2(
tourRepository.tour,
tourRepository.paymentOptions,
(tour, payments) => {'tour': tour, 'payments': payments},
).listen((combined) {
final tour = combined['tour'] as Tour?;
final payments = combined['payments'] as List<Payment>;
if (tour == null) {
return;
}
add(TourUpdated(tour: tour, payments: payments));
});
on<LoadTour>(_load); on<LoadTour>(_load);
on<UpdateTour>(_update);
on<AssignCarEvent>(_assignCar); on<AssignCarEvent>(_assignCar);
on<IncrementArticleScanAmount>(_increment); on<IncrementArticleScanAmount>(_increment);
on<ScanArticleEvent>(_scan); on<ScanArticleEvent>(_scan);
on<HoldDeliveryEvent>(_holdDelivery); on<HoldDeliveryEvent>(_holdDelivery);
on<CancelDeliveryEvent>(_cancelDelivery); on<CancelDeliveryEvent>(_cancelDelivery);
on<ReactivateDeliveryEvent>(_reactiveateDelivery); on<ReactivateDeliveryEvent>(_reactivateDelivery);
on<UnscanArticleEvent>(_unscan);
on<ResetScanAmountEvent>(_resetAmount);
on<AddDiscountEvent>(_addDiscount);
on<RemoveDiscountEvent>(_removeDiscount);
on<UpdateDiscountEvent>(_updateDiscount);
on<UpdateDeliveryOptionEvent>(_updateDeliveryOptions);
on<UpdateSelectedPaymentMethodEvent>(_updatePayment);
on<FinishDeliveryEvent>(_finishDelivery);
on<TourUpdated>(_updated);
on<RequestDeliveryDistanceEvent>(_calculateDistances);
} }
void _reactiveateDelivery( @override
Future<void> close() {
_combinedSubscription?.cancel();
return super.close();
}
void _calculateDistances(
RequestDeliveryDistanceEvent event,
Emitter<TourState> emit,
) async {
Map<String, double> distances = {};
opBloc.add(LoadOperation());
emit(TourRequestingDistances(tour: event.tour, payments: event.payments));
try {
for (final delivery in event.tour.deliveries) {
distances[delivery.id] = await DistanceService.getDistanceByRoad(
delivery.customer.address.toString(),
);
}
opBloc.add(FinishOperation());
} catch (e) {
debugPrint("Fehler beim Berechnen der Distanzen: $e");
opBloc.add(FailOperation(message: "Fehler beim Berechnen der Distanzen"));
return;
}
emit(
TourLoaded(
tour: event.tour,
paymentOptions: event.payments,
distances: distances,
),
);
}
void _updated(TourUpdated event, Emitter<TourState> emit) {
final currentState = state;
final tour = event.tour.copyWith();
final payments =
event.payments.map((payment) => payment.copyWith()).toList();
if (currentState is TourLoaded) {
debugPrint("TEST UPDATE");
emit(
TourLoaded(
tour: tour,
paymentOptions: payments,
distances: Map<String, double>.from(currentState.distances ?? {}),
),
);
}
if (currentState is TourLoading) {
add(
RequestDeliveryDistanceEvent(tour: tour.copyWith(), payments: payments),
);
}
}
void _reactivateDelivery(
ReactivateDeliveryEvent event, ReactivateDeliveryEvent event,
Emitter<TourState> emit, Emitter<TourState> emit,
) async { ) async {
@ -33,17 +122,9 @@ class TourBloc extends Bloc<TourEvent, TourState> {
if (currentState is TourLoaded) { if (currentState is TourLoaded) {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
Tour tourCopied = currentState.tour.copyWith(); await tourRepository.reactivateDelivery(event.deliveryId);
Delivery delivery = tourCopied.deliveries.firstWhere((delivery) => delivery.id == event.deliveryId);
delivery.state = DeliveryState.ongoing;
await tourRepository.updateDelivery(
delivery,
);
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
emit(TourLoaded(tour: tourCopied, distances: currentState.distances));
} catch (e, st) { } catch (e, st) {
debugPrint("$e"); debugPrint("$e");
debugPrint("$st"); debugPrint("$st");
@ -54,51 +135,35 @@ class TourBloc extends Bloc<TourEvent, TourState> {
} }
} }
void _holdDelivery( void _holdDelivery(HoldDeliveryEvent event, Emitter<TourState> emit) async {
HoldDeliveryEvent event, final currentState = state;
if (currentState is TourLoaded) {
opBloc.add(LoadOperation());
try {
await tourRepository.holdDelivery(event.deliveryId);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Zurückstellen der Lieferung"),
);
}
}
}
void _cancelDelivery(
CancelDeliveryEvent event,
Emitter<TourState> emit, Emitter<TourState> emit,
) async { ) async {
final currentState = state; final currentState = state;
if (currentState is TourLoaded) { if (currentState is TourLoaded) {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
Tour tourCopied = currentState.tour.copyWith(); await tourRepository.cancelDelivery(event.deliveryId);
Delivery delivery = tourCopied.deliveries.firstWhere((delivery) => delivery.id == event.deliveryId);
delivery.state = DeliveryState.onhold;
await tourRepository.updateDelivery(
delivery,
);
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
emit(TourLoaded(tour: tourCopied, distances: currentState.distances));
} catch (e, st) {
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Zurückstellen der Lieferung"),
);
}
}
}
void _cancelDelivery(CancelDeliveryEvent event, Emitter<TourState> emit) async {
final currentState = state;
if (currentState is TourLoaded) {
opBloc.add(LoadOperation());
try {
Tour tourCopied = currentState.tour.copyWith();
Delivery delivery = tourCopied.deliveries.firstWhere((delivery) => delivery.id == event.deliveryId);
delivery.state = DeliveryState.canceled;
await tourRepository.updateDelivery(
delivery,
);
opBloc.add(FinishOperation());
emit(TourLoaded(tour: tourCopied, distances: currentState.distances));
} catch (e, st) { } catch (e, st) {
debugPrint("$e"); debugPrint("$e");
debugPrint("$st"); debugPrint("$st");
@ -115,51 +180,30 @@ class TourBloc extends Bloc<TourEvent, TourState> {
if (currentState is TourLoaded) { if (currentState is TourLoaded) {
try { try {
if (currentState.tour.deliveries.any( switch (await tourRepository.scanArticle(
(delivery) => delivery.articles.any( event.deliveryId,
(article) => article.articleNumber == event.articleNumber, event.carId,
), event.articleNumber,
)) { )) {
var tourCopied = currentState.tour.copyWith(); case ScanResult.scanned:
var delivery = tourCopied.deliveries.firstWhere( opBloc.add(FinishOperation(message: 'Artikel gescannt'));
(delivery) => delivery.id == event.deliveryId, break;
case ScanResult.alreadyScanned:
opBloc.add(
FailOperation(message: 'Artikel wurde bereits gescannt'),
); );
var article = delivery.articles.firstWhere( break;
(article) => article.articleNumber == event.articleNumber, case ScanResult.notFound:
);
await tourRepository.scanArticle(article.internalId.toString());
if (article.scannedAmount < article.amount) {
article.scannedAmount += 1;
emit(TourLoaded(tour: tourCopied, distances: currentState.distances));
opBloc.add(FinishOperation(message: '${article.name} gescannt'));
} else {
opBloc.add( opBloc.add(
FailOperation( FailOperation(
message: 'Alle ${article.name} wurden bereits gescannt', message: 'Artikel ist für keine Lieferung vorgesehen',
), ),
); );
break;
} }
} else { } on TourNotFoundException catch (e) {
opBloc.add( opBloc.add(FailOperation(message: "Fehler beim Scannen des Artikels"));
FailOperation(
message: 'Fehler: Artikel ist für keine Lieferung vorgesehen',
),
);
} }
} catch (e, st) {
debugPrint(st.toString());
opBloc.add(FailOperation(message: "Fehler beim Scannnen des Artikels"));
}
}
}
Future<void> _update(UpdateTour event, Emitter<TourState> emit) async {
final currentState = state;
if (currentState is TourLoaded) {
emit(TourLoaded(tour: event.tour, distances: currentState.distances));
} }
} }
@ -170,62 +214,27 @@ class TourBloc extends Bloc<TourEvent, TourState> {
final currentState = state; final currentState = state;
if (currentState is TourLoaded) { if (currentState is TourLoaded) {
var deliveryCopied = currentState.tour.deliveries.firstWhere( opBloc.add(LoadOperation());
(delivery) => delivery.id == event.deliveryId, try {
await tourRepository.scanArticle(
event.deliveryId,
event.carId,
event.internalArticleId,
); );
var articleCopied = deliveryCopied.articles.firstWhere( opBloc.add(FinishOperation());
(article) => article.internalId == int.parse(event.internalArticleId), } catch (e, st) {
); debugPrint(st.toString());
articleCopied.scannedAmount += 1; opBloc.add(FailOperation(message: "Fehler beim Scannen des Artikels"));
emit(
TourLoaded(
tour: currentState.tour.copyWith(
deliveries:
currentState.tour.deliveries.map((delivery) {
if (delivery.id == event.deliveryId) {
return deliveryCopied;
} }
return delivery;
}).toList(),
),
distances: currentState.distances
),
);
} }
} }
Future<void> _assignCar(AssignCarEvent event, Emitter<TourState> emit) async { Future<void> _assignCar(AssignCarEvent event, Emitter<TourState> emit) async {
final currentState = state; final currentState = state;
if (currentState is TourLoaded) { if (currentState is TourLoaded) {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
var copiedTour = currentState.tour.copyWith();
var delivery = copiedTour.deliveries.firstWhere(
(delivery) => delivery.id == event.deliveryId,
);
try { try {
await tourRepository.assignCar(event.deliveryId, event.carId); await tourRepository.assignCar(event.deliveryId, event.carId);
delivery.carId = int.parse(event.carId);
emit(
TourLoaded(
tour: copiedTour.copyWith(
deliveries:
copiedTour.deliveries.map((d) {
if (d.id == delivery.id) {
return delivery;
}
return d;
}).toList(),
),
distances: currentState.distances
),
);
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e, st) { } catch (e, st) {
debugPrint(st.toString()); debugPrint(st.toString());
@ -239,16 +248,10 @@ class TourBloc extends Bloc<TourEvent, TourState> {
Future<void> _load(LoadTour event, Emitter<TourState> emit) async { Future<void> _load(LoadTour event, Emitter<TourState> emit) async {
opBloc.add(LoadOperation()); opBloc.add(LoadOperation());
try { try {
Tour tour = await tourRepository.loadAll(event.teamId); emit(TourLoading());
List<Payment> payments = await tourRepository.loadPaymentOptions(); await tourRepository.loadTourOfToday(event.teamId);
tour.paymentMethods = payments; await tourRepository.loadPaymentOptions();
Map<String, double> distances = {};
for (final delivery in tour.deliveries) {
distances[delivery.id] = await DistanceService.getDistanceByRoad(delivery.customer.address.toString());
}
emit(TourLoaded(tour: tour, distances: distances));
opBloc.add(FinishOperation()); opBloc.add(FinishOperation());
} catch (e) { } catch (e) {
opBloc.add( opBloc.add(
@ -256,4 +259,177 @@ class TourBloc extends Bloc<TourEvent, TourState> {
); );
} }
} }
void _finishDelivery(
FinishDeliveryEvent event,
Emitter<TourState> emit,
) async {
final currentState = state;
opBloc.add(LoadOperation());
if (currentState is TourLoaded) {
try {
await tourRepository.uploadDriverSignature(
event.deliveryId,
event.driverSignature,
);
await tourRepository.uploadCustomerSignature(
event.deliveryId,
event.customerSignature,
);
await tourRepository.finishDelivery(event.deliveryId);
opBloc.add(FinishOperation());
} catch (e, st) {
opBloc.add(FailOperation(message: "Failed to update delivery"));
debugPrint(st.toString());
}
}
}
void _updatePayment(
UpdateSelectedPaymentMethodEvent event,
Emitter<TourState> emit,
) async {
try {
opBloc.add(LoadOperation());
await tourRepository.updatePayment(event.deliveryId, event.payment);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint(st.toString());
opBloc.add(
FailOperation(message: "Fehler beim Aktualisieren des Betrags"),
);
}
}
void _updateDeliveryOptions(
UpdateDeliveryOptionEvent event,
Emitter<TourState> emit,
) async {
try {
opBloc.add(LoadOperation());
await tourRepository.updateOption(
event.deliveryId,
event.key,
event.value,
);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Aktualisieren der Optionen"),
);
}
}
void _updateDiscount(
UpdateDiscountEvent event,
Emitter<TourState> emit,
) async {
opBloc.add(LoadOperation());
try {
opBloc.add(FinishOperation());
await tourRepository.updateDiscount(
event.deliveryId,
event.reason,
event.value,
);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint(
"Fehler beim Hinzufügen eins Discounts zur Lieferung: ${event.deliveryId}:",
);
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Hinzufügen des Discounts: $e"),
);
}
}
void _removeDiscount(
RemoveDiscountEvent event,
Emitter<TourState> emit,
) async {
opBloc.add(LoadOperation());
try {
await tourRepository.removeDiscount(event.deliveryId);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint(
"Fehler beim Löschen des Discounts der Lieferung: ${event.deliveryId}:",
);
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Löschen des Discounts: $e"),
);
}
}
void _addDiscount(AddDiscountEvent event, Emitter<TourState> emit) async {
opBloc.add(LoadOperation());
try {
await tourRepository.addDiscount(
event.deliveryId,
event.reason,
event.value,
);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint(
"Fehler beim Hinzufügen eins Discounts zur Lieferung: ${event.deliveryId}:",
);
debugPrint("$e");
debugPrint("$st");
opBloc.add(
FailOperation(message: "Fehler beim Hinzufügen des Discounts: $e"),
);
}
}
void _unscan(UnscanArticleEvent event, Emitter<TourState> emit) async {
opBloc.add(LoadOperation());
try {
await tourRepository.unscan(
event.deliveryId,
event.articleId,
event.newAmount,
event.reason,
);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint("Fehler beim Unscan des Artikels: ${event.articleId}:");
debugPrint("$e");
debugPrint("$st");
opBloc.add(FailOperation(message: "Fehler beim Unscan des Artikels: $e"));
}
}
void _resetAmount(ResetScanAmountEvent event, Emitter<TourState> emit) async {
opBloc.add(LoadOperation());
try {
await tourRepository.resetScan(event.articleId, event.deliveryId);
opBloc.add(FinishOperation());
} catch (e, st) {
debugPrint("Fehler beim Unscan des Artikels: ${event.articleId}:");
debugPrint("$e");
debugPrint("$st");
opBloc.add(FailOperation(message: "Fehler beim Zurücksetzen: $e"));
}
}
} }

View File

@ -1,5 +1,9 @@
import 'dart:typed_data';
import 'package:hl_lieferservice/model/tour.dart'; import 'package:hl_lieferservice/model/tour.dart';
import '../../../../model/delivery.dart';
abstract class TourEvent {} abstract class TourEvent {}
class LoadTour extends TourEvent { class LoadTour extends TourEvent {
@ -8,6 +12,26 @@ class LoadTour extends TourEvent {
LoadTour({required this.teamId}); LoadTour({required this.teamId});
} }
class RequestDeliveryDistanceEvent extends TourEvent {
Tour tour;
List<Payment> payments;
RequestDeliveryDistanceEvent({required this.tour, required this.payments});
}
class TourUpdated extends TourEvent {
Tour tour;
List<Payment> payments;
TourUpdated({required this.tour, required this.payments});
}
class PaymentOptionsUpdated extends TourEvent {
List<Payment> options;
PaymentOptionsUpdated({required this.options});
}
class UpdateTour extends TourEvent { class UpdateTour extends TourEvent {
Tour tour; Tour tour;
@ -24,8 +48,9 @@ class AssignCarEvent extends TourEvent {
class IncrementArticleScanAmount extends TourEvent { class IncrementArticleScanAmount extends TourEvent {
String internalArticleId; String internalArticleId;
String deliveryId; String deliveryId;
String carId;
IncrementArticleScanAmount({required this.internalArticleId, required this.deliveryId}); IncrementArticleScanAmount({required this.internalArticleId, required this.deliveryId, required this.carId});
} }
class ScanArticleEvent extends TourEvent { class ScanArticleEvent extends TourEvent {
@ -53,3 +78,87 @@ class ReactivateDeliveryEvent extends TourEvent {
ReactivateDeliveryEvent({required this.deliveryId}); ReactivateDeliveryEvent({required this.deliveryId});
} }
class LoadDeliveryEvent extends TourEvent {
LoadDeliveryEvent({required this.delivery});
Delivery delivery;
}
class UnscanArticleEvent extends TourEvent {
UnscanArticleEvent({
required this.articleId,
required this.newAmount,
required this.reason,
required this.deliveryId
});
String articleId;
String deliveryId;
String reason;
int newAmount;
}
class ResetScanAmountEvent extends TourEvent {
ResetScanAmountEvent({required this.articleId, required this.deliveryId});
String articleId;
String deliveryId;
}
class AddDiscountEvent extends TourEvent {
AddDiscountEvent({
required this.deliveryId,
required this.value,
required this.reason,
});
String deliveryId;
String reason;
int value;
}
class RemoveDiscountEvent extends TourEvent {
RemoveDiscountEvent({required this.deliveryId});
String deliveryId;
}
class UpdateDiscountEvent extends TourEvent {
UpdateDiscountEvent({
required this.deliveryId,
required this.value,
required this.reason,
});
String deliveryId;
String? reason;
int? value;
}
class UpdateDeliveryOptionEvent extends TourEvent {
UpdateDeliveryOptionEvent({required this.key, required this.value, required this.deliveryId});
String deliveryId;
String key;
dynamic value;
}
class UpdateSelectedPaymentMethodEvent extends TourEvent {
UpdateSelectedPaymentMethodEvent({required this.payment, required this.deliveryId});
Payment payment;
String deliveryId;
}
class FinishDeliveryEvent extends TourEvent {
FinishDeliveryEvent({
required this.deliveryId,
required this.driverSignature,
required this.customerSignature,
});
String deliveryId;
Uint8List customerSignature;
Uint8List driverSignature;
}

View File

@ -6,9 +6,33 @@ class TourInitial extends TourState {}
class TourLoading extends TourState {} class TourLoading extends TourState {}
class TourRequestingDistances extends TourState {
Tour tour;
List<Payment> payments;
TourRequestingDistances({required this.tour, required this.payments});
}
class TourLoaded extends TourState { class TourLoaded extends TourState {
Tour tour; Tour tour;
Map<String, double> distances; Map<String, double>? distances;
List<Payment> paymentOptions;
TourLoaded({required this.tour, required this.distances}); TourLoaded({
required this.tour,
this.distances,
required this.paymentOptions,
});
TourLoaded copyWith({
Tour? tour,
Map<String, double>? distances,
List<Payment>? paymentOptions,
}) {
return TourLoaded(
tour: tour ?? this.tour,
distances: distances ?? this.distances,
paymentOptions: paymentOptions ?? this.paymentOptions,
);
}
} }

View File

@ -1,12 +1,24 @@
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:flutter_bloc/flutter_bloc.dart';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
import 'package:hl_lieferservice/feature/delivery/detail/presentation/delivery_detail_page.dart'; import 'package:hl_lieferservice/feature/delivery/detail/presentation/delivery_detail_page.dart';
import '../../../../bloc/app_bloc.dart';
import '../../../../bloc/app_states.dart';
import '../../../../widget/operations/bloc/operation_bloc.dart';
import '../../detail/bloc/note_bloc.dart';
import '../../detail/repository/note_repository.dart';
import '../../detail/service/notes_service.dart';
class DeliveryListItem extends StatelessWidget { class DeliveryListItem extends StatelessWidget {
final Delivery delivery; final Delivery delivery;
final double distance; final double distance;
const DeliveryListItem({super.key, required this.delivery, required this.distance}); const DeliveryListItem({
super.key,
required this.delivery,
required this.distance,
});
Widget _leading(BuildContext context) { Widget _leading(BuildContext context) {
if (delivery.state == DeliveryState.finished) { if (delivery.state == DeliveryState.finished) {
@ -17,6 +29,10 @@ class DeliveryListItem extends StatelessWidget {
return Icon(Icons.cancel_rounded, color: Colors.red); return Icon(Icons.cancel_rounded, color: Colors.red);
} }
if (delivery.state == DeliveryState.onhold) {
return Icon(Icons.pause_circle, color: Colors.orange);
}
return Column( return Column(
mainAxisAlignment: MainAxisAlignment.spaceEvenly, mainAxisAlignment: MainAxisAlignment.spaceEvenly,
children: [ children: [
@ -29,7 +45,19 @@ class DeliveryListItem extends StatelessWidget {
void _goToDelivery(BuildContext context) { void _goToDelivery(BuildContext context) {
Navigator.of(context).push( Navigator.of(context).push(
MaterialPageRoute( MaterialPageRoute(
builder: (context) => DeliveryDetail(delivery: delivery), builder:
(context) => BlocProvider(
create:
(context) => NoteBloc(
deliveryId: delivery.id,
opBloc: context.read<OperationBloc>(),
repository: NoteRepository(
service: NoteService(),
),
),
child: DeliveryDetail(deliveryId: delivery.id),
),
), ),
); );
} }

View File

@ -32,7 +32,7 @@ class _DeliveryListState extends State<DeliveryList> {
Delivery delivery = widget.deliveries[index]; Delivery delivery = widget.deliveries[index];
return DeliveryListItem( return DeliveryListItem(
delivery: delivery, delivery: delivery,
distance: widget.distances[delivery.id]!, distance: widget.distances[delivery.id] ?? 0.0,
); );
}, },
itemCount: widget.deliveries.length, itemCount: widget.deliveries.length,

View File

@ -23,7 +23,7 @@ class _DeliveryOverviewPageState extends State<DeliveryOverviewPage> {
return Center( return Center(
child: DeliveryOverview( child: DeliveryOverview(
tour: currentState.tour, tour: currentState.tour,
distances: currentState.distances, distances: currentState.distances ?? {},
), ),
); );
} }

View File

@ -1,32 +1,351 @@
import 'dart:typed_data';
import 'package:hl_lieferservice/feature/delivery/overview/service/delivery_info_service.dart'; import 'package:hl_lieferservice/feature/delivery/overview/service/delivery_info_service.dart';
import 'package:hl_lieferservice/model/delivery.dart'; import 'package:hl_lieferservice/model/delivery.dart';
import 'package:hl_lieferservice/model/tour.dart'; import 'package:hl_lieferservice/model/tour.dart';
import 'package:rxdart/rxdart.dart';
import '../../../../dto/discount_add_response.dart';
import '../../../../dto/discount_remove_response.dart';
import '../../../../dto/discount_update_response.dart';
import '../../../../model/article.dart';
import '../../detail/repository/note_repository.dart';
import '../../detail/service/notes_service.dart';
enum ScanResult { scanned, alreadyScanned, notFound }
class TourNotFoundException implements Exception {}
class TourRepository { class TourRepository {
DeliveryInfoService service; TourService service;
final _tourStream = BehaviorSubject<Tour?>();
final _paymentOptionsStream = BehaviorSubject<List<Payment>>.seeded([]);
Stream<Tour?> get tour => _tourStream.stream;
Stream<List<Payment>> get paymentOptions => _paymentOptionsStream.stream;
TourRepository({required this.service}); TourRepository({required this.service});
Future<Tour> loadAll(String userId) async { Future<void> loadTourOfToday(String userId) async {
Tour? tour = await service.getTourOfToday(userId); _tourStream.add(await service.getTourOfToday(userId));
return tour!;
} }
Future<List<Payment>> loadPaymentOptions() async { Future<void> loadPaymentOptions() async {
return (await service.getPaymentMethods()) _paymentOptionsStream.add(
(await service.getPaymentMethods())
.map((option) => Payment.fromDTO(option)) .map((option) => Payment.fromDTO(option))
.toList(); .toList(),
);
} }
Future<void> assignCar(String deliveryId, String carId) async { Future<void> assignCar(String deliveryId, String carId) async {
await service.assignCar(deliveryId, carId); await service.assignCar(deliveryId, carId);
final tour = _tourStream.value!;
final index = tour.deliveries.indexWhere(
(delivery) => delivery.id == deliveryId,
);
tour.deliveries[index].carId = int.parse(carId);
_tourStream.add(tour);
} }
Future<void> scanArticle(String internalArticleId) async { Future<ScanResult> scanArticle(
return await service.scanArticle(internalArticleId); String deliveryId,
String carId,
String articleNumber,
) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
} }
Future<void> updateDelivery(Delivery delivery) { final tour = _tourStream.value!;
return service.updateDelivery(delivery);
if (tour.deliveries.any(
(delivery) => delivery.articles.any(
(article) => article.articleNumber == articleNumber,
),
)) {
var delivery = tour.deliveries.firstWhere(
(delivery) => delivery.id == deliveryId,
);
var article = delivery.articles.firstWhere(
(article) => article.articleNumber == articleNumber,
);
await service.scanArticle(article.internalId.toString());
if (article.scannedAmount < article.amount) {
article.scannedAmount += 1;
_tourStream.add(tour);
return ScanResult.scanned;
} else {
return ScanResult.alreadyScanned;
}
} else {
return ScanResult.notFound;
}
}
Future<void> unscan(
String deliveryId,
String articleId,
int newAmount,
String reason,
) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
String? noteId = await service.unscanArticle(articleId, newAmount, reason);
Article article = tour.deliveries
.firstWhere((delivery) => delivery.id == deliveryId)
.articles
.firstWhere((article) => article.internalId == int.parse(articleId));
article.removeNoteId = noteId;
article.scannedRemovedAmount += newAmount;
article.scannedAmount -= newAmount;
_tourStream.add(tour);
}
Future<void> resetScan(String articleId, String deliveryId) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
await service.resetScannedArticleAmount(articleId);
Article article = tour.deliveries
.firstWhere((delivery) => delivery.id == deliveryId)
.articles
.firstWhere((article) => article.internalId == int.parse(articleId));
article.removeNoteId = null;
article.scannedRemovedAmount = 0;
article.scannedAmount = article.amount;
_tourStream.add(tour);
}
Future<void> uploadDriverSignature(
String deliveryId,
Uint8List signature,
) async {
NoteRepository noteRepository = NoteRepository(service: NoteService());
await noteRepository.addNamedImage(
deliveryId,
signature,
"delivery_${deliveryId}_signature_driver.jpg",
);
}
Future<void> uploadCustomerSignature(
String deliveryId,
Uint8List signature,
) async {
NoteRepository noteRepository = NoteRepository(service: NoteService());
await noteRepository.addNamedImage(
deliveryId,
signature,
"delivery_${deliveryId}_signature_customer.jpg",
);
}
Future<void> addDiscount(String deliveryId, String reason, int value) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
DiscountAddResponseDTO response = await service.addDiscount(
deliveryId,
value,
reason,
);
Delivery delivery = tour.deliveries.firstWhere(
(delivery) => delivery.id == deliveryId,
);
delivery.totalNetValue = response.values.receipt.net;
delivery.totalGrossValue = response.values.receipt.gross;
delivery.discount = Discount(
article: Article.fromDTO(response.values.article),
note: response.values.note.noteDescription,
noteId: response.values.note.rowId,
);
_tourStream.add(tour);
}
Future<void> removeDiscount(String deliveryId) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
DiscountRemoveResponseDTO response = await service.removeDiscount(
deliveryId,
);
Delivery delivery = tour.deliveries.firstWhere(
(delivery) => delivery.id == deliveryId,
);
delivery.articles =
delivery.articles
.where(
(article) =>
article.internalId != delivery.discount?.article.internalId,
)
.toList();
delivery.discount = null;
delivery.totalGrossValue = response.receipt.gross;
delivery.totalNetValue = response.receipt.net;
_tourStream.add(tour);
}
Future<void> updateDiscount(
String deliveryId,
String? reason,
int? value,
) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
DiscountUpdateResponseDTO response = await service.updateDiscount(
deliveryId,
reason,
value,
);
Delivery delivery = tour.deliveries.firstWhere(
(delivery) => delivery.id == deliveryId,
);
if (response.values?.receipt != null) {
delivery.totalNetValue = response.values!.receipt.net;
delivery.totalGrossValue = response.values!.receipt.gross;
}
String discountArticleNumber = delivery.discount!.article.articleNumber;
delivery.discount = Discount(
article:
response.values?.article != null
? Article.fromDTO(response.values!.article)
: delivery.discount!.article,
note:
response.values?.note != null
? response.values!.note.noteDescription
: delivery.discount!.note,
noteId:
response.values?.note != null
? response.values!.note.rowId
: delivery.discount!.noteId,
);
delivery.articles = [
...delivery.articles.where(
(article) => article.articleNumber != discountArticleNumber,
),
delivery.discount!.article,
];
_tourStream.add(tour);
}
Future<void> reactivateDelivery(String deliveryId) async {
await _changeState(deliveryId, DeliveryState.ongoing);
}
Future<void> holdDelivery(String deliveryId) async {
await _changeState(deliveryId, DeliveryState.onhold);
}
Future<void> cancelDelivery(String deliveryId) async {
await _changeState(deliveryId, DeliveryState.canceled);
}
Future<void> finishDelivery(String deliveryId) async {
await _changeState(deliveryId, DeliveryState.finished);
}
Future<void> _changeState(String deliveryId, DeliveryState state) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
Delivery delivery = tour.deliveries.firstWhere(
(delivery) => delivery.id == deliveryId,
);
delivery.state = state;
await _updateDelivery(delivery);
_tourStream.add(tour);
}
Future<void> _updateDelivery(Delivery newDelivery) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
await service.updateDelivery(newDelivery);
}
Future<void> updatePayment(String deliveryId, Payment payment) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
Delivery delivery = tour.deliveries.firstWhere(
(delivery) => delivery.id == deliveryId,
);
delivery.payment = payment;
await _updateDelivery(delivery);
_tourStream.add(tour);
}
Future<void> updateOption(
String deliveryId,
String key,
dynamic value,
) async {
if (!_tourStream.hasValue) {
throw TourNotFoundException();
}
Tour tour = _tourStream.value!;
Delivery delivery = tour.deliveries.firstWhere(
(delivery) => delivery.id == deliveryId,
);
delivery.options =
delivery.options.map((option) {
if (option.key == key) {
if (option.numerical) {
return option.copyWith(value: value);
} else {
return option.copyWith(value: value == true ? "1" : "0");
}
}
return option;
}).toList();
await _updateDelivery(delivery);
_tourStream.add(tour);
} }
} }

View File

@ -20,8 +20,8 @@ import '../../../../dto/discount_update_response.dart';
import '../../../../dto/scan_response.dart'; import '../../../../dto/scan_response.dart';
import '../../../authentication/exceptions.dart'; import '../../../authentication/exceptions.dart';
class DeliveryInfoService { class TourService {
DeliveryInfoService(); TourService();
Future<void> updateDelivery(Delivery delivery) async { Future<void> updateDelivery(Delivery delivery) async {
try { try {
@ -29,6 +29,7 @@ class DeliveryInfoService {
headers.addAll(getSessionOrThrow()); headers.addAll(getSessionOrThrow());
debugPrint(getSessionOrThrow().toString()); debugPrint(getSessionOrThrow().toString());
debugPrint(delivery.state.toString());
debugPrint(jsonEncode(DeliveryUpdateDTO.fromEntity(delivery).toJson())); debugPrint(jsonEncode(DeliveryUpdateDTO.fromEntity(delivery).toJson()));
var response = await post( var response = await post(
@ -97,7 +98,7 @@ class DeliveryInfoService {
/// List all available deliveries for today. /// List all available deliveries for today.
Future<Tour?> getTourOfToday(String userId) async { Future<Tour> getTourOfToday(String userId) async {
try { try {
var response = await post( var response = await post(
urlBuilder("_web_getDeliveries"), urlBuilder("_web_getDeliveries"),

View File

@ -1,7 +1,10 @@
import 'dart:typed_data';
import 'package:flutter/cupertino.dart'; import 'package:flutter/cupertino.dart';
import 'package:hl_lieferservice/dto/contact_person.dart'; import 'package:hl_lieferservice/dto/contact_person.dart';
import 'package:hl_lieferservice/dto/delivery.dart'; import 'package:hl_lieferservice/dto/delivery.dart';
import 'package:hl_lieferservice/dto/image.dart'; import 'package:hl_lieferservice/dto/image.dart';
import 'package:hl_lieferservice/dto/image_note_response.dart';
import 'package:hl_lieferservice/dto/note.dart'; import 'package:hl_lieferservice/dto/note.dart';
import 'package:hl_lieferservice/dto/note_template.dart'; import 'package:hl_lieferservice/dto/note_template.dart';
import 'package:hl_lieferservice/model/tour.dart'; import 'package:hl_lieferservice/model/tour.dart';
@ -87,24 +90,26 @@ class Discount {
} }
class ImageNote { class ImageNote {
const ImageNote({ ImageNote({
required this.name, required this.name,
required this.url, required this.url,
required this.objectId, required this.objectId,
this.data
}); });
final String name; final String name;
final String url; final String url;
final String objectId; final String objectId;
Uint8List? data;
factory ImageNote.fromDTO(ImageDTO dto) { factory ImageNote.fromDTO(ImageNoteDTO dto) {
return ImageNote(name: dto.name, url: dto.url, objectId: dto.oid); return ImageNote(name: dto.name, url: dto.url, objectId: dto.oid);
} }
factory ImageNote.make(String objectId, String name) { factory ImageNote.make(String objectId, String name, Uint8List? bytes) {
String url = "/v1/preview/1920_1080_100_png/$objectId"; String url = "/v1/preview/1920_1080_100_png/$objectId";
return ImageNote(name: name, url: url, objectId: objectId); return ImageNote(name: name, url: url, objectId: objectId, data: bytes);
} }
} }

View File

@ -21,6 +21,18 @@ class Payment {
id: dto.id, id: dto.id,
); );
} }
Payment copyWith({
String? description,
String? shortcode,
String? id,
}) {
return Payment(
description: description ?? this.description,
shortcode: shortcode ?? this.shortcode,
id: id ?? this.id,
);
}
} }
class Tour { class Tour {
@ -30,7 +42,7 @@ class Tour {
required this.driver, required this.driver,
required this.discountArticleNumber, required this.discountArticleNumber,
required this.paymentMethods, required this.paymentMethods,
}); }) : deliveriesPerCar = {};
final DateTime date; final DateTime date;
final String discountArticleNumber; final String discountArticleNumber;
@ -38,6 +50,8 @@ class Tour {
final List<Delivery> deliveries; final List<Delivery> deliveries;
List<Payment> paymentMethods; List<Payment> paymentMethods;
Map<String, List<Delivery>> deliveriesPerCar;
int getFinishedDeliveries(int carId) { int getFinishedDeliveries(int carId) {
return deliveries return deliveries
.where((delivery) => delivery.carId == carId) .where((delivery) => delivery.carId == carId)
@ -54,8 +68,9 @@ class Tour {
List<Payment>? paymentMethods, List<Payment>? paymentMethods,
}) { }) {
return Tour( return Tour(
date: date ?? this.date, date: date ?? this.date.copyWith(),
discountArticleNumber: discountArticleNumber ?? this.discountArticleNumber, discountArticleNumber:
discountArticleNumber ?? this.discountArticleNumber,
driver: driver ?? this.driver, driver: driver ?? this.driver,
deliveries: deliveries ?? this.deliveries, deliveries: deliveries ?? this.deliveries,
paymentMethods: paymentMethods ?? this.paymentMethods, paymentMethods: paymentMethods ?? this.paymentMethods,
@ -84,4 +99,18 @@ class Driver {
return "$salutation, $name"; return "$salutation, $name";
} }
Driver copyWith(
int? teamNumber,
String? name,
String? salutation,
List<Car>? cars,
) {
return Driver(
teamNumber: teamNumber ?? this.teamNumber,
name: name ?? this.name,
salutation: salutation ?? this.salutation,
cars: cars ?? this.cars,
);
}
} }

View File

@ -1,28 +0,0 @@
import 'dart:typed_data';
import 'package:hl_lieferservice/model/delivery.dart';
import 'package:hl_lieferservice/feature/delivery/detail/service/notes_service.dart';
class NoteRepository {
final NoteService service;
NoteRepository({required this.service});
Future<Note> addNote(String deliveryId, String content) async {
return service.addNote(content, int.parse(deliveryId));
}
Future<void> deleteNote(String noteId) async {
return service.deleteNote(int.parse(noteId));
}
Future<List<Note>> loadNotes(String deliveryId) async {
return [];
}
Future<List<Uint8List>> loadImages(String deliveryId) async {
return [];
}
Future<void> addImage(String deliveryId, Uint8List imageBytes) async {}
Future<void> deleteImage(String deliveryId, String imageIdentifier) async {}
}

View File

@ -5,14 +5,8 @@ import 'package:hl_lieferservice/feature/authentication/bloc/auth_bloc.dart';
import 'package:hl_lieferservice/feature/authentication/presentation/login_enforcer.dart'; import 'package:hl_lieferservice/feature/authentication/presentation/login_enforcer.dart';
import 'package:hl_lieferservice/feature/authentication/service/userinfo.dart'; import 'package:hl_lieferservice/feature/authentication/service/userinfo.dart';
import 'package:hl_lieferservice/feature/cars/presentation/car_management_page.dart'; import 'package:hl_lieferservice/feature/cars/presentation/car_management_page.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/delivery_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/bloc/note_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/detail/repository/delivery_repository.dart';
import 'package:hl_lieferservice/feature/delivery/detail/repository/note_repository.dart';
import 'package:hl_lieferservice/feature/delivery/detail/service/notes_service.dart';
import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart'; import 'package:hl_lieferservice/feature/delivery/overview/bloc/tour_bloc.dart';
import 'package:hl_lieferservice/feature/delivery/overview/repository/tour_repository.dart'; import 'package:hl_lieferservice/feature/delivery/overview/repository/tour_repository.dart';
import 'package:hl_lieferservice/feature/settings/bloc/settings_bloc.dart';
import 'package:hl_lieferservice/widget/home/bloc/navigation_bloc.dart'; import 'package:hl_lieferservice/widget/home/bloc/navigation_bloc.dart';
import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart'; import 'package:hl_lieferservice/widget/operations/bloc/operation_bloc.dart';
import 'package:hl_lieferservice/widget/operations/presentation/operation_view_enforcer.dart'; import 'package:hl_lieferservice/widget/operations/presentation/operation_view_enforcer.dart';
@ -53,33 +47,7 @@ class _DeliveryAppState extends State<DeliveryApp> {
(context) => TourBloc( (context) => TourBloc(
opBloc: context.read<OperationBloc>(), opBloc: context.read<OperationBloc>(),
tourRepository: TourRepository( tourRepository: TourRepository(
service: DeliveryInfoService(), service: TourService(),
),
),
),
BlocProvider(
create:
(context) => NoteBloc(
opBloc: context.read<OperationBloc>(),
repository: NoteRepository(
service: NoteService(
backendUrl: currentAppState.config.backendUrl,
),
),
),
),
BlocProvider(
create:
(context) => DeliveryBloc(
noteBloc: context.read<NoteBloc>(),
opBloc: context.read<OperationBloc>(),
noteRepository: NoteRepository(
service: NoteService(
backendUrl: currentAppState.config.backendUrl,
),
),
repository: DeliveryRepository(
service: DeliveryInfoService(),
), ),
), ),
), ),

View File

@ -864,6 +864,14 @@ packages:
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.5.0" version: "1.5.0"
rxdart:
dependency: "direct main"
description:
name: rxdart
sha256: "5c3004a4a8dbb94bd4bf5412a4def4acdaa12e12f269737a5751369e12d1a962"
url: "https://pub.dev"
source: hosted
version: "0.28.0"
shared_preferences: shared_preferences:
dependency: "direct main" dependency: "direct main"
description: description:

View File

@ -51,6 +51,7 @@ dependencies:
geolocator: ^14.0.2 geolocator: ^14.0.2
mobile_scanner: ^7.1.3 mobile_scanner: ^7.1.3
shared_preferences: ^2.5.3 shared_preferences: ^2.5.3
rxdart: ^0.28.0
dev_dependencies: dev_dependencies:
build_runner: ^2.5.4 build_runner: ^2.5.4