feat: complete sell

This commit is contained in:
alzalia1 2025-08-15 14:48:27 +02:00
parent b4375fbaa2
commit bee5e05296
5 changed files with 144 additions and 38 deletions

View file

@ -23,4 +23,12 @@ class BookInstanceRepository {
) async { ) async {
return await _apiClient.sendBook(book, owner, bal, price); return await _apiClient.sendBook(book, owner, bal, price);
} }
Future<Result<void>> sellBooks(List<BookInstance> books) async {
Map<String, double?> res = {};
for (BookInstance instance in books) {
res[instance.id.toString()] = instance.soldPrice;
}
return await _apiClient.sellBooks(res);
}
} }

View file

@ -310,6 +310,31 @@ class ApiClient {
} }
} }
Future<Result<void>> sellBooks(Map<String, double?> books) async {
final client = Client();
try {
final headers = await _getHeaders({"Content-Type": "application/json"});
debugPrint("\n\n\n\nMAP: $books\n\n\n\n");
final body = jsonEncode(books);
debugPrint("\n\n\n\nSENT: $body\n\n\n\n");
final response = await client.post(
Uri.parse("https://$apiBasePath/book_instance/sell/bulk"),
headers: headers,
body: body,
);
if (response.statusCode == 200) {
return Result.ok(response.statusCode);
} else {
throw "Unknown error";
}
} catch (e) {
debugPrint("\n\n\n\nERROR : ${e.toString()}\n\n\n\n");
return Result.error(Exception(e));
} finally {
client.close();
}
}
Future<Result<BookInstance>> sendBook( Future<Result<BookInstance>> sendBook(
Book book, Book book,
Owner owner, Owner owner,

View file

@ -50,14 +50,41 @@ class SellViewModel extends ChangeNotifier {
List<BookStack> get scannedBooks => _scannedBooks; List<BookStack> get scannedBooks => _scannedBooks;
bool isScanLoaded = false; bool isScanLoaded = false;
bool isSendingSell = false;
double minimumAmount = 0;
void sellBook(BookStack addedBook) { void sellBook(BookStack addedBook) {
minimumAmount += addedBook.instance.price;
_soldBooks.add(addedBook); _soldBooks.add(addedBook);
notifyListeners(); notifyListeners();
} }
void sendSell() { void sendSell(double givenAmount) async {
isSendingSell = true;
notifyListeners();
List<BookInstance> toSend = [];
int nbOfPl = 0;
for (BookStack book in _soldBooks) {
if (book.instance.price != 0) {
book.instance.soldPrice = book.instance.price;
givenAmount -= book.instance.price;
toSend.add(book.instance);
} else {
nbOfPl++;
}
}
if (nbOfPl != 0) {
double amountPerPl = givenAmount / nbOfPl;
for (BookStack book in _soldBooks) {
if (book.instance.price == 0) {
book.instance.soldPrice = amountPerPl;
toSend.add(book.instance);
}
}
}
await _bookInstanceRepository.sellBooks(toSend);
_soldBooks.clear(); _soldBooks.clear();
isSendingSell = false;
notifyListeners(); notifyListeners();
} }
@ -75,6 +102,9 @@ class SellViewModel extends ChangeNotifier {
switch (result) { switch (result) {
case Ok(): case Ok():
for (BookInstance instance in result.value) { for (BookInstance instance in result.value) {
if (instance.available == false) {
continue;
}
if (_soldBooks if (_soldBooks
.where((book) => book.instance.id == instance.id) .where((book) => book.instance.id == instance.id)
.isNotEmpty) { .isNotEmpty) {

View file

@ -25,7 +25,7 @@ class SellChoicePopup extends StatelessWidget {
children: [ children: [
(viewModel.scannedBooks.isEmpty) (viewModel.scannedBooks.isEmpty)
? Text( ? Text(
"Ce livre n'a jamais été rentré, ou vous l'avez déjà mis dans cette vente.", "Ce livre n'a jamais été rentré, il a déjà été vendu ou vous l'avez déjà mis dans cette vente.",
) )
: SizedBox(), : SizedBox(),
for (BookStack book in viewModel.scannedBooks) for (BookStack book in viewModel.scannedBooks)

View file

@ -17,6 +17,8 @@ class SellPage extends StatefulWidget {
} }
class _SellPageState extends State<SellPage> { class _SellPageState extends State<SellPage> {
TextEditingController price = TextEditingController();
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
return Scaffold( return Scaffold(
@ -70,52 +72,60 @@ class _SellPageState extends State<SellPage> {
), ),
SizedBox(height: 15), SizedBox(height: 15),
Expanded( Expanded(
child: ListView( child: (widget.viewModel.isSendingSell)
children: [ ? Center(child: CircularProgressIndicator())
(widget.viewModel.scannedBooks.isEmpty) : ListView(
? Center(child: Text("Aucun")) children: [
: SizedBox(), (widget.viewModel.scannedBooks.isEmpty)
for (BookStack book in widget.viewModel.soldBooks) ? Center(child: Text("Aucun"))
Padding( : SizedBox(),
padding: const EdgeInsets.symmetric( for (BookStack book
horizontal: 15, in widget.viewModel.soldBooks)
), Padding(
child: Card( padding: const EdgeInsets.symmetric(
child: ListTile( horizontal: 15,
leading: Text( ),
"${book.instance.price.toString()}", child: Card(
style: TextStyle(fontSize: 30), child: ListTile(
leading: Text(
"${book.instance.price.toString()}",
style: TextStyle(fontSize: 30),
),
title: Text(
"${book.book.title} · ${book.book.author}",
),
subtitle: Text(
"${book.owner.firstName} ${book.owner.lastName} (${book.shortId()})",
),
trailing: IconButton(
onPressed: () {
widget.viewModel.deleteBook(
book.instance.id,
);
},
icon: Icon(Icons.delete),
),
),
),
), ),
title: Text( ],
"${book.book.title} · ${book.book.author}",
),
subtitle: Text(
"${book.owner.firstName} ${book.owner.lastName} (${book.shortId()})",
),
trailing: IconButton(
onPressed: () {
widget.viewModel.deleteBook(
book.instance.id,
);
},
icon: Icon(Icons.delete),
),
),
),
), ),
],
),
), ),
SizedBox(height: 40), SizedBox(height: 40),
Text("Montant minimum à payer : 20€"), Text(
"Montant minimum à payer : ${widget.viewModel.minimumAmount.toString()}",
),
Padding( Padding(
padding: const EdgeInsets.symmetric(horizontal: 60.0), padding: const EdgeInsets.symmetric(horizontal: 60.0),
child: SizedBox( child: SizedBox(
child: TextField( child: TextField(
controller: price,
decoration: InputDecoration( decoration: InputDecoration(
labelText: "Argent reçu", labelText: "Argent reçu",
hintText:
"Utilisez un point (.) pour les virgules (,)",
helperText: helperText:
"L'argent reçu sera réparti automatiquement", "L'argent reçu sera réparti automatiquement.",
suffixText: "", suffixText: "",
border: OutlineInputBorder(), border: OutlineInputBorder(),
), ),
@ -129,7 +139,40 @@ class _SellPageState extends State<SellPage> {
children: [ children: [
IconButton( IconButton(
onPressed: () { onPressed: () {
widget.viewModel.sendSell(); if (double.tryParse(price.text) == null) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
"Veuillez indiquer un prix de vente valide",
),
behavior: SnackBarBehavior.floating,
),
);
return;
} else if (double.parse(price.text) <
widget.viewModel.minimumAmount) {
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
"Le prix de vente est inférieur au montant minimum",
),
behavior: SnackBarBehavior.floating,
),
);
return;
}
widget.viewModel.sendSell(
double.parse(price.text),
);
ScaffoldMessenger.of(context).showSnackBar(
SnackBar(
content: Text(
"La vente a bien été enregistrée.",
),
behavior: SnackBarBehavior.floating,
),
);
return;
}, },
icon: Icon(Icons.check), icon: Icon(Icons.check),
style: ButtonStyle( style: ButtonStyle(