Compare commits

..

4 commits

Author SHA1 Message Date
19c4243273 feat: manual register 2025-08-05 12:21:49 +02:00
cbf43f6d00 update readme 2025-08-05 10:44:21 +02:00
8af775a1d3 Owner name is correctly displayed 2025-08-05 10:43:30 +02:00
4d9a9ecab1 Delete useless scan_screen 2025-08-05 10:37:04 +02:00
6 changed files with 298 additions and 101 deletions

View file

@ -1,16 +1,3 @@
# seshat
A new Flutter project.
## Getting Started
This project is a starting point for a Flutter application.
A few resources to get you started if this is your first Flutter project:
- [Lab: Write your first Flutter app](https://docs.flutter.dev/get-started/codelab)
- [Cookbook: Useful Flutter samples](https://docs.flutter.dev/cookbook)
For help getting started with Flutter development, view the
[online documentation](https://docs.flutter.dev/), which offers tutorials,
samples, guidance on mobile development, and a full API reference.
Client android/iOS/web, écrit en dart x flutter, pour Alexandria.

View file

@ -2,6 +2,7 @@ import 'package:flutter/material.dart';
import 'package:flutter_svg/svg.dart';
import 'package:mobile_scanner/mobile_scanner.dart';
import 'package:seshat/ui/add_page/view_model/add_view_model.dart';
import 'package:seshat/ui/add_page/widgets/form_popup.dart';
import 'package:seshat/ui/add_page/widgets/owner_popup.dart';
import 'package:seshat/ui/core/ui/navigation_bar.dart';
@ -22,6 +23,8 @@ class _AddPageState extends State<AddPage> {
formats: [BarcodeFormat.ean13],
detectionTimeoutMs: 1000,
);
final theme = Theme.of(context);
// return Consumer<TabScreen>(
// builder: (context, screen, child) {
return Scaffold(
@ -44,10 +47,16 @@ class _AddPageState extends State<AddPage> {
margin: EdgeInsets.symmetric(horizontal: 50),
child: Column(
children: [
ListTile(
ListenableBuilder(
listenable: widget.viewModel,
builder: (context, child) => ListTile(
leading: Icon(Icons.person),
title: TextButton(
child: Text("No"),
child: Text(
(widget.viewModel.currentOwner == null)
? "Aucun"
: "${widget.viewModel.currentOwner!.firstName} ${widget.viewModel.currentOwner!.lastName}",
),
onPressed: () => _ownerDialogBuilder(
context,
controller,
@ -55,6 +64,7 @@ class _AddPageState extends State<AddPage> {
),
),
),
),
ListTile(
leading: Icon(Icons.attach_money),
title: TextButton(
@ -78,7 +88,11 @@ class _AddPageState extends State<AddPage> {
SvgPicture.asset('assets/scan-overlay.svg'),
Expanded(child: SizedBox()),
TextButton(
onPressed: () {},
style: ButtonStyle(
backgroundColor: WidgetStatePropertyAll(theme.cardColor),
),
onPressed: () =>
_formDialogBuilder(context, controller, widget.viewModel),
child: Text("Enregistrer manuellement"),
),
],
@ -99,6 +113,25 @@ void onBarcodeScan(
return;
}
Future<void> _formDialogBuilder(
BuildContext context,
MobileScannerController controller,
AddViewModel viewModel,
) {
controller.stop();
void exitPopup(BuildContext localContext) {
Navigator.of(localContext).pop();
controller.start();
}
return showDialog(
context: context,
barrierDismissible: false,
builder: (context) => FormPopup(viewModel: viewModel, exitPopup: exitPopup),
);
}
Future<void> _ownerDialogBuilder(
BuildContext context,
MobileScannerController controller,
@ -107,8 +140,8 @@ Future<void> _ownerDialogBuilder(
controller.stop();
void onPressAccept(BuildContext localContext) {
controller.start();
Navigator.of(localContext).pop();
controller.start();
}
return showDialog(

View file

@ -0,0 +1,252 @@
import 'package:flutter/material.dart';
import 'package:seshat/ui/add_page/view_model/add_view_model.dart';
class FormPopup extends StatelessWidget {
const FormPopup({
super.key,
required this.viewModel,
required this.exitPopup,
});
final AddViewModel viewModel;
final Function(BuildContext) exitPopup;
@override
Widget build(BuildContext context) {
return AlertDialog(
title: Center(child: Text("Type d'entrée")),
content: Column(
mainAxisSize: MainAxisSize.min,
children: [
Card(
clipBehavior: Clip.hardEdge,
child: InkWell(
splashColor: Colors.blue.withAlpha(30),
onTap: () {
debugPrint('Card tapped.');
Navigator.of(context).pop();
showDialog(
context: context,
barrierDismissible: false,
builder: (context) {
return _ManualEANPopup(exitPopup: exitPopup);
},
);
},
child: const SizedBox(
width: 300,
height: 100,
child: Center(
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(Icons.onetwothree),
SizedBox(width: 10),
Text('Entrer manuellement un EAN'),
],
),
),
),
),
),
Card(
clipBehavior: Clip.hardEdge,
child: InkWell(
splashColor: Colors.blue.withAlpha(30),
onTap: () {
debugPrint('Card tapped.');
Navigator.of(context).pop();
showDialog(
context: context,
barrierDismissible: false,
builder: (context) {
return _FullyManual(exitPopup: exitPopup);
},
);
},
child: const SizedBox(
width: 300,
height: 100,
child: Center(
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(Icons.history_edu),
SizedBox(width: 10),
Text('Entrer manuellement un livre'),
],
),
),
),
),
),
],
),
actions: [
TextButton(onPressed: () => exitPopup(context), child: Text("Annuler")),
],
);
}
}
class _ManualEANPopup extends StatefulWidget {
const _ManualEANPopup({required this.exitPopup});
final Function(BuildContext) exitPopup;
@override
State<_ManualEANPopup> createState() => _ManualEANPopupState();
}
class _ManualEANPopupState extends State<_ManualEANPopup> {
final GlobalKey<FormState> _formKey = GlobalKey<FormState>();
String? ean;
@override
Widget build(BuildContext context) {
return AlertDialog(
title: Text("Recherche par EAN"),
content: Form(
key: _formKey,
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
TextFormField(
decoration: InputDecoration(
labelText: "EAN",
border: OutlineInputBorder(),
),
keyboardType: TextInputType.number,
onSaved: (newValue) {
ean = newValue;
},
validator: (value) {
if (value == null ||
value.length != 13 ||
int.tryParse(value) == null) {
return "L'entrée n'est pas un code EAN-13 valide";
}
return null;
},
),
],
),
),
actions: [
TextButton(
onPressed: () {
widget.exitPopup(context);
},
child: Text("Annuler"),
),
TextButton(
onPressed: () {
if (_formKey.currentState!.validate()) {
_formKey.currentState!.save();
widget.exitPopup(context);
}
},
child: Text("Valider"),
),
],
);
}
}
class _FullyManual extends StatefulWidget {
const _FullyManual({required this.exitPopup});
final Function(BuildContext) exitPopup;
@override
State<_FullyManual> createState() => _FullyManualState();
}
class _FullyManualState extends State<_FullyManual> {
final GlobalKey<FormState> _formKey = GlobalKey<FormState>();
String? ean = "";
String? title;
String? author;
@override
Widget build(BuildContext context) {
return AlertDialog(
title: Text("Entrée manuelle"),
content: Form(
key: _formKey,
child: Column(
mainAxisSize: MainAxisSize.min,
children: [
TextFormField(
decoration: InputDecoration(
labelText: "EAN",
helperText: "Optionnel",
border: OutlineInputBorder(),
),
keyboardType: TextInputType.number,
validator: (value) {
if (value == null ||
(value.length != 13 && value.isNotEmpty) ||
int.tryParse(value) != null) {
return "Indiquez un EAN valide ou rien";
}
return null;
},
onSaved: (newValue) {
ean = newValue;
},
),
SizedBox(height: 10),
TextFormField(
decoration: InputDecoration(
labelText: "Titre",
border: OutlineInputBorder(),
),
validator: (value) {
if (value == null || value.isEmpty) {
return "Indiquez un titre";
}
return null;
},
onSaved: (newValue) {
title = newValue;
},
),
SizedBox(height: 10),
TextFormField(
decoration: InputDecoration(
labelText: "Auteur·ice",
border: OutlineInputBorder(),
),
validator: (value) {
if (value == null || value.isEmpty) {
return "Indiquez un·e auteur·ice";
}
return null;
},
onSaved: (newValue) {
author = newValue;
},
),
],
),
),
actions: [
TextButton(
onPressed: () {
widget.exitPopup(context);
},
child: Text("Annuler"),
),
TextButton(
onPressed: () {
if (_formKey.currentState!.validate()) {
_formKey.currentState!.save();
widget.exitPopup(context);
}
},
child: Text("Valider"),
),
],
);
}
}

View file

@ -143,9 +143,9 @@ class _OwnerPopupState extends State<OwnerPopup> {
onPressed: () {
if (_formKey.currentState!.validate()) {
_formKey.currentState!.save();
setState(() {
widget.viewModel.currentOwner = widget.viewModel
.addOwner(firstName!, lastName!, contact!);
setState(() {
showNewOwner = false;
});
}

View file

@ -1,75 +0,0 @@
import 'package:flutter/material.dart';
import 'package:flutter_svg/svg.dart';
import 'package:mobile_scanner/mobile_scanner.dart';
import 'package:provider/provider.dart';
// class ScanPage extends StatefulWidget {
// const ScanPage({super.key});
// @override
// State<ScanPage> createState() => _ScanPage();
// }
// class _ScanPage extends State<ScanPage> {
// final MobileScannerController controller = MobileScannerController(
// detectionTimeoutMs: 1000,
// );
// @override
// Widget build(BuildContext context) {
// return Stack(
// children: <Widget>[
// MobileScanner(
// controller: controller,
// onDetect: (result) {
// print(result.barcodes.first.rawValue);
// },
// ),
// SafeArea(
// child: Column(
// crossAxisAlignment: CrossAxisAlignment.center,
// children: [
// Center(
// child: Card(
// margin: EdgeInsets.symmetric(horizontal: 50),
// child: Column(
// children: [
// Consumer<TabScreen>(
// builder: (context, screen, child) {
// return ListTile(
// leading: Icon(Icons.person),
// title: TextButton(
// child: Text("No"),
// onPressed: () {
// screen.change("ownerPage");
// },
// ),
// );
// },
// ),
// ListTile(
// leading: Icon(Icons.attach_money),
// title: TextButton(
// child: Text("Demander à chaque fois"),
// onPressed: () {
// return;
// },
// ),
// ),
// ],
// ),
// ),
// ),
// Expanded(child: SizedBox()),
// SvgPicture.asset('assets/scan-overlay.svg'),
// Expanded(child: SizedBox()),
// TextButton(
// onPressed: () {},
// child: Text("Enregistrer manuellement"),
// ),
// ],
// ),
// ),
// ],
// );
// }
// }