From 4d9a9ecab12404f9ee37cec56434c8b45a8418af Mon Sep 17 00:00:00 2001 From: Alzalia Date: Tue, 5 Aug 2025 10:37:04 +0200 Subject: [PATCH 1/4] Delete useless scan_screen --- lib/ui/add_page/widgets/scan_screen.dart | 75 ------------------------ 1 file changed, 75 deletions(-) delete mode 100644 lib/ui/add_page/widgets/scan_screen.dart diff --git a/lib/ui/add_page/widgets/scan_screen.dart b/lib/ui/add_page/widgets/scan_screen.dart deleted file mode 100644 index 10d7e54..0000000 --- a/lib/ui/add_page/widgets/scan_screen.dart +++ /dev/null @@ -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 createState() => _ScanPage(); -// } - -// class _ScanPage extends State { -// final MobileScannerController controller = MobileScannerController( -// detectionTimeoutMs: 1000, -// ); -// @override -// Widget build(BuildContext context) { -// return Stack( -// children: [ -// 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( -// 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"), -// ), -// ], -// ), -// ), -// ], -// ); -// } -// } From 8af775a1d3a38fc142fb7ed85c618c6a1f13138b Mon Sep 17 00:00:00 2001 From: Alzalia Date: Tue, 5 Aug 2025 10:43:30 +0200 Subject: [PATCH 2/4] Owner name is correctly displayed --- lib/ui/add_page/widgets/add_page.dart | 23 +++++++++++++++-------- 1 file changed, 15 insertions(+), 8 deletions(-) diff --git a/lib/ui/add_page/widgets/add_page.dart b/lib/ui/add_page/widgets/add_page.dart index 07e4fe3..5cc3382 100644 --- a/lib/ui/add_page/widgets/add_page.dart +++ b/lib/ui/add_page/widgets/add_page.dart @@ -44,14 +44,21 @@ class _AddPageState extends State { margin: EdgeInsets.symmetric(horizontal: 50), child: Column( children: [ - ListTile( - leading: Icon(Icons.person), - title: TextButton( - child: Text("No"), - onPressed: () => _ownerDialogBuilder( - context, - controller, - widget.viewModel, + ListenableBuilder( + listenable: widget.viewModel, + builder: (context, child) => ListTile( + leading: Icon(Icons.person), + title: TextButton( + child: Text( + (widget.viewModel.currentOwner == null) + ? "Aucun" + : "${widget.viewModel.currentOwner!.firstName} ${widget.viewModel.currentOwner!.lastName}", + ), + onPressed: () => _ownerDialogBuilder( + context, + controller, + widget.viewModel, + ), ), ), ), From cbf43f6d00f35bac1d40396f8e1242ba3600c464 Mon Sep 17 00:00:00 2001 From: Alzalia Date: Tue, 5 Aug 2025 10:44:21 +0200 Subject: [PATCH 3/4] update readme --- README.md | 15 +-------------- 1 file changed, 1 insertion(+), 14 deletions(-) diff --git a/README.md b/README.md index c8b539b..304117b 100644 --- a/README.md +++ b/README.md @@ -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. From 19c4243273157d29831d839109f6505401562cd6 Mon Sep 17 00:00:00 2001 From: Alzalia Date: Tue, 5 Aug 2025 12:21:49 +0200 Subject: [PATCH 4/4] feat: manual register --- lib/ui/add_page/widgets/add_page.dart | 30 ++- lib/ui/add_page/widgets/form_popup.dart | 252 +++++++++++++++++++++++ lib/ui/add_page/widgets/form_screen.dart | 0 lib/ui/add_page/widgets/owner_popup.dart | 4 +- 4 files changed, 282 insertions(+), 4 deletions(-) create mode 100644 lib/ui/add_page/widgets/form_popup.dart delete mode 100644 lib/ui/add_page/widgets/form_screen.dart diff --git a/lib/ui/add_page/widgets/add_page.dart b/lib/ui/add_page/widgets/add_page.dart index 5cc3382..31022c3 100644 --- a/lib/ui/add_page/widgets/add_page.dart +++ b/lib/ui/add_page/widgets/add_page.dart @@ -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 { formats: [BarcodeFormat.ean13], detectionTimeoutMs: 1000, ); + + final theme = Theme.of(context); // return Consumer( // builder: (context, screen, child) { return Scaffold( @@ -85,7 +88,11 @@ class _AddPageState extends State { 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"), ), ], @@ -106,6 +113,25 @@ void onBarcodeScan( return; } +Future _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 _ownerDialogBuilder( BuildContext context, MobileScannerController controller, @@ -114,8 +140,8 @@ Future _ownerDialogBuilder( controller.stop(); void onPressAccept(BuildContext localContext) { - controller.start(); Navigator.of(localContext).pop(); + controller.start(); } return showDialog( diff --git a/lib/ui/add_page/widgets/form_popup.dart b/lib/ui/add_page/widgets/form_popup.dart new file mode 100644 index 0000000..519ea0d --- /dev/null +++ b/lib/ui/add_page/widgets/form_popup.dart @@ -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 _formKey = GlobalKey(); + 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 _formKey = GlobalKey(); + 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"), + ), + ], + ); + } +} diff --git a/lib/ui/add_page/widgets/form_screen.dart b/lib/ui/add_page/widgets/form_screen.dart deleted file mode 100644 index e69de29..0000000 diff --git a/lib/ui/add_page/widgets/owner_popup.dart b/lib/ui/add_page/widgets/owner_popup.dart index d0829ca..b85f5e8 100644 --- a/lib/ui/add_page/widgets/owner_popup.dart +++ b/lib/ui/add_page/widgets/owner_popup.dart @@ -143,9 +143,9 @@ class _OwnerPopupState extends State { onPressed: () { if (_formKey.currentState!.validate()) { _formKey.currentState!.save(); + widget.viewModel.currentOwner = widget.viewModel + .addOwner(firstName!, lastName!, contact!); setState(() { - widget.viewModel.currentOwner = widget.viewModel - .addOwner(firstName!, lastName!, contact!); showNewOwner = false; }); }