Add expense editor
This commit is contained in:
@ -1,15 +1,10 @@
|
||||
import 'dart:io';
|
||||
import 'dart:typed_data';
|
||||
|
||||
import 'package:alert_dialog/alert_dialog.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_hooks/flutter_hooks.dart';
|
||||
import 'package:flutter_pdfview/flutter_pdfview.dart';
|
||||
import 'package:hooks_riverpod/hooks_riverpod.dart';
|
||||
import 'package:logging/logging.dart';
|
||||
import 'package:moneymgr_mobile/services/storage/expenses.dart';
|
||||
import 'package:moneymgr_mobile/services/storage/prefs.dart';
|
||||
import 'package:moneymgr_mobile/utils/time_utils.dart';
|
||||
import 'package:moneymgr_mobile/widgets/expense_editor.dart';
|
||||
import 'package:riverpod_annotation/riverpod_annotation.dart';
|
||||
import 'package:scanbot_sdk/scanbot_sdk.dart';
|
||||
import 'package:scanbot_sdk/scanbot_sdk_ui_v2.dart' hide IconButton, EdgeInsets;
|
||||
@ -68,6 +63,7 @@ class ScanScreen extends HookConsumerWidget {
|
||||
AsyncData(:final value) when value != null => ExpenseEditor(
|
||||
file: value,
|
||||
onFinished: (e) {},
|
||||
onRescan: restartScan,
|
||||
),
|
||||
|
||||
// No data
|
||||
@ -119,91 +115,3 @@ class ScanErrorScreen extends StatelessWidget {
|
||||
);
|
||||
}
|
||||
}
|
||||
|
||||
class ExpenseEditor extends HookConsumerWidget {
|
||||
final Uint8List file;
|
||||
final Function(Expense) onFinished;
|
||||
|
||||
const ExpenseEditor({
|
||||
super.key,
|
||||
required this.file,
|
||||
required this.onFinished,
|
||||
});
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context, WidgetRef ref) {
|
||||
final serverConfig = ref.watch(prefsProvider).requireValue.serverConfig()!;
|
||||
|
||||
final labelController = useTextEditingController();
|
||||
final costController = useTextEditingController();
|
||||
final timeController = useState(DateTime.now());
|
||||
|
||||
// Pick a new date
|
||||
handlePickDate() async {
|
||||
final date = await showDatePicker(
|
||||
context: context,
|
||||
firstDate: DateTime(2000),
|
||||
lastDate: DateTime(2099),
|
||||
initialDate: timeController.value,
|
||||
);
|
||||
if (date != null) timeController.value = date;
|
||||
}
|
||||
|
||||
return ListView(
|
||||
children: [
|
||||
// Expense preview
|
||||
SizedBox(
|
||||
height: 200,
|
||||
child: PDFView(
|
||||
pdfData: file,
|
||||
onError: (e) {
|
||||
Logger.root.warning("Failed to render PDF $e");
|
||||
alert(context, content: Text("Failed to render PDF $e"));
|
||||
},
|
||||
fitPolicy: FitPolicy.BOTH,
|
||||
),
|
||||
),
|
||||
|
||||
SizedBox(height: 10),
|
||||
|
||||
// Cost
|
||||
TextField(
|
||||
controller: costController,
|
||||
keyboardType: TextInputType.number,
|
||||
decoration: const InputDecoration(labelText: 'Cost'),
|
||||
textInputAction: TextInputAction.done,
|
||||
),
|
||||
|
||||
SizedBox(height: 10),
|
||||
|
||||
// Label
|
||||
TextField(
|
||||
controller: labelController,
|
||||
keyboardType: TextInputType.text,
|
||||
decoration: const InputDecoration(labelText: 'Label'),
|
||||
textInputAction: TextInputAction.done,
|
||||
maxLength: serverConfig.constraints.inbox_entry_label.max,
|
||||
),
|
||||
|
||||
SizedBox(height: 10),
|
||||
|
||||
// Date
|
||||
TextField(
|
||||
enabled: true,
|
||||
readOnly: true,
|
||||
controller: TextEditingController(
|
||||
text: timeController.value.simpleDate,
|
||||
),
|
||||
keyboardType: TextInputType.datetime,
|
||||
decoration: InputDecoration(
|
||||
labelText: 'Date',
|
||||
suffixIcon: IconButton(
|
||||
onPressed: handlePickDate,
|
||||
icon: const Icon(Icons.date_range),
|
||||
),
|
||||
),
|
||||
),
|
||||
],
|
||||
);
|
||||
}
|
||||
}
|
||||
|
Reference in New Issue
Block a user