cdb_ui/lib/pages/transaction.dart

335 lines
11 KiB
Dart
Raw Normal View History

2024-09-20 07:24:22 +00:00
import 'package:cdb_ui/api.dart';
2024-09-21 15:15:12 +00:00
import 'package:cdb_ui/pages/consume.dart';
2024-09-20 07:24:22 +00:00
import 'package:flutter/material.dart';
2024-09-21 15:15:12 +00:00
import 'package:intl/intl.dart';
2024-09-23 06:55:40 +00:00
import 'package:qr_bar_code/qr/qr.dart';
2024-09-24 14:19:15 +00:00
import 'package:qr_bar_code_scanner_dialog/qr_bar_code_scanner_dialog.dart';
2024-09-20 07:24:22 +00:00
2024-09-24 14:19:15 +00:00
class TransactionPage extends StatefulWidget {
2024-09-23 06:55:40 +00:00
final Transaction transaction;
2024-09-23 07:03:40 +00:00
final Function? refresh;
2024-09-20 07:24:22 +00:00
2024-09-23 07:03:40 +00:00
const TransactionPage(this.transaction, {this.refresh, super.key});
2024-09-20 07:24:22 +00:00
2024-09-24 14:19:15 +00:00
@override
State<TransactionPage> createState() => _TransactionPageState();
}
class _TransactionPageState extends State<TransactionPage> {
2024-09-20 07:24:22 +00:00
@override
Widget build(BuildContext context) {
return Scaffold(
2024-09-24 12:55:14 +00:00
appBar: AppBar(
2024-09-24 14:19:15 +00:00
title: Text(widget.transaction.item),
2024-09-24 12:55:14 +00:00
actions: [
IconButton(
2024-09-25 08:49:21 +00:00
onPressed: () {
API().getLocations().then((locations) {
List<String> locationList = locations.keys.toList();
String? selectedLocationID;
showDialog(
context: context,
builder: (BuildContext context) {
return AlertDialog(
title: const Text('Move Transaction'),
content: Row(
mainAxisSize: MainAxisSize.min,
children: [
DropdownButton<String>(
value: selectedLocationID,
onChanged: (value) {
selectedLocationID = value!;
API()
.moveTransaction(widget.transaction.uuid,
selectedLocationID!)
.then((x) {
Navigator.of(context).pop();
2024-09-24 14:19:15 +00:00
});
2024-09-25 08:49:21 +00:00
2024-09-24 14:19:15 +00:00
setState(() {});
2024-09-24 12:55:14 +00:00
},
2024-09-25 08:49:21 +00:00
items: locationList
.map<DropdownMenuItem<String>>((locationID) {
return DropdownMenuItem<String>(
value: locationID,
child: Text(locations[locationID]!.name),
);
}).toList(),
),
IconButton(
onPressed: () {
API().getLocations().then((locations) {
QrBarCodeScannerDialog()
.getScannedQrBarCode(
context: context,
onCode: (code) {
// library is retarded
code = code!.replaceFirst(
"Code scanned = ", "");
if (!locations
.containsKey(code)) {
ScaffoldMessenger.of(context)
.showSnackBar(
SnackBar(
content: Text(
'The location $code does not exist.')),
);
return;
}
API()
.moveTransaction(
widget.transaction.uuid,
selectedLocationID!)
.then(
(x) {
Navigator.of(context).pop();
},
);
});
});
setState(() {});
},
icon: const Icon(Icons.qr_code))
],
),
);
},
);
});
2024-09-24 12:55:14 +00:00
},
icon: const Icon(Icons.move_up))
],
),
2024-09-20 07:24:22 +00:00
body: Column(
children: [
2024-09-24 14:19:15 +00:00
Text("UUID: ${widget.transaction.uuid}"),
2024-09-23 06:55:40 +00:00
QRCode(
2024-09-24 14:19:15 +00:00
data: widget.transaction.uuid,
2024-09-23 06:55:40 +00:00
size: 22,
semanticsLabel: "Transaction UUID",
),
2024-09-23 18:19:30 +00:00
// todo : human names
2024-09-24 14:19:15 +00:00
Text("${widget.transaction.item} - ${widget.transaction.variant}"),
2024-09-23 06:55:40 +00:00
2024-09-24 14:19:15 +00:00
Text("Added: ${tsFormat(widget.transaction.timestamp)}"),
2024-09-23 06:55:40 +00:00
2024-09-24 14:19:15 +00:00
if (widget.transaction.expired) const Text("Transaction is Expired!"),
2024-09-23 06:55:40 +00:00
2024-09-24 14:19:15 +00:00
IconText(Icons.money, widget.transaction.price.format(),
2024-09-23 06:55:40 +00:00
color: Colors.green),
2024-09-24 14:19:15 +00:00
if (widget.transaction.origin != null)
IconText(Icons.store, widget.transaction.origin!,
color: Colors.blue),
2024-09-23 06:55:40 +00:00
2024-09-24 14:19:15 +00:00
if (widget.transaction.location != null)
IconText(Icons.location_city, widget.transaction.location!.name),
2024-09-23 06:55:40 +00:00
2024-09-24 14:19:15 +00:00
if (widget.transaction.note != null) Text(widget.transaction.note!),
2024-09-23 06:55:40 +00:00
2024-09-24 14:19:15 +00:00
if (widget.transaction.consumed != null) ...[
2024-09-23 06:55:40 +00:00
const Divider(),
2024-09-24 14:19:15 +00:00
Text(
"Consumed on: ${tsFormat(widget.transaction.consumed!.timestamp)}"),
IconText(Icons.store, widget.transaction.consumed!.destination,
2024-09-23 06:55:40 +00:00
color: Colors.blue),
2024-09-24 14:19:15 +00:00
IconText(Icons.money, widget.transaction.consumed!.price.format(),
2024-09-23 06:55:40 +00:00
color: Colors.green),
]
2024-09-20 07:24:22 +00:00
// todo : chart with price history
],
),
2024-09-23 07:03:40 +00:00
floatingActionButton: FloatingActionButton(
onPressed: () {
Navigator.of(context).pushReplacement(MaterialPageRoute(
builder: (context) {
2024-09-24 14:19:15 +00:00
return ConsumePage(widget.transaction, widget.refresh ?? () {});
2024-09-23 07:03:40 +00:00
},
));
},
child: const Icon(Icons.receipt_long)),
2024-09-20 07:24:22 +00:00
);
}
}
2024-09-21 15:15:12 +00:00
class TransactionCard extends StatelessWidget {
final Transaction t;
final Function refresh;
2024-09-23 14:34:39 +00:00
final Function(Transaction)? onTap;
final Function(Transaction)? onLongPress;
2024-09-21 15:15:12 +00:00
2024-09-23 14:34:39 +00:00
const TransactionCard(this.t, this.refresh,
{this.onTap, this.onLongPress, super.key});
2024-09-21 15:15:12 +00:00
@override
Widget build(BuildContext context) {
return InkWell(
onTap: () {
2024-09-23 14:34:39 +00:00
if (onTap != null) {
onTap!(t);
return;
}
2024-09-21 15:15:12 +00:00
Navigator.of(context).push(MaterialPageRoute(
builder: (context) {
2024-09-23 07:03:40 +00:00
return ConsumePage(t, refresh);
2024-09-21 15:15:12 +00:00
},
));
},
2024-09-22 20:15:43 +00:00
onLongPress: () {
2024-09-23 14:34:39 +00:00
if (onLongPress != null) {
onLongPress!(t);
return;
}
2024-09-22 20:15:43 +00:00
Navigator.of(context).push(MaterialPageRoute(
builder: (context) {
return TransactionPage(t);
},
));
},
2024-09-21 15:15:12 +00:00
child: Card(
color: t.expired ? Colors.red[100] : Colors.black,
margin: const EdgeInsets.symmetric(vertical: 8, horizontal: 16),
shape: RoundedRectangleBorder(
borderRadius: BorderRadius.circular(10),
),
elevation: 4,
child: Padding(
padding: const EdgeInsets.all(12),
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
Row(
children: [
Text(
t.item,
style: const TextStyle(fontSize: 16),
),
const SizedBox(
width: 4,
),
Text(
t.variant,
style: TextStyle(fontSize: 14, color: Colors.grey[400]),
),
],
),
2024-09-25 16:14:42 +00:00
if (t.timestamp != 0)
Text(
tsFormat(t.timestamp),
style: TextStyle(fontSize: 14, color: Colors.grey[700]),
),
2024-09-21 15:15:12 +00:00
],
),
if ((t.note ?? "").isNotEmpty) ...[
const SizedBox(
2024-09-23 18:19:30 +00:00
height: 4,
),
Text(t.note!)
2024-09-21 15:15:12 +00:00
],
2024-09-23 18:19:30 +00:00
const SizedBox(
height: 10,
),
2024-09-21 15:15:12 +00:00
IconText(Icons.money,
"${t.price.value.toStringAsFixed(2)} ${t.price.currency}",
color: Colors.green),
if (t.origin != null) ...[
const SizedBox(height: 8),
IconText(Icons.store, t.origin!, color: Colors.blue)
],
2024-09-23 06:42:58 +00:00
if (t.location != null) ...[
const SizedBox(
height: 8,
),
2024-09-23 18:19:30 +00:00
IconText(Icons.location_city, t.location!.name)
2024-09-23 06:42:58 +00:00
]
2024-09-21 15:15:12 +00:00
],
),
),
),
);
}
}
String tsFormat(int ts) {
DateTime dateTime = DateTime.fromMillisecondsSinceEpoch(ts * 1000);
return DateFormat('yyyy-MM-dd HH:mm:ss').format(dateTime);
}
class IconText extends StatelessWidget {
final IconData icon;
final String text;
final Color? color;
const IconText(this.icon, this.text, {super.key, this.color});
@override
Widget build(BuildContext context) {
return Row(
children: [
Icon(icon, size: 18, color: color),
const SizedBox(width: 6),
Text(
text,
style: const TextStyle(fontSize: 16, fontWeight: FontWeight.bold),
),
],
);
}
}
2024-09-24 16:00:13 +00:00
class TransactionSelectPage extends StatelessWidget {
final Function(Transaction) onSelect;
final List<Transaction> selections;
final List<Transaction>? exclude;
const TransactionSelectPage(this.selections,
{super.key, required this.onSelect, this.exclude});
@override
Widget build(BuildContext context) {
var selectionList = [];
for (var s in selections) {
if (exclude?.any((x) => x.uuid == s.uuid) ?? false) {
continue;
}
selectionList.add(s);
}
2024-09-24 16:14:51 +00:00
if (selectionList.isEmpty) {
Navigator.of(context).pop();
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(content: Text('No Transactions to select')),
);
}
2024-09-24 16:00:13 +00:00
return Scaffold(
appBar: AppBar(
title: const Text("Select a Transaction"),
),
body: ListView(
children: selectionList
.map((x) => TransactionCard(
x,
() {},
onLongPress: (x) {},
onTap: (t) {
onSelect(t);
Navigator.of(context).pop();
},
))
.toList()),
);
}
2024-09-25 08:49:21 +00:00
}