From f53f6dd69710c2329aaddd87616f5f21d1b09558 Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Tue, 19 Aug 2025 13:07:18 +0200 Subject: [PATCH 01/29] feat: web build script --- scripts/build-web.sh | 1 + 1 file changed, 1 insertion(+) create mode 100755 scripts/build-web.sh diff --git a/scripts/build-web.sh b/scripts/build-web.sh new file mode 100755 index 0000000..c7456c3 --- /dev/null +++ b/scripts/build-web.sh @@ -0,0 +1 @@ +flutter build web --release --wasm --base-href /app/ From 73e907ef7e5e153af5325a4651cf9d6d18329875 Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Tue, 19 Aug 2025 13:07:48 +0200 Subject: [PATCH 02/29] update flutter_secure_storage to 10.0.0-beta.4 for wasm compatibility --- macos/Flutter/GeneratedPluginRegistrant.swift | 4 +- pubspec.lock | 64 ++++++++----------- pubspec.yaml | 3 +- 3 files changed, 32 insertions(+), 39 deletions(-) diff --git a/macos/Flutter/GeneratedPluginRegistrant.swift b/macos/Flutter/GeneratedPluginRegistrant.swift index c44ff01..aecce92 100644 --- a/macos/Flutter/GeneratedPluginRegistrant.swift +++ b/macos/Flutter/GeneratedPluginRegistrant.swift @@ -5,12 +5,12 @@ import FlutterMacOS import Foundation -import flutter_secure_storage_macos +import flutter_secure_storage_darwin import mobile_scanner import path_provider_foundation func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) { - FlutterSecureStoragePlugin.register(with: registry.registrar(forPlugin: "FlutterSecureStoragePlugin")) + FlutterSecureStorageDarwinPlugin.register(with: registry.registrar(forPlugin: "FlutterSecureStorageDarwinPlugin")) MobileScannerPlugin.register(with: registry.registrar(forPlugin: "MobileScannerPlugin")) PathProviderPlugin.register(with: registry.registrar(forPlugin: "PathProviderPlugin")) } diff --git a/pubspec.lock b/pubspec.lock index f140cfe..97d2e53 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -151,50 +151,50 @@ packages: dependency: "direct main" description: name: flutter_secure_storage - sha256: "9cad52d75ebc511adfae3d447d5d13da15a55a92c9410e50f67335b6d21d16ea" + sha256: f7eceb0bc6f4fd0441e29d43cab9ac2a1c5ffd7ea7b64075136b718c46954874 url: "https://pub.dev" source: hosted - version: "9.2.4" + version: "10.0.0-beta.4" + flutter_secure_storage_darwin: + dependency: transitive + description: + name: flutter_secure_storage_darwin + sha256: f226f2a572bed96bc6542198ebaec227150786e34311d455a7e2d3d06d951845 + url: "https://pub.dev" + source: hosted + version: "0.1.0" flutter_secure_storage_linux: dependency: transitive description: name: flutter_secure_storage_linux - sha256: be76c1d24a97d0b98f8b54bce6b481a380a6590df992d0098f868ad54dc8f688 + sha256: "9b4b73127e857cd3117d43a70fa3dddadb6e0b253be62e6a6ab85caa0742182c" url: "https://pub.dev" source: hosted - version: "1.2.3" - flutter_secure_storage_macos: - dependency: transitive - description: - name: flutter_secure_storage_macos - sha256: "6c0a2795a2d1de26ae202a0d78527d163f4acbb11cde4c75c670f3a0fc064247" - url: "https://pub.dev" - source: hosted - version: "3.1.3" + version: "2.0.1" flutter_secure_storage_platform_interface: dependency: transitive description: name: flutter_secure_storage_platform_interface - sha256: cf91ad32ce5adef6fba4d736a542baca9daf3beac4db2d04be350b87f69ac4a8 + sha256: "8ceea1223bee3c6ac1a22dabd8feefc550e4729b3675de4b5900f55afcb435d6" url: "https://pub.dev" source: hosted - version: "1.1.2" + version: "2.0.1" flutter_secure_storage_web: dependency: transitive description: name: flutter_secure_storage_web - sha256: f4ebff989b4f07b2656fb16b47852c0aab9fed9b4ec1c70103368337bc1886a9 + sha256: "4c3f233e739545c6cb09286eeec1cc4744138372b985113acc904f7263bef517" url: "https://pub.dev" source: hosted - version: "1.2.1" + version: "2.0.0" flutter_secure_storage_windows: dependency: transitive description: name: flutter_secure_storage_windows - sha256: b20b07cb5ed4ed74fc567b78a72936203f587eba460af1df11281c9326cd3709 + sha256: ff32af20f70a8d0e59b2938fc92de35b54a74671041c814275afd80e27df9f21 url: "https://pub.dev" source: hosted - version: "3.1.2" + version: "4.0.0" flutter_svg: dependency: "direct main" description: @@ -253,14 +253,6 @@ packages: url: "https://pub.dev" source: hosted version: "0.20.2" - js: - dependency: transitive - description: - name: js - sha256: f2c445dce49627136094980615a031419f7f3eb393237e4ecd97ac15dea343f3 - url: "https://pub.dev" - source: hosted - version: "0.6.7" json_annotation: dependency: transitive description: @@ -273,26 +265,26 @@ packages: dependency: transitive description: name: leak_tracker - sha256: "6bb818ecbdffe216e81182c2f0714a2e62b593f4a4f13098713ff1685dfb6ab0" + sha256: "8dcda04c3fc16c14f48a7bb586d4be1f0d1572731b6d81d51772ef47c02081e0" url: "https://pub.dev" source: hosted - version: "10.0.9" + version: "11.0.1" leak_tracker_flutter_testing: dependency: transitive description: name: leak_tracker_flutter_testing - sha256: f8b613e7e6a13ec79cfdc0e97638fddb3ab848452eff057653abd3edba760573 + sha256: "1dbc140bb5a23c75ea9c4811222756104fbcd1a27173f0c34ca01e16bea473c1" url: "https://pub.dev" source: hosted - version: "3.0.9" + version: "3.0.10" leak_tracker_testing: dependency: transitive description: name: leak_tracker_testing - sha256: "6ba465d5d76e67ddf503e1161d1f4a6bc42306f9d66ca1e8f079a47290fb06d3" + sha256: "8d5a2d49f4a66b49744b23b018848400d23e54caf9463f4eb20df3eb8acb2eb1" url: "https://pub.dev" source: hosted - version: "3.0.1" + version: "3.0.2" lints: dependency: transitive description: @@ -510,10 +502,10 @@ packages: dependency: transitive description: name: test_api - sha256: fb31f383e2ee25fbbfe06b40fe21e1e458d14080e3c67e7ba0acfde4df4e0bbd + sha256: "522f00f556e73044315fa4585ec3270f1808a4b186c936e612cab0b565ff1e00" url: "https://pub.dev" source: hosted - version: "0.7.4" + version: "0.7.6" typed_data: dependency: transitive description: @@ -550,10 +542,10 @@ packages: dependency: transitive description: name: vector_math - sha256: "80b3257d1492ce4d091729e3a67a60407d227c27241d6927be0130c98e741803" + sha256: d530bd74fea330e6e364cda7a85019c434070188383e1cd8d9777ee586914c5b url: "https://pub.dev" source: hosted - version: "2.1.4" + version: "2.2.0" vm_service: dependency: transitive description: diff --git a/pubspec.yaml b/pubspec.yaml index 0ffed5e..2c18afe 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -45,7 +45,8 @@ dependencies: http: ^1.4.0 web_socket_channel: ^3.0.3 nested: ^1.0.0 - flutter_secure_storage: ^9.2.4 + #flutter_secure_storage: ^9.2.4 + flutter_secure_storage: ^10.0.0-beta.4 rxdart: ^0.28.0 intl: ^0.20.2 flutter_launcher_icons: ^0.14.4 From 8a464cc665903ced5549c0316a8c542c8928e860 Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Tue, 19 Aug 2025 21:23:44 +0200 Subject: [PATCH 03/29] fix: incorrect minSdk version --- android/app/build.gradle.kts | 2 +- pubspec.lock | 20 ++++++++++---------- 2 files changed, 11 insertions(+), 11 deletions(-) diff --git a/android/app/build.gradle.kts b/android/app/build.gradle.kts index d2ea4a2..a8000a2 100644 --- a/android/app/build.gradle.kts +++ b/android/app/build.gradle.kts @@ -24,7 +24,7 @@ android { applicationId = "fr.ueauvergne.seshat" // You can update the following values to match your application needs. // For more information, see: https://flutter.dev/to/review-gradle-config. - minSdk = flutter.minSdkVersion + minSdk = 23 targetSdk = flutter.targetSdkVersion versionCode = flutter.versionCode versionName = flutter.versionName diff --git a/pubspec.lock b/pubspec.lock index 97d2e53..40b0cde 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -265,26 +265,26 @@ packages: dependency: transitive description: name: leak_tracker - sha256: "8dcda04c3fc16c14f48a7bb586d4be1f0d1572731b6d81d51772ef47c02081e0" + sha256: "6bb818ecbdffe216e81182c2f0714a2e62b593f4a4f13098713ff1685dfb6ab0" url: "https://pub.dev" source: hosted - version: "11.0.1" + version: "10.0.9" leak_tracker_flutter_testing: dependency: transitive description: name: leak_tracker_flutter_testing - sha256: "1dbc140bb5a23c75ea9c4811222756104fbcd1a27173f0c34ca01e16bea473c1" + sha256: f8b613e7e6a13ec79cfdc0e97638fddb3ab848452eff057653abd3edba760573 url: "https://pub.dev" source: hosted - version: "3.0.10" + version: "3.0.9" leak_tracker_testing: dependency: transitive description: name: leak_tracker_testing - sha256: "8d5a2d49f4a66b49744b23b018848400d23e54caf9463f4eb20df3eb8acb2eb1" + sha256: "6ba465d5d76e67ddf503e1161d1f4a6bc42306f9d66ca1e8f079a47290fb06d3" url: "https://pub.dev" source: hosted - version: "3.0.2" + version: "3.0.1" lints: dependency: transitive description: @@ -502,10 +502,10 @@ packages: dependency: transitive description: name: test_api - sha256: "522f00f556e73044315fa4585ec3270f1808a4b186c936e612cab0b565ff1e00" + sha256: fb31f383e2ee25fbbfe06b40fe21e1e458d14080e3c67e7ba0acfde4df4e0bbd url: "https://pub.dev" source: hosted - version: "0.7.6" + version: "0.7.4" typed_data: dependency: transitive description: @@ -542,10 +542,10 @@ packages: dependency: transitive description: name: vector_math - sha256: d530bd74fea330e6e364cda7a85019c434070188383e1cd8d9777ee586914c5b + sha256: "80b3257d1492ce4d091729e3a67a60407d227c27241d6927be0130c98e741803" url: "https://pub.dev" source: hosted - version: "2.2.0" + version: "2.1.4" vm_service: dependency: transitive description: From 587cc0689a4783b1e7ca5a66b2fe79f76376987e Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Tue, 19 Aug 2025 22:57:44 +0200 Subject: [PATCH 04/29] fix: try at fixing incorrect scan box size on smaller screens --- android/app/build.gradle.kts | 2 +- .../reports/problems/problems-report.html | 2 +- lib/ui/add_page/widgets/add_page.dart | 7 ++++++- lib/ui/sell_page/widgets/scan_screen.dart | 7 ++++++- pubspec.lock | 20 +++++++++---------- 5 files changed, 24 insertions(+), 14 deletions(-) diff --git a/android/app/build.gradle.kts b/android/app/build.gradle.kts index a8000a2..80cd4ad 100644 --- a/android/app/build.gradle.kts +++ b/android/app/build.gradle.kts @@ -24,7 +24,7 @@ android { applicationId = "fr.ueauvergne.seshat" // You can update the following values to match your application needs. // For more information, see: https://flutter.dev/to/review-gradle-config. - minSdk = 23 + minSdkVersion(24) targetSdk = flutter.targetSdkVersion versionCode = flutter.versionCode versionName = flutter.versionName diff --git a/android/build/reports/problems/problems-report.html b/android/build/reports/problems/problems-report.html index f7fc0b1..d098403 100644 --- a/android/build/reports/problems/problems-report.html +++ b/android/build/reports/problems/problems-report.html @@ -650,7 +650,7 @@ code + .copy-button { diff --git a/lib/ui/add_page/widgets/add_page.dart b/lib/ui/add_page/widgets/add_page.dart index 39808ac..3e85857 100644 --- a/lib/ui/add_page/widgets/add_page.dart +++ b/lib/ui/add_page/widgets/add_page.dart @@ -123,6 +123,12 @@ class _AddPageState extends State { } }, ), + Center( + child: SvgPicture.asset( + 'assets/scan-overlay.svg', + height: (MediaQuery.sizeOf(context).height / 5) * 2, + ), + ), SafeArea( child: SingleChildScrollView( child: Column( @@ -175,7 +181,6 @@ class _AddPageState extends State { ), ), ), - Center(child: SvgPicture.asset('assets/scan-overlay.svg')), SafeArea( child: Column( mainAxisAlignment: MainAxisAlignment.end, diff --git a/lib/ui/sell_page/widgets/scan_screen.dart b/lib/ui/sell_page/widgets/scan_screen.dart index 1b15ee8..b500795 100644 --- a/lib/ui/sell_page/widgets/scan_screen.dart +++ b/lib/ui/sell_page/widgets/scan_screen.dart @@ -51,6 +51,12 @@ class _ScanScreenState extends State { await widget.viewModel.scanBook(barcodes); }, ), + Center( + child: SvgPicture.asset( + 'assets/scan-overlay.svg', + height: (MediaQuery.sizeOf(context).height / 5) * 2, + ), + ), SafeArea( child: Column( children: [ @@ -63,7 +69,6 @@ class _ScanScreenState extends State { ], ), ), - Center(child: SvgPicture.asset('assets/scan-overlay.svg')), Column( mainAxisAlignment: MainAxisAlignment.end, crossAxisAlignment: CrossAxisAlignment.center, diff --git a/pubspec.lock b/pubspec.lock index 40b0cde..97d2e53 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -265,26 +265,26 @@ packages: dependency: transitive description: name: leak_tracker - sha256: "6bb818ecbdffe216e81182c2f0714a2e62b593f4a4f13098713ff1685dfb6ab0" + sha256: "8dcda04c3fc16c14f48a7bb586d4be1f0d1572731b6d81d51772ef47c02081e0" url: "https://pub.dev" source: hosted - version: "10.0.9" + version: "11.0.1" leak_tracker_flutter_testing: dependency: transitive description: name: leak_tracker_flutter_testing - sha256: f8b613e7e6a13ec79cfdc0e97638fddb3ab848452eff057653abd3edba760573 + sha256: "1dbc140bb5a23c75ea9c4811222756104fbcd1a27173f0c34ca01e16bea473c1" url: "https://pub.dev" source: hosted - version: "3.0.9" + version: "3.0.10" leak_tracker_testing: dependency: transitive description: name: leak_tracker_testing - sha256: "6ba465d5d76e67ddf503e1161d1f4a6bc42306f9d66ca1e8f079a47290fb06d3" + sha256: "8d5a2d49f4a66b49744b23b018848400d23e54caf9463f4eb20df3eb8acb2eb1" url: "https://pub.dev" source: hosted - version: "3.0.1" + version: "3.0.2" lints: dependency: transitive description: @@ -502,10 +502,10 @@ packages: dependency: transitive description: name: test_api - sha256: fb31f383e2ee25fbbfe06b40fe21e1e458d14080e3c67e7ba0acfde4df4e0bbd + sha256: "522f00f556e73044315fa4585ec3270f1808a4b186c936e612cab0b565ff1e00" url: "https://pub.dev" source: hosted - version: "0.7.4" + version: "0.7.6" typed_data: dependency: transitive description: @@ -542,10 +542,10 @@ packages: dependency: transitive description: name: vector_math - sha256: "80b3257d1492ce4d091729e3a67a60407d227c27241d6927be0130c98e741803" + sha256: d530bd74fea330e6e364cda7a85019c434070188383e1cd8d9777ee586914c5b url: "https://pub.dev" source: hosted - version: "2.1.4" + version: "2.2.0" vm_service: dependency: transitive description: From aa3fda9c88c9b932a9eeead53fd25baf7b3ba2fe Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Tue, 19 Aug 2025 23:17:45 +0200 Subject: [PATCH 05/29] fix: fixed a bug that would cause empty scanned barcodes to be treated as no-empty --- lib/ui/add_page/widgets/add_page.dart | 3 +++ lib/ui/sell_page/widgets/scan_screen.dart | 3 +++ 2 files changed, 6 insertions(+) diff --git a/lib/ui/add_page/widgets/add_page.dart b/lib/ui/add_page/widgets/add_page.dart index 3e85857..ab71bd4 100644 --- a/lib/ui/add_page/widgets/add_page.dart +++ b/lib/ui/add_page/widgets/add_page.dart @@ -80,6 +80,9 @@ class _AddPageState extends State { MobileScanner( controller: controller, onDetect: (barcodes) async { + if (barcodes.barcodes.isEmpty) { + return; + } if (widget.viewModel.currentOwner == null) { ScaffoldMessenger.of(context).showSnackBar( SnackBar( diff --git a/lib/ui/sell_page/widgets/scan_screen.dart b/lib/ui/sell_page/widgets/scan_screen.dart index b500795..ceb9920 100644 --- a/lib/ui/sell_page/widgets/scan_screen.dart +++ b/lib/ui/sell_page/widgets/scan_screen.dart @@ -41,6 +41,9 @@ class _ScanScreenState extends State { MobileScanner( controller: controller, onDetect: (barcodes) async { + if (barcodes.barcodes.isEmpty) { + return; + } controller.stop(); showDialog( context: context, From 77a2ce129a70355894e92d06e6930b02761d0689 Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 10:38:29 +0200 Subject: [PATCH 06/29] feat(ci): add automatic web build and deployment on release --- .forgejo/workflows/deploy.yml | 15 +++++++++++++++ 1 file changed, 15 insertions(+) create mode 100644 .forgejo/workflows/deploy.yml diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml new file mode 100644 index 0000000..d4318a8 --- /dev/null +++ b/.forgejo/workflows/deploy.yml @@ -0,0 +1,15 @@ +on: + release: + types: [published] +jobs: + test: + runs-on: docker + steps: + - run: apt install sshpass + - uses: actions/checkout@v4 + - name: Setup Flutter SDK + run: | + export SETUP_FLUTTER_BRANCH=main + curl -fsSL https://raw.githubusercontent.com/flutter-actions/setup-flutter/${SETUP_FLUTTER_BRANCH}/install.sh | bash -s -- 3.35.1 stable + - run: bash scripts/build-web.sh + - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From 08e986f55efed0fdd4cb78d1783a5881eed5d958 Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 11:02:56 +0200 Subject: [PATCH 07/29] fix(ci): update apt before pulling sshpass --- .forgejo/workflows/deploy.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index d4318a8..ad619e1 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -5,7 +5,7 @@ jobs: test: runs-on: docker steps: - - run: apt install sshpass + - run: apt update && apt install sshpass - uses: actions/checkout@v4 - name: Setup Flutter SDK run: | From 7695f4d563d0bd66644876f5c7ebf61ddb5901af Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 11:10:41 +0200 Subject: [PATCH 08/29] fix(ci): add flutter tool cache as safe git directory --- .forgejo/workflows/deploy.yml | 4 +++- 1 file changed, 3 insertions(+), 1 deletion(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index ad619e1..a321284 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -10,6 +10,8 @@ jobs: - name: Setup Flutter SDK run: | export SETUP_FLUTTER_BRANCH=main - curl -fsSL https://raw.githubusercontent.com/flutter-actions/setup-flutter/${SETUP_FLUTTER_BRANCH}/install.sh | bash -s -- 3.35.1 stable + export FLUTTER_VERSION=3.35.1 + git config --global --add safe.directory /workspace/UEAuvergne/Seshat/.setup-flutter/tool_cache/flutter/${FLUTTER_VERSION}/stable/flutter + curl -fsSL https://raw.githubusercontent.com/flutter-actions/setup-flutter/${SETUP_FLUTTER_BRANCH}/install.sh | bash -s -- ${FLUTTER_VERSION} stable - run: bash scripts/build-web.sh - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From 82236a83ad97287d21b946d00f8e4795f18ee63a Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 11:24:26 +0200 Subject: [PATCH 09/29] fix(ci): add flutter bin to PATH --- .forgejo/workflows/deploy.yml | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index a321284..dc19da5 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -12,6 +12,8 @@ jobs: export SETUP_FLUTTER_BRANCH=main export FLUTTER_VERSION=3.35.1 git config --global --add safe.directory /workspace/UEAuvergne/Seshat/.setup-flutter/tool_cache/flutter/${FLUTTER_VERSION}/stable/flutter - curl -fsSL https://raw.githubusercontent.com/flutter-actions/setup-flutter/${SETUP_FLUTTER_BRANCH}/install.sh | bash -s -- ${FLUTTER_VERSION} stable - - run: bash scripts/build-web.sh + curl -fsSL https://raw.githubusercontent.com/flutter-actions/setup-flutter/${SETUP_FLUTTER_BRANCH}/install.sh | bash -s -- ${FLUTTER_VERSION} ${FLUTTER_CHANNEL} + export FLUTTER_RUNNER_TOOL_CACHE="${RUNNER_TOOL_CACHE}/flutter/${FLUTTER_VERSION}/${FLUTTER_CHANNEL}" + export PATH=$PATH:${FLUTTER_RUNNER_TOOL_CACHE}/flutter/bin + - run: ./scripts/build-web.sh - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From 594c54930e8f738bc20994ad4c6d7ce90ae213ad Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 11:45:29 +0200 Subject: [PATCH 10/29] fix(ci): use flutter build command instead of script --- .forgejo/workflows/deploy.yml | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index dc19da5..04141ee 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -13,7 +13,6 @@ jobs: export FLUTTER_VERSION=3.35.1 git config --global --add safe.directory /workspace/UEAuvergne/Seshat/.setup-flutter/tool_cache/flutter/${FLUTTER_VERSION}/stable/flutter curl -fsSL https://raw.githubusercontent.com/flutter-actions/setup-flutter/${SETUP_FLUTTER_BRANCH}/install.sh | bash -s -- ${FLUTTER_VERSION} ${FLUTTER_CHANNEL} - export FLUTTER_RUNNER_TOOL_CACHE="${RUNNER_TOOL_CACHE}/flutter/${FLUTTER_VERSION}/${FLUTTER_CHANNEL}" - export PATH=$PATH:${FLUTTER_RUNNER_TOOL_CACHE}/flutter/bin - - run: ./scripts/build-web.sh + + - run: flutter build web --release --wasm --base-href /app/ - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From 7ca31c676c8953bff26784e3819d850f686faf32 Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 11:51:45 +0200 Subject: [PATCH 11/29] fix(ci): change flutter action --- .forgejo/workflows/deploy.yml | 12 +++++------- 1 file changed, 5 insertions(+), 7 deletions(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index 04141ee..ef5481e 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -7,12 +7,10 @@ jobs: steps: - run: apt update && apt install sshpass - uses: actions/checkout@v4 - - name: Setup Flutter SDK - run: | - export SETUP_FLUTTER_BRANCH=main - export FLUTTER_VERSION=3.35.1 - git config --global --add safe.directory /workspace/UEAuvergne/Seshat/.setup-flutter/tool_cache/flutter/${FLUTTER_VERSION}/stable/flutter - curl -fsSL https://raw.githubusercontent.com/flutter-actions/setup-flutter/${SETUP_FLUTTER_BRANCH}/install.sh | bash -s -- ${FLUTTER_VERSION} ${FLUTTER_CHANNEL} - + - name: Set up Flutter + uses: https://github.com/subosito/flutter-action@v2 + with: + channel: stable + flutter-version: 3.35.1 - run: flutter build web --release --wasm --base-href /app/ - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From cbe51a8566dc6ede9df8ff0c34399d6700f573ab Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 11:55:48 +0200 Subject: [PATCH 12/29] fix(ci): add jq to apt install list --- .forgejo/workflows/deploy.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index ef5481e..7def02a 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -5,7 +5,7 @@ jobs: test: runs-on: docker steps: - - run: apt update && apt install sshpass + - run: apt update && apt install sshpass jq - uses: actions/checkout@v4 - name: Set up Flutter uses: https://github.com/subosito/flutter-action@v2 From 9144ef74d72aad4277e7e8cb226220b7c231c01e Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 11:57:43 +0200 Subject: [PATCH 13/29] fix(ci): assume yes in apt install --- .forgejo/workflows/deploy.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index 7def02a..39c6cdf 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -5,7 +5,7 @@ jobs: test: runs-on: docker steps: - - run: apt update && apt install sshpass jq + - run: apt update && apt install -y sshpass jq - uses: actions/checkout@v4 - name: Set up Flutter uses: https://github.com/subosito/flutter-action@v2 From b18339e441c2dc4a37a4da4e9bdc9f3ded52b518 Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 12:03:41 +0200 Subject: [PATCH 14/29] fix(ci): flutter safe directory --- .forgejo/workflows/deploy.yml | 1 + 1 file changed, 1 insertion(+) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index 39c6cdf..51456fa 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -12,5 +12,6 @@ jobs: with: channel: stable flutter-version: 3.35.1 + - run: git config --global --add safe.directory /opt/hostedtoolcache/flutter/stable-3.35.1-x64 - run: flutter build web --release --wasm --base-href /app/ - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From e5d383548d9531c62637269ed0f3514779298210 Mon Sep 17 00:00:00 2001 From: Ninjdai Date: Wed, 20 Aug 2025 12:04:48 +0200 Subject: [PATCH 15/29] feat(ci): enable flutter cache --- .forgejo/workflows/deploy.yml | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index 51456fa..3555aa2 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -11,7 +11,8 @@ jobs: uses: https://github.com/subosito/flutter-action@v2 with: channel: stable - flutter-version: 3.35.1 + flutter-version: 3.35.1 + cache: true - run: git config --global --add safe.directory /opt/hostedtoolcache/flutter/stable-3.35.1-x64 - run: flutter build web --release --wasm --base-href /app/ - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From ebeab2db9443879a0e2a80196096859b3af43bcd Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 12:13:33 +0200 Subject: [PATCH 16/29] fix: no decimal on prices on ios --- lib/data/services/api_client.dart | 228 ++++++++++-------- .../add_page/widgets/confirmation_popup.dart | 8 +- lib/ui/add_page/widgets/form_popup.dart | 12 +- lib/ui/sell_page/widgets/sell_page.dart | 4 +- pubspec.lock | 10 +- pubspec.yaml | 2 +- 6 files changed, 161 insertions(+), 103 deletions(-) diff --git a/lib/data/services/api_client.dart b/lib/data/services/api_client.dart index 383f269..2d9e273 100644 --- a/lib/data/services/api_client.dart +++ b/lib/data/services/api_client.dart @@ -1,10 +1,10 @@ import 'dart:convert'; -import 'dart:math'; import 'package:flutter/rendering.dart'; import 'package:flutter/widgets.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:http/http.dart'; +import 'package:logger/logger.dart'; import 'package:seshat/config/constants.dart'; import 'package:seshat/domain/models/accounting.dart'; import 'package:seshat/domain/models/bal.dart'; @@ -22,8 +22,6 @@ extension StringExtension on String { } } -typedef AuthHeaderProvider = String? Function(); - class ApiClient { ApiClient({String? host, int? port}); @@ -31,11 +29,17 @@ class ApiClient { String? token; bool isReady = false; FlutterSecureStorage? _secureStorage; + Logger log = Logger( + printer: PrettyPrinter( + colors: true, + lineLength: 100, + methodCount: 0, + dateTimeFormat: DateTimeFormat.dateAndTime, + ), + ); Future _initStore() async { - _secureStorage ??= const FlutterSecureStorage( - aOptions: AndroidOptions(encryptedSharedPreferences: true), - ); + _secureStorage ??= const FlutterSecureStorage(aOptions: AndroidOptions()); } Future> _getHeaders([ @@ -54,20 +58,25 @@ class ApiClient { */ Future> getAccounting(int balId) async { + final url = "https://$apiBasePath/bal/$balId/accounting"; + log.i("Fetching: getAccounting ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/bal/$balId/accounting"), - headers: headers, - ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Accounting.fromJSON(json)); - } else { - throw "Unknown error"; + final response = await client.get(Uri.parse(url), headers: headers); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Accounting.fromJSON(json)); + case 403: + throw "You don't own the specified BAL"; + case 404: + throw "No BAL with this is exists in database"; + default: + throw "Unknown error of code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -75,25 +84,32 @@ class ApiClient { } Future> returnToId(int balId, int ownerId, String type) async { + final url = + "https://$apiBasePath/bal/${balId.toString()}/accounting/return/${ownerId.toString()}"; + log.i("Fetching: returnToId ($url)"); final client = Client(); try { final headers = await _getHeaders({"Content-Type": "application/json"}); final body = jsonEncode({"return_type": type.capitalize()}); - debugPrint(body); final response = await client.post( - Uri.parse( - "https://$apiBasePath/bal/${balId.toString()}/accounting/return/${ownerId.toString()}", - ), + Uri.parse(url), headers: headers, body: body, ); - if (response.statusCode == 200) { - return Result.ok(response); - } else { - throw "Unknown error ${response.statusCode.toString()}"; + switch (response.statusCode) { + case 200: + return Result.ok(response); + case 403: + throw "You don't own the specified BAL or owner"; + case 404: + throw "No BAL or owner with this id exists in database"; + case 409: + throw "Books and money have already been returned, or there was nothing to return"; + default: + throw "Unknown error of code ${response.statusCode.toString()}"; } } catch (e) { - debugPrint(e.toString()); + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -107,20 +123,27 @@ class ApiClient { */ Future> getBalStats(int id) async { + final url = "https://$apiBasePath/bal/${id.toString()}/stats"; + log.i("Fetching: getBalStats ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/bal/${id.toString()}/stats"), - headers: headers, - ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(BalStats.fromJSON(json)); - } else { - throw "Unknown error"; + final response = await client.get(Uri.parse(url), headers: headers); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(BalStats.fromJSON(json)); + case 403: + throw "You don't own the specified BAL"; + case 404: + throw "No BAL with this id exists in the database"; + case 409: + throw "The specified BAL is not ended yet"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -128,26 +151,27 @@ class ApiClient { } Future> stopBal(int id) async { + final url = "https://$apiBasePath/bal/${id.toString()}/stop"; + log.i("Fetching: stopBal ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.post( - Uri.parse("https://$apiBasePath/bal/${id.toString()}/stop"), - headers: headers, - ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Bal.fromJSON(json)); - } else if (response.statusCode == 403) { - throw "You don't own the specified BAL"; - } else if (response.statusCode == 404) { - throw "No BAL with specified ID found"; - } else if (response.statusCode == 409) { - throw "Selected BAL was not on ongoing state"; - } else { - throw "Unknown error"; + final response = await client.post(Uri.parse(url), headers: headers); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Bal.fromJSON(json)); + case 403: + throw "You don't own the specified BAL"; + case 404: + throw "No BAL with specified ID found"; + case 409: + throw "Selected BAL was not on ongoing state"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -155,26 +179,27 @@ class ApiClient { } Future> startBal(int id) async { + final url = "https://$apiBasePath/bal/${id.toString()}/start"; + log.i("Fetching: startBal ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.post( - Uri.parse("https://$apiBasePath/bal/${id.toString()}/start"), - headers: headers, - ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Bal.fromJSON(json)); - } else if (response.statusCode == 403) { - throw "You don't own the specified BAL"; - } else if (response.statusCode == 404) { - throw "No BAL with specified ID found"; - } else if (response.statusCode == 409) { - throw "Cannot have multiple BAl ongoing at the same time!"; - } else { - throw "Unknown error"; + final response = await client.post(Uri.parse(url), headers: headers); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Bal.fromJSON(json)); + case 403: + throw "You don't own the specified BAL"; + case 404: + throw "No BAL with specified ID found"; + case 409: + throw "Cannot have multiple BAl ongoing at the same time!"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -187,6 +212,8 @@ class ApiClient { DateTime start, DateTime end, ) async { + final url = "https://$apiBasePath/bal/${id.toString()}"; + log.i("Fetching: editBal ($url)"); final client = Client(); try { final headers = await _getHeaders({"Content-Type": "application/json"}); @@ -196,17 +223,23 @@ class ApiClient { "end_timestamp": (end.millisecondsSinceEpoch / 1000).round(), }; final response = await client.patch( - Uri.parse("https://$apiBasePath/bal/${id.toString()}"), + Uri.parse(url), headers: headers, body: jsonEncode(body), ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Bal.fromJSON(json)); - } else { - throw Exception("Something went wrong"); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Bal.fromJSON(json)); + case 403: + throw "You don't own the specified BAL"; + case 404: + throw "No bal with specified id"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -214,24 +247,25 @@ class ApiClient { } Future> getBalById(int id) async { + final url = "https://$apiBasePath/bal/${id.toString()}"; + log.i("Fetching: getBalById ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/bal/${id.toString()}"), - headers: headers, - ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Bal.fromJSON(json)); - } else if (response.statusCode == 403) { - throw Exception("You don't own the specified bal"); - } else { - return Result.error( - Exception("No bal wirth this id exists the database"), - ); + final response = await client.get(Uri.parse(url), headers: headers); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Bal.fromJSON(json)); + case 403: + throw "You don't own the specified BAL"; + case 404: + throw "No BAL with this id found in database"; + default: + throw "Unknown error"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -239,6 +273,8 @@ class ApiClient { } Future> addBal(String name, DateTime start, DateTime end) async { + final url = "https://$apiBasePath/bal"; + log.i("Fetching: addBal ($url)"); final client = Client(); try { final headers = await _getHeaders({"Content-Type": "application/json"}); @@ -248,17 +284,21 @@ class ApiClient { "end_timestamp": (end.millisecondsSinceEpoch / 1000).round(), }; final response = await client.post( - Uri.parse("https://$apiBasePath/bal"), + Uri.parse(url), headers: headers, body: jsonEncode(body), ); - if (response.statusCode == 201) { - final json = jsonDecode(response.body); - return Result.ok(Bal.fromJSON(json)); - } else { - throw Exception("Something went wrong"); + switch (response.statusCode) { + case 201: + final json = jsonDecode(response.body); + return Result.ok(Bal.fromJSON(json)); + case 400: + throw "Time cannot go backwards"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -266,20 +306,14 @@ class ApiClient { } Future>> getBals() async { + final url = "https://$apiBasePath/bals"; + log.i("Fetching: getBals ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/bals"), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body) as List; - debugPrint("\n\n\n\nRECEIVED : $json\n\n\n\n"); - debugPrint( - "\n\n\n\nFORMATTED : ${json.map((element) => Bal.fromJSON(element)).toList()}\n\n\n\n", - ); - return Result.ok(json.map((element) => Bal.fromJSON(element)).toList()); } else { throw Exception("Something wrong happened"); diff --git a/lib/ui/add_page/widgets/confirmation_popup.dart b/lib/ui/add_page/widgets/confirmation_popup.dart index f026ad6..6a6ca74 100644 --- a/lib/ui/add_page/widgets/confirmation_popup.dart +++ b/lib/ui/add_page/widgets/confirmation_popup.dart @@ -1,3 +1,5 @@ +import 'dart:nativewrappers/_internal/vm/lib/ffi_patch.dart'; + import 'package:flutter/material.dart'; import 'package:seshat/domain/models/book.dart'; import 'package:seshat/ui/add_page/view_model/add_view_model.dart'; @@ -80,12 +82,16 @@ class _ConfirmationPopupState extends State { border: OutlineInputBorder(), suffixText: "€", ), - keyboardType: TextInputType.number, + keyboardType: TextInputType.numberWithOptions( + decimal: true, + ), validator: (value) { if (value == null || value.isEmpty) { return "Indiquez un prix"; } else if (num.tryParse(value) == null) { return "Le prix doit être un nombre"; + } else if (num.parse(value) < 0) { + return "Le prix doit être positif ou nul"; } return null; }, diff --git a/lib/ui/add_page/widgets/form_popup.dart b/lib/ui/add_page/widgets/form_popup.dart index 7f6af10..59368e0 100644 --- a/lib/ui/add_page/widgets/form_popup.dart +++ b/lib/ui/add_page/widgets/form_popup.dart @@ -145,12 +145,16 @@ class _ManualEANPopupState extends State<_ManualEANPopup> { border: OutlineInputBorder(), suffixText: "€", ), - keyboardType: TextInputType.number, + keyboardType: TextInputType.numberWithOptions( + decimal: true, + ), validator: (value) { if (value == null || value.isEmpty) { return "Indiquez un prix"; } else if (num.tryParse(value) == null) { return "Le prix doit être un nombre"; + } else if (num.parse(value) < 0) { + return "Le prix doit être positif ou nul"; } return null; }, @@ -273,12 +277,16 @@ class _FullyManualState extends State<_FullyManual> { border: OutlineInputBorder(), suffixText: "€", ), - keyboardType: TextInputType.number, + keyboardType: TextInputType.numberWithOptions( + decimal: true, + ), validator: (value) { if (value == null || value.isEmpty) { return "Indiquez un prix"; } else if (num.tryParse(value) == null) { return "Le prix doit être un nombre"; + } else if (num.parse(value) < 0) { + return "Le prix doit être positif ou nul"; } return null; }, diff --git a/lib/ui/sell_page/widgets/sell_page.dart b/lib/ui/sell_page/widgets/sell_page.dart index 46b2938..dd0404b 100644 --- a/lib/ui/sell_page/widgets/sell_page.dart +++ b/lib/ui/sell_page/widgets/sell_page.dart @@ -129,7 +129,9 @@ class _SellPageState extends State { suffixText: "€", border: OutlineInputBorder(), ), - keyboardType: TextInputType.number, + keyboardType: TextInputType.numberWithOptions( + decimal: true, + ), ), ), ), diff --git a/pubspec.lock b/pubspec.lock index 97d2e53..39a3563 100644 --- a/pubspec.lock +++ b/pubspec.lock @@ -293,8 +293,16 @@ packages: url: "https://pub.dev" source: hosted version: "5.1.1" - logging: + logger: dependency: "direct main" + description: + name: logger + sha256: "55d6c23a6c15db14920e037fe7e0dc32e7cdaf3b64b4b25df2d541b5b6b81c0c" + url: "https://pub.dev" + source: hosted + version: "2.6.1" + logging: + dependency: transitive description: name: logging sha256: c8245ada5f1717ed44271ed1c26b8ce85ca3228fd2ffdb75468ab01979309d61 diff --git a/pubspec.yaml b/pubspec.yaml index 2c18afe..fc17ee5 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -41,7 +41,6 @@ dependencies: provider: ^6.1.5 flutter_svg: ^2.2.0 go_router: ^16.0.0 - logging: ^1.3.0 http: ^1.4.0 web_socket_channel: ^3.0.3 nested: ^1.0.0 @@ -51,6 +50,7 @@ dependencies: intl: ^0.20.2 flutter_launcher_icons: ^0.14.4 fl_chart: ^1.0.0 + logger: ^2.6.1 dev_dependencies: flutter_test: From 025afd3bed161ab1486b8e9d8e8b41ad4cf0ed9a Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 12:19:38 +0200 Subject: [PATCH 17/29] fix: invisible text on small screens --- lib/ui/sell_page/widgets/sell_page.dart | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/lib/ui/sell_page/widgets/sell_page.dart b/lib/ui/sell_page/widgets/sell_page.dart index dd0404b..0ae364e 100644 --- a/lib/ui/sell_page/widgets/sell_page.dart +++ b/lib/ui/sell_page/widgets/sell_page.dart @@ -122,8 +122,7 @@ class _SellPageState extends State { controller: price, decoration: InputDecoration( labelText: "Argent reçu", - hintText: - "Utilisez un point (.) pour les virgules (,)", + hintText: "Les décimales sont avec des .", helperText: "L'argent reçu sera réparti automatiquement.", suffixText: "€", From 609af329e364d1e73dc67112fd6f2672ef051a13 Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 12:25:26 +0200 Subject: [PATCH 18/29] fix: check for owners sooner on manual entry --- lib/ui/add_page/widgets/add_page.dart | 23 +++++++++++++++---- .../add_page/widgets/confirmation_popup.dart | 2 -- 2 files changed, 18 insertions(+), 7 deletions(-) diff --git a/lib/ui/add_page/widgets/add_page.dart b/lib/ui/add_page/widgets/add_page.dart index ab71bd4..d66e827 100644 --- a/lib/ui/add_page/widgets/add_page.dart +++ b/lib/ui/add_page/widgets/add_page.dart @@ -195,11 +195,24 @@ class _AddPageState extends State { theme.cardColor, ), ), - onPressed: () => _formDialogBuilder( - context, - controller, - widget.viewModel, - ), + onPressed: () { + if (widget.viewModel.currentOwner == null) { + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text( + "Attention : vous devez choisir un·e propriétaire", + ), + behavior: SnackBarBehavior.floating, + ), + ); + } + + _formDialogBuilder( + context, + controller, + widget.viewModel, + ); + }, child: Text("Enregistrer manuellement"), ), ), diff --git a/lib/ui/add_page/widgets/confirmation_popup.dart b/lib/ui/add_page/widgets/confirmation_popup.dart index 6a6ca74..980ddee 100644 --- a/lib/ui/add_page/widgets/confirmation_popup.dart +++ b/lib/ui/add_page/widgets/confirmation_popup.dart @@ -1,5 +1,3 @@ -import 'dart:nativewrappers/_internal/vm/lib/ffi_patch.dart'; - import 'package:flutter/material.dart'; import 'package:seshat/domain/models/book.dart'; import 'package:seshat/ui/add_page/view_model/add_view_model.dart'; From 892cd03f796de705c162ed988f717983647b2ebd Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 12:43:35 +0200 Subject: [PATCH 19/29] fix: stats didn't display when stopping bal --- lib/data/services/api_client.dart | 74 +++++++++---------- .../bal_page/view_model/bal_view_model.dart | 15 +++- 2 files changed, 45 insertions(+), 44 deletions(-) diff --git a/lib/data/services/api_client.dart b/lib/data/services/api_client.dart index 2d9e273..3bc766a 100644 --- a/lib/data/services/api_client.dart +++ b/lib/data/services/api_client.dart @@ -327,13 +327,12 @@ class ApiClient { } Future> getCurrentBal() async { + final url = "https://$apiBasePath/bal/current"; + log.i("Fetching: getCurrentBal ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/bal/current"), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body); return Result.ok(Bal.fromJSON(json)); @@ -356,13 +355,12 @@ class ApiClient { */ Future> getBookById(int id) async { + final url = "https://$apiBasePath/book/id/${id.toString()}"; + log.i("Fetching: getBookById ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/book/id/${id.toString()}"), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body); return Result.ok(Book.fromJSON(json)); @@ -377,13 +375,12 @@ class ApiClient { } Future> getBookByEAN(String ean) async { + final url = "https://$apiBasePath/book/ean/$ean"; + log.i("Fetching: getBookByEan ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/book/ean/$ean"), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body); return Result.ok(Book.fromJSON(json)); @@ -408,25 +405,24 @@ class ApiClient { String title, String author, ) async { + final url = "https://$apiBasePath/bal/${balId.toString()}/search"; + log.i("Fetching: getBookInstancesBySearch ($url)"); final client = Client(); try { final headers = await _getHeaders({"Content-Type": "application/json"}); final body = jsonEncode({"title": title, "author": author}); - debugPrint("\n\n\n\n$body\n\n\n\n"); final response = await client.post( - Uri.parse("https://$apiBasePath/bal/${balId.toString()}/search"), + Uri.parse(url), headers: headers, body: body, ); if (response.statusCode == 200) { final json = jsonDecode(response.body) as List; - debugPrint("\n\n\n\nJSON : $json\n\n\n\n"); return Result.ok(json.map((el) => SearchResult.fromJSON(el)).toList()); } else { throw "Unknown Error"; } } catch (e) { - debugPrint("\n\n\n\nERROR: ${e.toString()}\n\n\n\n"); return Result.error(Exception("API $e")); } finally { client.close(); @@ -437,15 +433,13 @@ class ApiClient { int balId, int ean, ) async { + final url = + "https://$apiBasePath/bal/${balId.toString()}/ean/${ean.toString()}/book_instances"; + log.i("Fetching: getBookInstancesByEan ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse( - "https://$apiBasePath/bal/${balId.toString()}/ean/${ean.toString()}/book_instances", - ), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body) as List; return Result.ok(json.map((el) => BookInstance.fromJSON(el)).toList()); @@ -460,14 +454,14 @@ class ApiClient { } Future> sellBooks(Map books) async { + final url = "https://$apiBasePath/book_instance/sell/bulk"; + log.i("Fetching: sellBooks ($url)"); 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"), + Uri.parse(url), headers: headers, body: body, ); @@ -477,7 +471,6 @@ class ApiClient { throw "Unknown error"; } } catch (e) { - debugPrint("\n\n\n\nERROR : ${e.toString()}\n\n\n\n"); return Result.error(Exception(e)); } finally { client.close(); @@ -490,6 +483,8 @@ class ApiClient { Bal bal, double price, ) async { + final url = "https://$apiBasePath/book_instance"; + log.i("Fetching: sendBook ($url)"); final client = Client(); try { final headers = await _getHeaders({"Content-Type": "application/json"}); @@ -500,7 +495,7 @@ class ApiClient { "price": price, }); final response = await client.post( - Uri.parse("https://$apiBasePath/book_instance"), + Uri.parse(url), headers: headers, body: body, ); @@ -526,13 +521,12 @@ class ApiClient { */ Future> getOwnerById(int id) async { + final url = "https://$apiBasePath/owner/${id.toString()}"; + log.i("Fetching: getOwnerById ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/owner/${id.toString()}"), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body); return Result.ok(Owner.fromJSON(json)); @@ -547,13 +541,12 @@ class ApiClient { } Future> getSectionOwner() async { + final url = "https://$apiBasePath/owner/self"; + log.i("Fetching: getSectionOwner ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/owner/self"), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body); return Result.ok(Owner.fromJSON(json)); @@ -569,13 +562,12 @@ class ApiClient { /// Call on `/owners` to get a list of all [Owner]s Future>> getOwners() async { + final url = "https://$apiBasePath/owners"; + log.i("Fetching: getOwners ($url)"); final client = Client(); try { final headers = await _getHeaders(); - final response = await client.get( - Uri.parse("https://$apiBasePath/owners"), - headers: headers, - ); + final response = await client.get(Uri.parse(url), headers: headers); if (response.statusCode == 200) { final json = jsonDecode(response.body) as List; return Result.ok( @@ -597,6 +589,8 @@ class ApiClient { String lastName, String contact, ) async { + final url = "https://$apiBasePath/owner"; + log.i("Fetching: addOwner ($url)"); final client = Client(); try { final headers = await _getHeaders({"Content-Type": "application/json"}); @@ -606,7 +600,7 @@ class ApiClient { "contact": contact, }; final response = await client.post( - Uri.parse("https://$apiBasePath/owner"), + Uri.parse(url), headers: headers, body: jsonEncode(body), ); diff --git a/lib/ui/bal_page/view_model/bal_view_model.dart b/lib/ui/bal_page/view_model/bal_view_model.dart index f9947f1..ad5e47d 100644 --- a/lib/ui/bal_page/view_model/bal_view_model.dart +++ b/lib/ui/bal_page/view_model/bal_view_model.dart @@ -36,14 +36,24 @@ class BalViewModel extends ChangeNotifier { bool isABalOngoing = false; Future> stopBal(int id) async { + isLoaded = false; + notifyListeners(); final result = await _balRepository.stopBal(id); switch (result) { case Ok(): _bal = result.value; - notifyListeners(); break; default: } + final result2 = await _loadEnded(); + switch (result2) { + case Ok(): + isLoaded = true; + break; + case Error(): + break; + } + notifyListeners(); return result; } @@ -149,16 +159,13 @@ class BalViewModel extends ChangeNotifier { default: break; } - debugPrint("$isLoaded"); if (_bal?.state == BalState.ended) { final result2 = await _loadEnded(); - debugPrint("Hello"); switch (result2) { case Ok(): isLoaded = true; break; case Error(): - debugPrint("No ${result2.error}"); break; } } From ec26aa873cbe488572123d2e8519ac8d23b7fb5f Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 13:21:05 +0200 Subject: [PATCH 20/29] fix: ux on decimals --- lib/ui/sell_page/widgets/sell_page.dart | 14 ++++++++++---- 1 file changed, 10 insertions(+), 4 deletions(-) diff --git a/lib/ui/sell_page/widgets/sell_page.dart b/lib/ui/sell_page/widgets/sell_page.dart index 0ae364e..ee151e7 100644 --- a/lib/ui/sell_page/widgets/sell_page.dart +++ b/lib/ui/sell_page/widgets/sell_page.dart @@ -122,7 +122,6 @@ class _SellPageState extends State { controller: price, decoration: InputDecoration( labelText: "Argent reçu", - hintText: "Les décimales sont avec des .", helperText: "L'argent reçu sera réparti automatiquement.", suffixText: "€", @@ -140,7 +139,10 @@ class _SellPageState extends State { children: [ IconButton( onPressed: () { - if (double.tryParse(price.text) == null) { + if (double.tryParse( + price.text.replaceFirst(",", "."), + ) == + null) { ScaffoldMessenger.of(context).showSnackBar( SnackBar( content: Text( @@ -150,7 +152,9 @@ class _SellPageState extends State { ), ); return; - } else if (double.parse(price.text) < + } else if (double.parse( + price.text.replaceFirst(",", "."), + ) < widget.viewModel.minimumAmount) { ScaffoldMessenger.of(context).showSnackBar( SnackBar( @@ -163,7 +167,9 @@ class _SellPageState extends State { return; } widget.viewModel.sendSell( - double.parse(price.text), + double.parse( + price.text.replaceFirst(",", "."), + ), ); ScaffoldMessenger.of(context).showSnackBar( SnackBar( From 50074890c0024c8782109925c01a7e7e029d5ee9 Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 13:27:37 +0200 Subject: [PATCH 21/29] fix: another try at displaying comma on ios keyboard --- lib/ui/add_page/widgets/confirmation_popup.dart | 1 + lib/ui/add_page/widgets/form_popup.dart | 1 + lib/ui/sell_page/widgets/sell_page.dart | 1 + 3 files changed, 3 insertions(+) diff --git a/lib/ui/add_page/widgets/confirmation_popup.dart b/lib/ui/add_page/widgets/confirmation_popup.dart index 980ddee..776dc5b 100644 --- a/lib/ui/add_page/widgets/confirmation_popup.dart +++ b/lib/ui/add_page/widgets/confirmation_popup.dart @@ -82,6 +82,7 @@ class _ConfirmationPopupState extends State { ), keyboardType: TextInputType.numberWithOptions( decimal: true, + signed: true, ), validator: (value) { if (value == null || value.isEmpty) { diff --git a/lib/ui/add_page/widgets/form_popup.dart b/lib/ui/add_page/widgets/form_popup.dart index 59368e0..2a3f1de 100644 --- a/lib/ui/add_page/widgets/form_popup.dart +++ b/lib/ui/add_page/widgets/form_popup.dart @@ -279,6 +279,7 @@ class _FullyManualState extends State<_FullyManual> { ), keyboardType: TextInputType.numberWithOptions( decimal: true, + signed: true, ), validator: (value) { if (value == null || value.isEmpty) { diff --git a/lib/ui/sell_page/widgets/sell_page.dart b/lib/ui/sell_page/widgets/sell_page.dart index ee151e7..e7832b1 100644 --- a/lib/ui/sell_page/widgets/sell_page.dart +++ b/lib/ui/sell_page/widgets/sell_page.dart @@ -129,6 +129,7 @@ class _SellPageState extends State { ), keyboardType: TextInputType.numberWithOptions( decimal: true, + signed: true, ), ), ), From ce7338db2a05f15fc1d8f4fc798b50741091e076 Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 14:12:45 +0200 Subject: [PATCH 22/29] fix: didn't check for empty sell --- lib/ui/sell_page/widgets/sell_page.dart | 11 +++++++++++ 1 file changed, 11 insertions(+) diff --git a/lib/ui/sell_page/widgets/sell_page.dart b/lib/ui/sell_page/widgets/sell_page.dart index e7832b1..b04ae81 100644 --- a/lib/ui/sell_page/widgets/sell_page.dart +++ b/lib/ui/sell_page/widgets/sell_page.dart @@ -140,6 +140,17 @@ class _SellPageState extends State { children: [ IconButton( onPressed: () { + if (widget.viewModel.scannedBooks.isEmpty) { + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text( + "La vente doit comporter au moins un livre", + ), + behavior: SnackBarBehavior.floating, + ), + ); + return; + } if (double.tryParse( price.text.replaceFirst(",", "."), ) == From 8188fc61c99857994ec18d92a4cb27cb3bba99cc Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 14:13:08 +0200 Subject: [PATCH 23/29] fix: worflow mistake --- .forgejo/workflows/deploy.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.forgejo/workflows/deploy.yml b/.forgejo/workflows/deploy.yml index 3555aa2..7008872 100644 --- a/.forgejo/workflows/deploy.yml +++ b/.forgejo/workflows/deploy.yml @@ -15,4 +15,4 @@ jobs: cache: true - run: git config --global --add safe.directory /opt/hostedtoolcache/flutter/stable-3.35.1-x64 - run: flutter build web --release --wasm --base-href /app/ - - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} + - run: sshpass -p "${{ secrets.DEPLOY_PASSWORD }}" scp -o StrictHostKeyChecking=accept-new -rp build/web/* ${{ secrets.DEPLOY_USERNAME }}@${{ secrets.DEPLOY_ADDRESS }}:${{ secrets.DEPLOY_PATH }} From 0d1b5ce68ec31d3d68fcb06d3f17f2c20e82765e Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 14:28:38 +0200 Subject: [PATCH 24/29] feat: added the number of returns left to manage --- lib/ui/bal_page/widget/ended/bal_ended_screen.dart | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/lib/ui/bal_page/widget/ended/bal_ended_screen.dart b/lib/ui/bal_page/widget/ended/bal_ended_screen.dart index 70a9905..6f5043c 100644 --- a/lib/ui/bal_page/widget/ended/bal_ended_screen.dart +++ b/lib/ui/bal_page/widget/ended/bal_ended_screen.dart @@ -35,7 +35,10 @@ class _BalEndedScreenState extends State controller: tabController, tabs: [ Tab(text: "Statistiques"), - Tab(text: "À rendre"), + Tab( + text: + "À rendre (${widget.viewModel.owedToOwners?.length.toString() ?? "0"})", + ), ], ), ), From 6359efa0c336939f01534c531f5ed7bb0a1c1d93 Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 15:22:23 +0200 Subject: [PATCH 25/29] fix: made better and error managment on api client --- lib/data/services/api_client.dart | 169 +++++++++++++++++++----------- lib/routing/router.dart | 5 - 2 files changed, 106 insertions(+), 68 deletions(-) diff --git a/lib/data/services/api_client.dart b/lib/data/services/api_client.dart index 3bc766a..631c406 100644 --- a/lib/data/services/api_client.dart +++ b/lib/data/services/api_client.dart @@ -1,7 +1,6 @@ import 'dart:convert'; -import 'package:flutter/rendering.dart'; -import 'package:flutter/widgets.dart'; +import 'package:flutter/material.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:http/http.dart'; import 'package:logger/logger.dart'; @@ -312,14 +311,18 @@ class ApiClient { try { final headers = await _getHeaders(); final response = await client.get(Uri.parse(url), headers: headers); - if (response.statusCode == 200) { - final json = jsonDecode(response.body) as List; - return Result.ok(json.map((element) => Bal.fromJSON(element)).toList()); - } else { - throw Exception("Something wrong happened"); + + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body) as List; + return Result.ok( + json.map((element) => Bal.fromJSON(element)).toList(), + ); + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { - debugPrint("ERROR: ${e.toString()}"); + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -361,13 +364,17 @@ class ApiClient { try { final headers = await _getHeaders(); final response = await client.get(Uri.parse(url), headers: headers); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Book.fromJSON(json)); - } else { - throw Exception("The book was not found"); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Book.fromJSON(json)); + case 404: + throw "No book with this id exists in database"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception("API $e")); } finally { client.close(); @@ -381,13 +388,17 @@ class ApiClient { try { final headers = await _getHeaders(); final response = await client.get(Uri.parse(url), headers: headers); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Book.fromJSON(json)); - } else { - throw Exception("The book was not found"); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Book.fromJSON(json)); + case 404: + throw "No book with this EAN found in the database of BNF"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception("API $e")); } finally { client.close(); @@ -416,13 +427,19 @@ class ApiClient { headers: headers, body: body, ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body) as List; - return Result.ok(json.map((el) => SearchResult.fromJSON(el)).toList()); - } else { - throw "Unknown Error"; + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body) as List; + return Result.ok( + json.map((el) => SearchResult.fromJSON(el)).toList(), + ); + case 403: + throw "You do not own the BAL"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception("API $e")); } finally { client.close(); @@ -440,13 +457,19 @@ class ApiClient { try { final headers = await _getHeaders(); final response = await client.get(Uri.parse(url), headers: headers); - if (response.statusCode == 200) { - final json = jsonDecode(response.body) as List; - return Result.ok(json.map((el) => BookInstance.fromJSON(el)).toList()); - } else { - throw "Unknown Error"; + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body) as List; + return Result.ok( + json.map((el) => BookInstance.fromJSON(el)).toList(), + ); + case 403: + throw "You do not own the BAL"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception("API $e")); } finally { client.close(); @@ -465,12 +488,20 @@ class ApiClient { headers: headers, body: body, ); - if (response.statusCode == 200) { - return Result.ok(response.statusCode); - } else { - throw "Unknown error"; + switch (response.statusCode) { + case 200: + return Result.ok(response.statusCode); + case 403: + throw "You don't own one of the specified books"; + case 404: + throw "One of the books was not found"; + case 409: + throw "One of the books wasn't available"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -499,13 +530,14 @@ class ApiClient { headers: headers, body: body, ); - if (response.statusCode == 201) { - final json = jsonDecode(response.body); - return Result.ok(BookInstance.fromJSON(json)); - } else if (response.statusCode == 403) { - throw Exception("You don't own that book instance"); - } else { - throw Exception("Something wrong happened"); + switch (response.statusCode) { + case 201: + final json = jsonDecode(response.body); + return Result.ok(BookInstance.fromJSON(json)); + case 403: + throw "You don't own that book instance"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { return Result.error(Exception(e)); @@ -527,13 +559,19 @@ class ApiClient { try { final headers = await _getHeaders(); final response = await client.get(Uri.parse(url), headers: headers); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Owner.fromJSON(json)); - } else { - throw Exception("The owner was not found"); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Owner.fromJSON(json)); + case 403: + throw "You do not own specified owner"; + case 404: + throw "No owner with this id exists"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception("API $e")); } finally { client.close(); @@ -547,13 +585,15 @@ class ApiClient { try { final headers = await _getHeaders(); final response = await client.get(Uri.parse(url), headers: headers); - if (response.statusCode == 200) { - final json = jsonDecode(response.body); - return Result.ok(Owner.fromJSON(json)); - } else { - throw "Unknown error"; + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body); + return Result.ok(Owner.fromJSON(json)); + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -568,13 +608,14 @@ class ApiClient { try { final headers = await _getHeaders(); final response = await client.get(Uri.parse(url), headers: headers); - if (response.statusCode == 200) { - final json = jsonDecode(response.body) as List; - return Result.ok( - json.map((element) => Owner.fromJSON(element)).toList(), - ); - } else { - throw Exception("Invalid request"); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body) as List; + return Result.ok( + json.map((element) => Owner.fromJSON(element)).toList(), + ); + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } on Exception catch (error) { return Result.error(error); @@ -604,14 +645,16 @@ class ApiClient { headers: headers, body: jsonEncode(body), ); - if (response.statusCode == 201) { - final json = jsonDecode(response.body); - return Result.ok(Owner.fromJSON(json)); - } else { - throw Exception("Invalid request"); + switch (response.statusCode) { + case 201: + final json = jsonDecode(response.body); + return Result.ok(Owner.fromJSON(json)); + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } - } on Exception catch (error) { - return Result.error(error); + } catch (e) { + log.e(e.toString()); + return Result.error(Exception(e)); } finally { client.close(); } diff --git a/lib/routing/router.dart b/lib/routing/router.dart index c330f51..3c3cdb8 100644 --- a/lib/routing/router.dart +++ b/lib/routing/router.dart @@ -72,11 +72,6 @@ GoRouter router(AuthRepository authRepository) => GoRouter( ); return NoTransitionPage(child: AddPage(viewModel: viewModel)); }, - // routes: [ - // GoRoute(path: Routes.addForm), - // GoRoute(path: Routes.addOwner), - // GoRoute(path: Routes.addPrice), - // ], ), GoRoute( path: Routes.sell, From 59e1c2558cc4ea3ebce145659456804e6a3a5aaa Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Wed, 20 Aug 2025 18:20:37 +0200 Subject: [PATCH 26/29] fix: completed better error managment for all api clients --- lib/data/services/api_client.dart | 3 +- lib/data/services/auth_client.dart | 82 +++++++++++++++---------- lib/data/services/websocket_client.dart | 17 +++-- 3 files changed, 65 insertions(+), 37 deletions(-) diff --git a/lib/data/services/api_client.dart b/lib/data/services/api_client.dart index 631c406..0310bea 100644 --- a/lib/data/services/api_client.dart +++ b/lib/data/services/api_client.dart @@ -1,6 +1,5 @@ import 'dart:convert'; -import 'package:flutter/material.dart'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:http/http.dart'; import 'package:logger/logger.dart'; @@ -38,7 +37,7 @@ class ApiClient { ); Future _initStore() async { - _secureStorage ??= const FlutterSecureStorage(aOptions: AndroidOptions()); + _secureStorage ??= const FlutterSecureStorage(); } Future> _getHeaders([ diff --git a/lib/data/services/auth_client.dart b/lib/data/services/auth_client.dart index 4d77b4f..8b5653b 100644 --- a/lib/data/services/auth_client.dart +++ b/lib/data/services/auth_client.dart @@ -1,29 +1,39 @@ import 'dart:convert'; +import 'dart:ffi'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; +import 'package:logger/logger.dart'; import 'package:seshat/config/constants.dart'; import 'package:seshat/utils/result.dart'; -import "package:http/http.dart" as http; +import "package:http/http.dart"; class AuthClient { AuthClient(); FlutterSecureStorage? _secureStorage; + Logger log = Logger( + printer: PrettyPrinter( + colors: true, + lineLength: 100, + methodCount: 0, + dateTimeFormat: DateTimeFormat.dateAndTime, + ), + ); Future _initStore() async { - _secureStorage ??= const FlutterSecureStorage( - aOptions: AndroidOptions(encryptedSharedPreferences: true), - ); + _secureStorage ??= const FlutterSecureStorage(); } Future> hasValidToken() async { + final url = "https://$apiBasePath/token-check"; + log.i("Fetching: hasValidToken ($url)"); + final client = Client(); try { await _initStore(); bool hasToken = await _secureStorage!.containsKey(key: "token"); if (hasToken) { var token = await _secureStorage!.read(key: "token"); - var url = Uri.parse("https://$apiBasePath/token-check"); - var response = await http.post( - url, + var response = await client.post( + Uri.parse(url), headers: {"Content-Type": "application/json"}, body: jsonEncode({"token": token}), ); @@ -34,33 +44,41 @@ class AuthClient { } return Result.ok(false); } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); + } finally { + client.close(); } } Future> login(String username, String password) async { - var client = http.Client(); + final url = "https://$apiBasePath/auth"; + log.i("Logging in: $url"); + var client = Client(); try { await _initStore(); - var url = Uri.parse("https://$apiBasePath/auth"); var response = await client.post( - url, - headers: { - "Content-Type": "application/json", - "Accept": "application/json", - }, + Uri.parse(url), + headers: {"Content-Type": "application/json"}, body: jsonEncode({"password": password, "username": username}), ); - if (response.statusCode == 200) { - var json = jsonDecode(response.body); - await _secureStorage!.write(key: "token", value: json["access_token"]); - return Result.ok(json["access_token"]); - } else if (response.statusCode == 401) { - return Result.error(Exception("Wrong credentials")); - } else { - return Result.error(Exception("Token creation error")); + switch (response.statusCode) { + case 200: + var json = jsonDecode(response.body); + await _secureStorage!.write( + key: "token", + value: json["access_token"], + ); + return Result.ok(json["access_token"]); + case 401: + throw "Wrong credentials"; + case 500: + throw "Token creation error"; + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); @@ -68,18 +86,20 @@ class AuthClient { } Future> getRemoteApiVersion() async { - final client = http.Client(); + final url = "https://$apiBasePath/version"; + log.i("Fetching: getRemoteApiVersion ($url)"); + final client = Client(); try { - final response = await client.get( - Uri.parse("https://$apiBasePath/version"), - ); - if (response.statusCode == 200) { - final json = jsonDecode(response.body) as int; - return Result.ok(json); - } else { - throw "Something wrong happened"; + final response = await client.get(Uri.parse(url)); + switch (response.statusCode) { + case 200: + final json = jsonDecode(response.body) as int; + return Result.ok(json); + default: + throw "Unknown error with code ${response.statusCode.toString()}"; } } catch (e) { + log.e(e.toString()); return Result.error(Exception(e)); } finally { client.close(); diff --git a/lib/data/services/websocket_client.dart b/lib/data/services/websocket_client.dart index f8df18a..89cf8b9 100644 --- a/lib/data/services/websocket_client.dart +++ b/lib/data/services/websocket_client.dart @@ -2,6 +2,7 @@ import 'dart:async'; import 'dart:convert'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; +import 'package:logger/logger.dart'; import 'package:rxdart/rxdart.dart'; import 'package:seshat/config/constants.dart'; import 'package:seshat/domain/models/owner.dart'; @@ -15,20 +16,28 @@ class WebsocketClient { sync: true, ); late final StreamSubscription sub; + Logger log = Logger( + printer: PrettyPrinter( + colors: true, + lineLength: 100, + methodCount: 0, + dateTimeFormat: DateTimeFormat.dateAndTime, + ), + ); Stream get owners => _ownersController.stream; Future _initStore() async { - _secureStorage ??= const FlutterSecureStorage( - aOptions: AndroidOptions(encryptedSharedPreferences: true), - ); + _secureStorage ??= const FlutterSecureStorage(); } Future connect() async { + final url = "wss://$apiBasePath/ws"; + log.i("Webocket: $url"); await _initStore(); if (_channel != null) return; - _channel = WebSocketChannel.connect(Uri.parse("wss://$apiBasePath/ws")); + _channel = WebSocketChannel.connect(Uri.parse(url)); await _channel!.ready; var token = await _secureStorage!.read(key: "token"); From dad000a1b911967a5138c289425651eaabd011d5 Mon Sep 17 00:00:00 2001 From: alzalia1 Date: Sat, 23 Aug 2025 12:35:36 +0200 Subject: [PATCH 27/29] fix: continuing error managment and documentation --- lib/data/repositories/auth_repository.dart | 6 +- lib/data/repositories/bal_repository.dart | 70 ++++++++---- .../book_instance_repository.dart | 9 +- lib/data/repositories/book_repository.dart | 7 +- lib/data/repositories/owner_repository.dart | 31 +++-- lib/data/services/api_client.dart | 91 ++++++++++----- lib/data/services/auth_client.dart | 8 +- lib/data/services/websocket_client.dart | 17 ++- lib/routing/router.dart | 2 +- .../add_page/view_model/add_view_model.dart | 48 ++++++-- lib/ui/add_page/widgets/add_page.dart | 2 +- .../add_page/widgets/confirmation_popup.dart | 6 +- lib/ui/auth/viewmodel/login_view_model.dart | 5 + .../bal_page/view_model/bal_view_model.dart | 81 ++++++++----- lib/ui/bal_page/widget/bal_page.dart | 4 +- .../widget/ended/bal_ended_screen.dart | 2 +- .../widget/ongoing/bal_ongoing_screen.dart | 6 +- .../widget/pending/bal_pending_screen.dart | 14 ++- .../home_page/view_model/home_view_model.dart | 22 +++- lib/ui/home_page/widgets/home_page.dart | 8 +- .../sell_page/view_model/sell_view_model.dart | 107 ++++++++++++------ lib/ui/sell_page/widgets/scan_screen.dart | 2 +- .../sell_page/widgets/sell_choice_popup.dart | 4 +- lib/ui/sell_page/widgets/sell_page.dart | 19 ++-- 24 files changed, 389 insertions(+), 182 deletions(-) diff --git a/lib/data/repositories/auth_repository.dart b/lib/data/repositories/auth_repository.dart index 2b56ddb..c55d564 100644 --- a/lib/data/repositories/auth_repository.dart +++ b/lib/data/repositories/auth_repository.dart @@ -2,6 +2,7 @@ import 'package:flutter/foundation.dart'; import 'package:seshat/data/services/auth_client.dart'; import 'package:seshat/utils/result.dart'; +/// Repository to manage authentification class AuthRepository extends ChangeNotifier { AuthRepository({required AuthClient authClient}) : _authClient = authClient; @@ -9,6 +10,7 @@ class AuthRepository extends ChangeNotifier { bool? _isAuthenticated; + /// Checks the validity of the token if not already checked Future get isLoggedIn async { if (_isAuthenticated != null) { return _isAuthenticated!; @@ -25,6 +27,7 @@ class AuthRepository extends ChangeNotifier { } } + /// Logs in the user Future> login(String username, String password) async { try { final result = await _authClient.login(username, password); @@ -33,13 +36,14 @@ class AuthRepository extends ChangeNotifier { _isAuthenticated = true; return Result.ok(()); case Error(): - return Result.error(result.error); + return result; } } catch (e) { return Result.error(Exception(e)); } } + /// Gets the API's remote version Future> getRemoteApiVersion() async { return await _authClient.getRemoteApiVersion(); } diff --git a/lib/data/repositories/bal_repository.dart b/lib/data/repositories/bal_repository.dart index ccce2e4..97bc023 100644 --- a/lib/data/repositories/bal_repository.dart +++ b/lib/data/repositories/bal_repository.dart @@ -5,13 +5,19 @@ import 'package:seshat/domain/models/bal_stats.dart'; import 'package:seshat/domain/models/enums.dart'; import 'package:seshat/utils/result.dart'; +/// Repository to manage [Bal] class BalRepository { BalRepository({required ApiClient apiClient}) : _apiClient = apiClient; final ApiClient _apiClient; - List? _bals; - Accounting? accounting; + /// [List] of all the user's [Bal] + List? _bals; + + /// [Accounting] of [Bal], mapped by [Bal] id + final Map _accountingMap = {}; + + /// Gets a list of all [Bal] from cache or remote Future>> getBals() async { if (_bals != null) { return Result.ok(_bals!); @@ -26,6 +32,7 @@ class BalRepository { } } + /// Gets a list of all [Bal] from remote only Future>> _getBalsNoCache() async { final result = await _apiClient.getBals(); switch (result) { @@ -37,15 +44,16 @@ class BalRepository { } } - Future> balById(int id) async { + /// Gets a [Bal] by [balId], either from cache or remote + Future> balById(int balId) async { if (_bals == null) { await getBals(); } - Bal? bal = _bals!.where((bal) => bal.id == id).firstOrNull; + Bal? bal = _bals!.where((bal) => bal.id == balId).firstOrNull; if (bal != null) { return Result.ok(bal); } - final result = await _apiClient.getBalById(id); + final result = await _apiClient.getBalById(balId); switch (result) { case Ok(): return Result.ok(result.value); @@ -54,11 +62,13 @@ class BalRepository { } } + /// Return wether or not a [Bal] is currently [BalState.ongoing] bool isABalOngoing() { return _bals?.where((bal) => bal.state == BalState.ongoing).isNotEmpty ?? false; } + /// Gets the [Bal] that is [BalState.ongoing] Future ongoingBal() async { if (_bals == null) { await _getBalsNoCache(); @@ -66,12 +76,14 @@ class BalRepository { return _bals!.where((bal) => bal.state == BalState.ongoing).firstOrNull; } + /// Stops a [Bal] and refresh cache Future> stopBal(int id) async { final result = await _apiClient.stopBal(id); _getBalsNoCache(); return result; } + /// Starts a [Bal] and refresh cache Future> startBal(int id) async { if (isABalOngoing()) { return Result.error( @@ -83,52 +95,62 @@ class BalRepository { return result; } + /// Changes a [Bal]'s [name], [startTime] or [endTime] Future> editBal( int id, String name, - DateTime start, - DateTime end, + DateTime startTime, + DateTime endTime, ) async { - final result = await _apiClient.editBal(id, name, start, end); + final result = await _apiClient.editBal(id, name, startTime, endTime); await _getBalsNoCache(); return result; } - Future> addBal(String name, DateTime start, DateTime end) async { - final result = await _apiClient.addBal(name, start, end); + /// Creates a [Bal] from its [name], [startTime] and [endTime] + Future> addBal( + String name, + DateTime startTime, + DateTime endTime, + ) async { + final result = await _apiClient.addBal(name, startTime, endTime); await _getBalsNoCache(); return result; } - Future> getBalStats(int id) async { - return _apiClient.getBalStats(id); + /// Gets a [BalStats] from its [balId] + Future> getBalStats(int balId) async { + return _apiClient.getBalStats(balId); } + /// Get [Accounting] of a [Bal] from remote only Future> getAccountingNoCache(int balId) async { final result = await _apiClient.getAccounting(balId); switch (result) { case Ok(): - accounting = result.value; + _accountingMap[balId] = result.value; break; default: } return result; } + /// Get [Accounting] of a [Bal] from cache or remote Future> getAccounting(int balId) async { - if (accounting != null) { - return Result.ok(accounting!); + if (_accountingMap[balId] != null) { + return Result.ok(_accountingMap[balId]!); } final result = await _apiClient.getAccounting(balId); switch (result) { case Ok(): - accounting = result.value; + _accountingMap[balId] = result.value; break; default: } return result; } + /// Manages what returning (of type [ReturnType]) does to cache and notifies remote Future> returnToId( int balId, int ownerId, @@ -139,26 +161,32 @@ class BalRepository { case Ok(): switch (type) { case ReturnType.books: - final owner = accounting?.owners + final owner = _accountingMap[balId]?.owners .where((el) => el.ownerId == ownerId) .firstOrNull; if (owner?.owedMoney == 0) { - accounting?.owners.removeWhere((el) => el.ownerId == ownerId); + _accountingMap[balId]?.owners.removeWhere( + (el) => el.ownerId == ownerId, + ); } owner?.owed = []; owner?.owedInstances = []; break; case ReturnType.money: - final owner = accounting?.owners + final owner = _accountingMap[balId]?.owners .where((el) => el.ownerId == ownerId) .firstOrNull; if (owner?.owed == null || owner!.owed.isEmpty) { - accounting?.owners.removeWhere((el) => el.ownerId == ownerId); + _accountingMap[balId]?.owners.removeWhere( + (el) => el.ownerId == ownerId, + ); } owner?.owedMoney = 0; break; case ReturnType.all: - accounting?.owners.removeWhere((el) => el.ownerId == ownerId); + _accountingMap[balId]?.owners.removeWhere( + (el) => el.ownerId == ownerId, + ); break; } break; diff --git a/lib/data/repositories/book_instance_repository.dart b/lib/data/repositories/book_instance_repository.dart index 4a07167..7a60d19 100644 --- a/lib/data/repositories/book_instance_repository.dart +++ b/lib/data/repositories/book_instance_repository.dart @@ -6,16 +6,19 @@ import 'package:seshat/domain/models/owner.dart'; import 'package:seshat/domain/models/search_result.dart'; import 'package:seshat/utils/result.dart'; +/// Repository to manage [BookInstance] class BookInstanceRepository { BookInstanceRepository({required ApiClient apiClient}) : _apiClient = apiClient; final ApiClient _apiClient; + /// Gets a [List] from an [ean] Future>> getByEan(int balId, int ean) async { return await _apiClient.getBookInstanceByEAN(balId, ean); } + /// Gets a [List] from a [title] and [author] Future>> getBySearch( int balId, String title, @@ -24,15 +27,17 @@ class BookInstanceRepository { return await _apiClient.getBookInstanceBySearch(balId, title, author); } - Future> sendBook( + /// Sends a new [BookInstance]'s [book], [owner], [bal] and [price] + Future> sendNewBookInstance( Book book, Owner owner, Bal bal, double price, ) async { - return await _apiClient.sendBook(book, owner, bal, price); + return await _apiClient.sendNewBookInstance(book, owner, bal, price); } + /// Sells a [List] Future> sellBooks(List books) async { Map res = {}; for (BookInstance instance in books) { diff --git a/lib/data/repositories/book_repository.dart b/lib/data/repositories/book_repository.dart index 984d85b..5d38be1 100644 --- a/lib/data/repositories/book_repository.dart +++ b/lib/data/repositories/book_repository.dart @@ -2,16 +2,19 @@ import 'package:seshat/data/services/api_client.dart'; import 'package:seshat/domain/models/book.dart'; import 'package:seshat/utils/result.dart'; +/// Repository to manage [Book] class BookRepository { BookRepository({required ApiClient apiClient}) : _apiClient = apiClient; final ApiClient _apiClient; + /// Gets a [Book] by its [ean] Future> getBookByEAN(String ean) async { return await _apiClient.getBookByEAN(ean); } - Future> getBookById(int id) async { - return await _apiClient.getBookById(id); + /// Gets a [Book] by its [bookId] + Future> getBookById(int bookId) async { + return await _apiClient.getBookById(bookId); } } diff --git a/lib/data/repositories/owner_repository.dart b/lib/data/repositories/owner_repository.dart index 5020123..817bea0 100644 --- a/lib/data/repositories/owner_repository.dart +++ b/lib/data/repositories/owner_repository.dart @@ -5,6 +5,7 @@ import 'package:seshat/data/services/websocket_client.dart'; import 'package:seshat/domain/models/owner.dart'; import 'package:seshat/utils/result.dart'; +/// Repository to manage [Owner] class OwnerRepository { OwnerRepository({ required ApiClient apiClient, @@ -14,18 +15,25 @@ class OwnerRepository { final ApiClient _apiClient; final WebsocketClient _wsClient; - late final StreamSubscription sub; - List? _cachedOwners; - Owner? _sectionOwner; - Future> get sectionOwner async { - if (_sectionOwner != null) { - return Result.ok(_sectionOwner!); + /// [StreamSubscription] to the [Stream] for [_wsClient] + late final StreamSubscription sub; + + /// [List] of owners, updated by [_wsClient] + List? _cachedOwners; + + /// [Owner] of the current user + Owner? _ownerOfUser; + + /// [Owner] of the current user + Future> get ownerOfUser async { + if (_ownerOfUser != null) { + return Result.ok(_ownerOfUser!); } - final result = await _apiClient.getSectionOwner(); + final result = await _apiClient.getOwnerOfUser(); switch (result) { case Ok(): - _sectionOwner = result.value; + _ownerOfUser = result.value; break; default: break; @@ -33,16 +41,17 @@ class OwnerRepository { return result; } - Future> getOwnerById(int id) async { + /// Gets an [Owner] from its [ownerId] + Future> getOwnerById(int ownerId) async { if (_cachedOwners != null) { final result1 = _cachedOwners! - .where((owner) => owner.id == id) + .where((owner) => owner.id == ownerId) .firstOrNull; if (result1 != null) { return Result.ok(result1); } } - return await _apiClient.getOwnerById(id); + return await _apiClient.getOwnerById(ownerId); } /// Adds an [Owner] to the database, and gets the resulting [Owner]. diff --git a/lib/data/services/api_client.dart b/lib/data/services/api_client.dart index 0310bea..c638d4c 100644 --- a/lib/data/services/api_client.dart +++ b/lib/data/services/api_client.dart @@ -9,9 +9,9 @@ import 'package:seshat/domain/models/bal.dart'; import 'package:seshat/domain/models/bal_stats.dart'; import 'package:seshat/domain/models/book.dart'; import 'package:seshat/domain/models/book_instance.dart'; +import 'package:seshat/domain/models/enums.dart'; import 'package:seshat/domain/models/owner.dart'; import 'package:seshat/domain/models/search_result.dart'; -import 'package:seshat/utils/command.dart'; import 'package:seshat/utils/result.dart'; extension StringExtension on String { @@ -20,12 +20,17 @@ extension StringExtension on String { } } +/// API Client to manage all authenticated REST routes class ApiClient { - ApiClient({String? host, int? port}); + ApiClient(); - late final Command0 load; + /// JWT for registration String? token; + + /// Readiness of the API Client bool isReady = false; + + /// Storage to access JWT FlutterSecureStorage? _secureStorage; Logger log = Logger( printer: PrettyPrinter( @@ -36,10 +41,12 @@ class ApiClient { ), ); + /// Initializes connection to the [_secureStorage] Future _initStore() async { _secureStorage ??= const FlutterSecureStorage(); } + /// Generates authorization headers and option [additionalHeaders] Future> _getHeaders([ Map? additionalHeaders, ]) async { @@ -55,6 +62,7 @@ class ApiClient { * ======================== */ + /// Gets data about a BAL's needed returns Future> getAccounting(int balId) async { final url = "https://$apiBasePath/bal/$balId/accounting"; log.i("Fetching: getAccounting ($url)"); @@ -81,6 +89,8 @@ class ApiClient { } } + /// Notifies the server that either Books, Money or All has been returned to the user + /// [type] is the stringified version of [ReturnType] Future> returnToId(int balId, int ownerId, String type) async { final url = "https://$apiBasePath/bal/${balId.toString()}/accounting/return/${ownerId.toString()}"; @@ -120,8 +130,9 @@ class ApiClient { * ================= */ - Future> getBalStats(int id) async { - final url = "https://$apiBasePath/bal/${id.toString()}/stats"; + /// Get stats about a BAL's performance + Future> getBalStats(int balId) async { + final url = "https://$apiBasePath/bal/${balId.toString()}/stats"; log.i("Fetching: getBalStats ($url)"); final client = Client(); try { @@ -148,8 +159,9 @@ class ApiClient { } } - Future> stopBal(int id) async { - final url = "https://$apiBasePath/bal/${id.toString()}/stop"; + /// Stops a BAL, putting it's [BalState] to [BalState.ended] + Future> stopBal(int balId) async { + final url = "https://$apiBasePath/bal/${balId.toString()}/stop"; log.i("Fetching: stopBal ($url)"); final client = Client(); try { @@ -176,8 +188,9 @@ class ApiClient { } } - Future> startBal(int id) async { - final url = "https://$apiBasePath/bal/${id.toString()}/start"; + /// Starts a BAL, putting it's [BalState] to [BalState.ongoing] + Future> startBal(int balId) async { + final url = "https://$apiBasePath/bal/${balId.toString()}/start"; log.i("Fetching: startBal ($url)"); final client = Client(); try { @@ -204,21 +217,22 @@ class ApiClient { } } + /// Changes the information about a [Bal], such as its [name], [startTime] or [endTime] Future> editBal( - int id, + int balId, String name, - DateTime start, - DateTime end, + DateTime startTime, + DateTime endTime, ) async { - final url = "https://$apiBasePath/bal/${id.toString()}"; + final url = "https://$apiBasePath/bal/${balId.toString()}"; log.i("Fetching: editBal ($url)"); final client = Client(); try { final headers = await _getHeaders({"Content-Type": "application/json"}); final body = { "name": name, - "start_timestamp": (start.millisecondsSinceEpoch / 1000).round(), - "end_timestamp": (end.millisecondsSinceEpoch / 1000).round(), + "start_timestamp": (startTime.millisecondsSinceEpoch / 1000).round(), + "end_timestamp": (endTime.millisecondsSinceEpoch / 1000).round(), }; final response = await client.patch( Uri.parse(url), @@ -244,8 +258,9 @@ class ApiClient { } } - Future> getBalById(int id) async { - final url = "https://$apiBasePath/bal/${id.toString()}"; + /// Gets a [Bal] from it's [balId] + Future> getBalById(int balId) async { + final url = "https://$apiBasePath/bal/${balId.toString()}"; log.i("Fetching: getBalById ($url)"); final client = Client(); try { @@ -270,7 +285,12 @@ class ApiClient { } } - Future> addBal(String name, DateTime start, DateTime end) async { + /// Adds a [Bal] from it's [name], [startTime] and [endTime] + Future> addBal( + String name, + DateTime startTime, + DateTime endTime, + ) async { final url = "https://$apiBasePath/bal"; log.i("Fetching: addBal ($url)"); final client = Client(); @@ -278,8 +298,8 @@ class ApiClient { final headers = await _getHeaders({"Content-Type": "application/json"}); final body = { "name": name, - "start_timestamp": (start.millisecondsSinceEpoch / 1000).round(), - "end_timestamp": (end.millisecondsSinceEpoch / 1000).round(), + "start_timestamp": (startTime.millisecondsSinceEpoch / 1000).round(), + "end_timestamp": (endTime.millisecondsSinceEpoch / 1000).round(), }; final response = await client.post( Uri.parse(url), @@ -303,6 +323,7 @@ class ApiClient { } } + /// Gets a [List] of all [Bal] Future>> getBals() async { final url = "https://$apiBasePath/bals"; log.i("Fetching: getBals ($url)"); @@ -328,7 +349,8 @@ class ApiClient { } } - Future> getCurrentBal() async { + /// Gets the ongoing BAL for the user + Future> getOngoingBal() async { final url = "https://$apiBasePath/bal/current"; log.i("Fetching: getCurrentBal ($url)"); final client = Client(); @@ -356,8 +378,9 @@ class ApiClient { * =================== */ - Future> getBookById(int id) async { - final url = "https://$apiBasePath/book/id/${id.toString()}"; + /// Gets a [Book] by its [bookId] + Future> getBookById(int bookId) async { + final url = "https://$apiBasePath/book/id/${bookId.toString()}"; log.i("Fetching: getBookById ($url)"); final client = Client(); try { @@ -380,6 +403,7 @@ class ApiClient { } } + /// Gets a [Book] from its [ean] Future> getBookByEAN(String ean) async { final url = "https://$apiBasePath/book/ean/$ean"; log.i("Fetching: getBookByEan ($url)"); @@ -410,6 +434,7 @@ class ApiClient { * ============================= */ + /// Gets a [BookInstance] from it's [title], [author] or both Future>> getBookInstanceBySearch( int balId, String title, @@ -445,6 +470,7 @@ class ApiClient { } } + /// Gets a [BookInstance] from it's [ean] Future>> getBookInstanceByEAN( int balId, int ean, @@ -475,6 +501,10 @@ class ApiClient { } } + /// Notifies the server of the sell of multiple [BookInstance]. [books] is in the form of + /// ```dart + /// final books = {"aBookInstanceId": 6.0} // and its price + /// ``` Future> sellBooks(Map books) async { final url = "https://$apiBasePath/book_instance/sell/bulk"; log.i("Fetching: sellBooks ($url)"); @@ -507,7 +537,8 @@ class ApiClient { } } - Future> sendBook( + /// Creates a new [BookInstance] from it's [book], it's [owner], it's [bal] and it's [price] + Future> sendNewBookInstance( Book book, Owner owner, Bal bal, @@ -551,8 +582,9 @@ class ApiClient { * ==================== */ - Future> getOwnerById(int id) async { - final url = "https://$apiBasePath/owner/${id.toString()}"; + /// Gets an [Owner] by it's [ownerId] + Future> getOwnerById(int ownerId) async { + final url = "https://$apiBasePath/owner/${ownerId.toString()}"; log.i("Fetching: getOwnerById ($url)"); final client = Client(); try { @@ -577,7 +609,8 @@ class ApiClient { } } - Future> getSectionOwner() async { + /// Get the owner of the current user + Future> getOwnerOfUser() async { final url = "https://$apiBasePath/owner/self"; log.i("Fetching: getSectionOwner ($url)"); final client = Client(); @@ -599,7 +632,7 @@ class ApiClient { } } - /// Call on `/owners` to get a list of all [Owner]s + /// Get a [List] of all [Owner] Future>> getOwners() async { final url = "https://$apiBasePath/owners"; log.i("Fetching: getOwners ($url)"); @@ -623,7 +656,7 @@ class ApiClient { } } - /// Adds an owner to the database + /// Adds an [Owner] from its [firstName], [lastName] and [contact] Future> addOwner( String firstName, String lastName, diff --git a/lib/data/services/auth_client.dart b/lib/data/services/auth_client.dart index 8b5653b..3dc8496 100644 --- a/lib/data/services/auth_client.dart +++ b/lib/data/services/auth_client.dart @@ -1,5 +1,4 @@ import 'dart:convert'; -import 'dart:ffi'; import 'package:flutter_secure_storage/flutter_secure_storage.dart'; import 'package:logger/logger.dart'; @@ -7,8 +6,11 @@ import 'package:seshat/config/constants.dart'; import 'package:seshat/utils/result.dart'; import "package:http/http.dart"; +/// API Client to manage all unauthenticated REST routes class AuthClient { AuthClient(); + + /// Storage to access JWT FlutterSecureStorage? _secureStorage; Logger log = Logger( printer: PrettyPrinter( @@ -19,10 +21,12 @@ class AuthClient { ), ); + /// Initializes connection to the [_secureStorage] Future _initStore() async { _secureStorage ??= const FlutterSecureStorage(); } + /// Verifies the validity of the token stored in [_secureStorage] Future> hasValidToken() async { final url = "https://$apiBasePath/token-check"; log.i("Fetching: hasValidToken ($url)"); @@ -51,6 +55,7 @@ class AuthClient { } } + /// Logs a user in from its [username] and [password] Future> login(String username, String password) async { final url = "https://$apiBasePath/auth"; log.i("Logging in: $url"); @@ -85,6 +90,7 @@ class AuthClient { } } + /// Gets the API version of the server Future> getRemoteApiVersion() async { final url = "https://$apiBasePath/version"; log.i("Fetching: getRemoteApiVersion ($url)"); diff --git a/lib/data/services/websocket_client.dart b/lib/data/services/websocket_client.dart index 89cf8b9..1d85989 100644 --- a/lib/data/services/websocket_client.dart +++ b/lib/data/services/websocket_client.dart @@ -8,13 +8,23 @@ import 'package:seshat/config/constants.dart'; import 'package:seshat/domain/models/owner.dart'; import 'package:web_socket_channel/web_socket_channel.dart'; +/// API Client to manages connections to WebSockets class WebsocketClient { - WebSocketChannel? _channel; + /// Storage to access JWT FlutterSecureStorage? _secureStorage; + + /// Raw channel of data from WebSocket + WebSocketChannel? _channel; + + /// Global WebSocket Stream final BehaviorSubject _baseController = BehaviorSubject(); + + /// WebSocket Stream dedicated to [Owner] entries final BehaviorSubject _ownersController = BehaviorSubject( sync: true, ); + + /// Subscription to [_baseController] late final StreamSubscription sub; Logger log = Logger( printer: PrettyPrinter( @@ -25,12 +35,15 @@ class WebsocketClient { ), ); + /// Gets a stream of [Owner] Stream get owners => _ownersController.stream; + /// Initializes connection to the [_secureStorage] Future _initStore() async { _secureStorage ??= const FlutterSecureStorage(); } + /// Connects to the websocket Future connect() async { final url = "wss://$apiBasePath/ws"; log.i("Webocket: $url"); @@ -69,11 +82,13 @@ class WebsocketClient { } } + /// Disconnects from the websocket void _handleDisconnect() { sub.cancel(); _channel = null; } + /// Closes all connections void dispose() { sub.cancel(); _channel?.sink.close(); diff --git a/lib/routing/router.dart b/lib/routing/router.dart index 3c3cdb8..fd530ec 100644 --- a/lib/routing/router.dart +++ b/lib/routing/router.dart @@ -55,7 +55,7 @@ GoRouter router(AuthRepository authRepository) => GoRouter( pageBuilder: (context, state) { final viewModel = BalViewModel( balRepository: context.read(), - id: int.parse(state.pathParameters["id"] ?? ""), + selectedBalId: int.parse(state.pathParameters["id"] ?? ""), ownerRepository: context.read(), ); return NoTransitionPage(child: BalPage(viewModel: viewModel)); diff --git a/lib/ui/add_page/view_model/add_view_model.dart b/lib/ui/add_page/view_model/add_view_model.dart index 0e685ba..9334ea7 100644 --- a/lib/ui/add_page/view_model/add_view_model.dart +++ b/lib/ui/add_page/view_model/add_view_model.dart @@ -38,18 +38,29 @@ class AddViewModel extends ChangeNotifier { * ==================== */ + /// Owner currently selected in the ui Owner? _currentOwner; + + /// Owner currently selected in the ui Owner? get currentOwner => _currentOwner; set currentOwner(Owner? owner) { _currentOwner = owner; notifyListeners(); } - Owner? _sectionOwner; - Owner? get sectionOwner => _sectionOwner; + /// Owner of the current user + Owner? _ownerOfUser; + /// Owner of the current user + Owner? get ownerOfUser => _ownerOfUser; + + /// All the [Owner] List _owners = []; + + /// All the [Owner] List? get owners => _owners; + + /// Adds an owner from it's [firstName], [lastName] and [contact] Future> addOwner( String firstName, String lastName, @@ -85,8 +96,11 @@ class AddViewModel extends ChangeNotifier { * ================= */ - Bal? _currentBal; - Bal? get currentBal => _currentBal; + /// Ongoing [Bal] + Bal? _ongoingBal; + + /// Ongoing [Bal] + Bal? get ongoingBal => _ongoingBal; /* * =================== @@ -94,7 +108,10 @@ class AddViewModel extends ChangeNotifier { * =================== */ + /// Wether to ask for a price bool _askPrice = true; + + /// Wether to ask for a price bool get askPrice => _askPrice; set askPrice(bool newValue) { _askPrice = newValue; @@ -107,21 +124,26 @@ class AddViewModel extends ChangeNotifier { * ================================= */ - /// Sends an api request with a [bacorde], then gets the [Book] that was - /// either created or retrieved. Sens the [Book] back wrapped in a [Result]. + /// Retrieves the book associated with an ean through a [barcode] Future> scanBook(BarcodeCapture barcode) async { var ean = barcode.barcodes.first.rawValue!; var result = await _bookRepository.getBookByEAN(ean); return result; } - Future> sendBook( + /// Creates a new Book Instance from its [book], [owner], [bal] and [price] + Future> sendNewBookInstance( Book book, Owner owner, Bal bal, double price, ) async { - return await _bookInstanceRepository.sendBook(book, owner, bal, price); + return await _bookInstanceRepository.sendNewBookInstance( + book, + owner, + bal, + price, + ); } /* @@ -130,9 +152,11 @@ class AddViewModel extends ChangeNotifier { * ================================= */ + /// Command to load the view model late final Command0 load; bool isLoaded = false; + /// Manages the loaders Future> _load() async { final result1 = await _loadOwners(); switch (result1) { @@ -153,11 +177,12 @@ class AddViewModel extends ChangeNotifier { return result2; } + /// Loads all necessary data about [Bal]s Future> _loadBal() async { final result = await _balRepository.getBals(); switch (result) { case Ok(): - _currentBal = result.value + _ongoingBal = result.value .where((bal) => bal.state == BalState.ongoing) .firstOrNull; break; @@ -168,6 +193,7 @@ class AddViewModel extends ChangeNotifier { return result; } + /// Loads all the necessary data about [Owner]s Future> _loadOwners() async { final result = await _ownerRepository.getOwners(); switch (result) { @@ -182,10 +208,10 @@ class AddViewModel extends ChangeNotifier { return result; } - final result2 = await _ownerRepository.sectionOwner; + final result2 = await _ownerRepository.ownerOfUser; switch (result2) { case Ok(): - _sectionOwner = result2.value; + _ownerOfUser = result2.value; break; default: } diff --git a/lib/ui/add_page/widgets/add_page.dart b/lib/ui/add_page/widgets/add_page.dart index d66e827..4d22005 100644 --- a/lib/ui/add_page/widgets/add_page.dart +++ b/lib/ui/add_page/widgets/add_page.dart @@ -45,7 +45,7 @@ class _AddPageState extends State { listenable: widget.viewModel, builder: (context, child) => switch (widget.viewModel.isLoaded) { false => AwaitLoading(), - true => switch (widget.viewModel.currentBal) { + true => switch (widget.viewModel.ongoingBal) { null => Center( child: SizedBox( width: 300, diff --git a/lib/ui/add_page/widgets/confirmation_popup.dart b/lib/ui/add_page/widgets/confirmation_popup.dart index 776dc5b..e57f02c 100644 --- a/lib/ui/add_page/widgets/confirmation_popup.dart +++ b/lib/ui/add_page/widgets/confirmation_popup.dart @@ -123,10 +123,10 @@ class _ConfirmationPopupState extends State { _formKey.currentState!.save(); } - var result = await widget.viewModel.sendBook( + var result = await widget.viewModel.sendNewBookInstance( widget.book, widget.viewModel.currentOwner!, - widget.viewModel.currentBal!, + widget.viewModel.ongoingBal!, price, ); @@ -142,7 +142,7 @@ class _ConfirmationPopupState extends State { ), content: Text( (widget.viewModel.currentOwner!.id == - widget.viewModel.sectionOwner!.id) + widget.viewModel.ownerOfUser!.id) ? "Ce livre appartient à la section. Vous pouvez mettre le code, ou poser une gomette, ..." : "Identifiant propriétaire de ce livre. Pensez à l'écrire pour retrouver lae propriétaire du livre lors de la vente ou du retour !", ), diff --git a/lib/ui/auth/viewmodel/login_view_model.dart b/lib/ui/auth/viewmodel/login_view_model.dart index 190ea46..51179f0 100644 --- a/lib/ui/auth/viewmodel/login_view_model.dart +++ b/lib/ui/auth/viewmodel/login_view_model.dart @@ -13,8 +13,10 @@ class LoginViewModel extends ChangeNotifier { final AuthRepository _authRepository; + /// Command to login with added capabilities late Command1 login; + /// Logins the user with credentials [(String username, String password)] Future> _login((String, String) credentials) async { final (username, password) = credentials; final result = await _authRepository.login(username, password); @@ -27,10 +29,12 @@ class LoginViewModel extends ChangeNotifier { * ================================= */ + /// Loads all necessary data late final Command0 load; bool isLoaded = false; bool isUpToDate = false; + /// Manages loaders Future> _load() async { final result1 = await _loadApiVersion(); switch (result1) { @@ -44,6 +48,7 @@ class LoginViewModel extends ChangeNotifier { return result1; } + /// Loads the current remote api version and compares to local hardcoded [apiVersion] Future> _loadApiVersion() async { final result = await _authRepository.getRemoteApiVersion(); switch (result) { diff --git a/lib/ui/bal_page/view_model/bal_view_model.dart b/lib/ui/bal_page/view_model/bal_view_model.dart index ad5e47d..c3cdbb9 100644 --- a/lib/ui/bal_page/view_model/bal_view_model.dart +++ b/lib/ui/bal_page/view_model/bal_view_model.dart @@ -14,7 +14,7 @@ import 'package:seshat/utils/result.dart'; class BalViewModel extends ChangeNotifier { BalViewModel({ required BalRepository balRepository, - required this.id, + required this.selectedBalId, required OwnerRepository ownerRepository, }) : _balRepository = balRepository, _ownerRepository = ownerRepository { @@ -30,18 +30,26 @@ class BalViewModel extends ChangeNotifier { * ===================== */ - Bal? _bal; - int id; - Bal? get bal => _bal; + /// Selected [Bal] + Bal? _selectedBal; + + /// Selected [Bal] + Bal? get selectedBal => _selectedBal; + + /// Selected [Bal.id] from path parameters + int selectedBalId; + + /// Is one of the [Bal] [BalState.ongoing] bool isABalOngoing = false; - Future> stopBal(int id) async { + /// Stops a [Bal] + Future> stopBal(int balId) async { isLoaded = false; notifyListeners(); - final result = await _balRepository.stopBal(id); + final result = await _balRepository.stopBal(balId); switch (result) { case Ok(): - _bal = result.value; + _selectedBal = result.value; break; default: } @@ -57,14 +65,15 @@ class BalViewModel extends ChangeNotifier { return result; } - Future> startBal(int id) async { + /// Starts a [Bal] + Future> startBal(int balId) async { if (isABalOngoing) { return Result.error(Exception("Cannot have multiple BALs ongoing !")); } - final result = await _balRepository.startBal(id); + final result = await _balRepository.startBal(balId); switch (result) { case Ok(): - _bal = result.value; + _selectedBal = result.value; notifyListeners(); break; default: @@ -72,21 +81,20 @@ class BalViewModel extends ChangeNotifier { return result; } + /// Edits a [Bal]'s [name], [startTime] or [endTime] Future> editBal( int id, String name, - DateTime start, - DateTime end, + DateTime startTime, + DateTime endTime, ) async { - final result = await _balRepository.editBal(id, name, start, end); + final result = await _balRepository.editBal(id, name, startTime, endTime); switch (result) { case Ok(): - debugPrint("\n\n\n\nDID EDIT\n\n\n\n"); - _bal = result.value; + _selectedBal = result.value; notifyListeners(); break; case Error(): - debugPrint("\n\n\n\nERROR: ${result.error}"); break; } return result; @@ -99,11 +107,16 @@ class BalViewModel extends ChangeNotifier { */ // Specific to ended state + + /// Owners a book or money is owed to List? owedToOwners; - double? totalOwed; + + /// Statistics about the [_selectedBal] BalStats? stats; - Future applyAccountingOwners( + /// Froms [books], updates [owners] to include all the necessary information + /// See [the api doc](https://bal.ueauvergne.fr/docs/#/bal-api/get_bal_accounting) for more details + Future _updateOwedToOwnersWithBooks( List owners, Map books, ) async { @@ -124,15 +137,20 @@ class BalViewModel extends ChangeNotifier { } } + /// Returns either Books, Money or All ([ReturnType]) to an [Owner] Future> returnById(ReturnType type, int ownerId) async { - final result = await _balRepository.returnToId(id, ownerId, type); - final result2 = await _balRepository.getAccounting(id); + final result = await _balRepository.returnToId( + selectedBalId, + ownerId, + type, + ); + final result2 = await _balRepository.getAccounting(selectedBalId); switch (result2) { case Ok(): - applyAccountingOwners(result2.value.owners, result2.value.books); + _updateOwedToOwnersWithBooks(result2.value.owners, result2.value.books); break; case Error(): - debugPrint(result2.error.toString()); + break; } notifyListeners(); return result; @@ -144,22 +162,25 @@ class BalViewModel extends ChangeNotifier { * ================================= */ + /// Loads all the necessary information late final Command0 load; bool isLoaded = false; + /// Manages loaders Future> _load() async { isABalOngoing = _balRepository.isABalOngoing(); final result1 = await _loadBal(); switch (result1) { case Ok(): - isLoaded = (_bal == null || _bal?.state != BalState.ended) + isLoaded = + (_selectedBal == null || _selectedBal?.state != BalState.ended) ? true : false; break; default: break; } - if (_bal?.state == BalState.ended) { + if (_selectedBal?.state == BalState.ended) { final result2 = await _loadEnded(); switch (result2) { case Ok(): @@ -173,11 +194,12 @@ class BalViewModel extends ChangeNotifier { return result1; } + /// Loads all common [Bal] information Future> _loadBal() async { - final result = await _balRepository.balById(id); + final result = await _balRepository.balById(selectedBalId); switch (result) { case Ok(): - _bal = result.value; + _selectedBal = result.value; break; case Error(): break; @@ -186,15 +208,16 @@ class BalViewModel extends ChangeNotifier { return result; } + /// Loads [Bal] information when it is [BalState.ended] Future> _loadEnded() async { - final result = await _balRepository.getAccountingNoCache(id); + final result = await _balRepository.getAccountingNoCache(selectedBalId); switch (result) { case Ok(): - applyAccountingOwners(result.value.owners, result.value.books); + _updateOwedToOwnersWithBooks(result.value.owners, result.value.books); break; default: } - final result2 = await _balRepository.getBalStats(id); + final result2 = await _balRepository.getBalStats(selectedBalId); switch (result2) { case Ok(): stats = result2.value; diff --git a/lib/ui/bal_page/widget/bal_page.dart b/lib/ui/bal_page/widget/bal_page.dart index 2347dc3..e7e3467 100644 --- a/lib/ui/bal_page/widget/bal_page.dart +++ b/lib/ui/bal_page/widget/bal_page.dart @@ -27,14 +27,14 @@ class _BalPageState extends State { bottomNavigationBar: AppNavigationBar(startIndex: 0), body: AwaitLoading(), ), - true => switch (widget.viewModel.bal == null) { + true => switch (widget.viewModel.selectedBal == null) { true => Scaffold( bottomNavigationBar: AppNavigationBar(startIndex: 0), body: Center( child: Text("La BAL référencée n'est pas accessible"), ), ), - false => switch (widget.viewModel.bal!.state) { + false => switch (widget.viewModel.selectedBal!.state) { BalState.pending => BalPendingScreen(viewModel: widget.viewModel), BalState.ongoing => BalOngoingScreen(viewModel: widget.viewModel), BalState.ended => BalEndedScreen(viewModel: widget.viewModel), diff --git a/lib/ui/bal_page/widget/ended/bal_ended_screen.dart b/lib/ui/bal_page/widget/ended/bal_ended_screen.dart index 6f5043c..2b62688 100644 --- a/lib/ui/bal_page/widget/ended/bal_ended_screen.dart +++ b/lib/ui/bal_page/widget/ended/bal_ended_screen.dart @@ -30,7 +30,7 @@ class _BalEndedScreenState extends State return Scaffold( bottomNavigationBar: AppNavigationBar(startIndex: 0), appBar: AppBar( - title: Text(widget.viewModel.bal!.name), + title: Text(widget.viewModel.selectedBal!.name), bottom: TabBar( controller: tabController, tabs: [ diff --git a/lib/ui/bal_page/widget/ongoing/bal_ongoing_screen.dart b/lib/ui/bal_page/widget/ongoing/bal_ongoing_screen.dart index a9ca41e..db9fd01 100644 --- a/lib/ui/bal_page/widget/ongoing/bal_ongoing_screen.dart +++ b/lib/ui/bal_page/widget/ongoing/bal_ongoing_screen.dart @@ -11,7 +11,7 @@ class BalOngoingScreen extends StatelessWidget { Widget build(BuildContext context) { return Scaffold( bottomNavigationBar: AppNavigationBar(startIndex: 0), - appBar: AppBar(title: Text(viewModel.bal!.name)), + appBar: AppBar(title: Text(viewModel.selectedBal!.name)), body: Padding( padding: const EdgeInsets.all(10.0), child: Column( @@ -38,7 +38,9 @@ class BalOngoingScreen extends StatelessWidget { ), TextButton( onPressed: () async { - await viewModel.stopBal(viewModel.bal!.id); + await viewModel.stopBal( + viewModel.selectedBal!.id, + ); if (context.mounted) { Navigator.of(context).pop(); } diff --git a/lib/ui/bal_page/widget/pending/bal_pending_screen.dart b/lib/ui/bal_page/widget/pending/bal_pending_screen.dart index 2e57c22..e5d7164 100644 --- a/lib/ui/bal_page/widget/pending/bal_pending_screen.dart +++ b/lib/ui/bal_page/widget/pending/bal_pending_screen.dart @@ -15,7 +15,7 @@ class BalPendingScreen extends StatelessWidget { return Scaffold( bottomNavigationBar: AppNavigationBar(startIndex: 0), appBar: AppBar( - title: Text(viewModel.bal!.name), + title: Text(viewModel.selectedBal!.name), actions: [ IconButton( onPressed: () { @@ -57,7 +57,9 @@ class BalPendingScreen extends StatelessWidget { ), TextButton( onPressed: () async { - await viewModel.startBal(viewModel.bal!.id); + await viewModel.startBal( + viewModel.selectedBal!.id, + ); if (context.mounted) { Navigator.of(context).pop(); } @@ -96,8 +98,8 @@ class _EditPopup extends State { firstDate: DateTime(DateTime.now().year - 1), lastDate: DateTime(DateTime.now().year + 2), initialDateRange: DateTimeRange( - start: start ?? widget.viewModel.bal!.startTime, - end: end ?? widget.viewModel.bal!.endTime, + start: start ?? widget.viewModel.selectedBal!.startTime, + end: end ?? widget.viewModel.selectedBal!.endTime, ), ); @@ -126,7 +128,7 @@ class _EditPopup extends State { labelText: "Nom de la BAL", border: OutlineInputBorder(), ), - initialValue: widget.viewModel.bal!.name, + initialValue: widget.viewModel.selectedBal!.name, validator: (value) { if (value == null || value.isEmpty) { return "Veuillez entrer un nom"; @@ -169,7 +171,7 @@ class _EditPopup extends State { if (_formKey.currentState!.validate()) { _formKey.currentState!.save(); - final Bal bal = widget.viewModel.bal!; + final Bal bal = widget.viewModel.selectedBal!; final result = await widget.viewModel.editBal( bal.id, diff --git a/lib/ui/home_page/view_model/home_view_model.dart b/lib/ui/home_page/view_model/home_view_model.dart index 29ffeda..e6b68d2 100644 --- a/lib/ui/home_page/view_model/home_view_model.dart +++ b/lib/ui/home_page/view_model/home_view_model.dart @@ -18,18 +18,25 @@ class HomeViewModel extends ChangeNotifier { * ================= */ + /// [List] of all [Bal] List _bals = []; + + /// [List] of all [Bal] List get bals => _bals; - Bal? _currentBal; - Bal? get currentBal => _currentBal; + /// [Bal] currently [BalState.ongoing] + Bal? _ongoingBal; + /// [Bal] currently [BalState.ongoing] + Bal? get ongoingBal => _ongoingBal; + + /// Creates a [Bal] from its [name], [startTime] and [endTime] Future> createBal( String name, - DateTime start, - DateTime end, + DateTime startTime, + DateTime endTime, ) async { - final result = await _balRepository.addBal(name, start, end); + final result = await _balRepository.addBal(name, startTime, endTime); switch (result) { case Ok(): final result2 = await _balRepository.getBals(); @@ -54,9 +61,11 @@ class HomeViewModel extends ChangeNotifier { * ================================= */ + /// Command to load all necessary data late final Command0 load; bool isLoaded = false; + /// Manages loaders Future> _load() async { final result2 = await _loadBal(); switch (result2) { @@ -70,12 +79,13 @@ class HomeViewModel extends ChangeNotifier { return result2; } + /// Loads data about [Bal] Future> _loadBal() async { final result = await _balRepository.getBals(); switch (result) { case Ok(): _bals = result.value..sort((a, b) => a.compareTo(b)); - _currentBal = _bals + _ongoingBal = _bals .where((bal) => bal.state == BalState.ongoing) .firstOrNull; break; diff --git a/lib/ui/home_page/widgets/home_page.dart b/lib/ui/home_page/widgets/home_page.dart index adc6684..2f56dbd 100644 --- a/lib/ui/home_page/widgets/home_page.dart +++ b/lib/ui/home_page/widgets/home_page.dart @@ -38,7 +38,7 @@ class _HomePageState extends State { : ListView( children: [ for (Bal bal in widget.viewModel.bals.where( - (el) => el.id != widget.viewModel.currentBal?.id, + (el) => el.id != widget.viewModel.ongoingBal?.id, )) Padding( padding: const EdgeInsets.symmetric( @@ -81,20 +81,20 @@ class _HomePageState extends State { ], ), ), - switch (widget.viewModel.currentBal == null) { + switch (widget.viewModel.ongoingBal == null) { true => SizedBox(), false => Padding( padding: const EdgeInsets.symmetric(horizontal: 10.0), child: Card( child: ListTile( leading: Icon(Icons.event_available), - title: Text(widget.viewModel.currentBal!.name), + title: Text(widget.viewModel.ongoingBal!.name), subtitle: Text("BAL en cours"), trailing: IconButton( onPressed: () { _moveToBal( context, - widget.viewModel.currentBal!.id, + widget.viewModel.ongoingBal!.id, ); }, icon: Icon(Icons.arrow_forward), diff --git a/lib/ui/sell_page/view_model/sell_view_model.dart b/lib/ui/sell_page/view_model/sell_view_model.dart index 549b328..77b0393 100644 --- a/lib/ui/sell_page/view_model/sell_view_model.dart +++ b/lib/ui/sell_page/view_model/sell_view_model.dart @@ -31,10 +31,13 @@ class SellViewModel extends ChangeNotifier { final BookRepository _bookRepository; final OwnerRepository _ownerRepository; - bool _showScan = false; - bool get showScan => _showScan; - set showScan(bool newValue) { - _showScan = newValue; + /// Wether to show the scan screen + bool _showScanScreen = false; + + /// Wether to show the scan screen + bool get showScanScreen => _showScanScreen; + set showScanScreen(bool newValue) { + _showScanScreen = newValue; notifyListeners(); } @@ -44,56 +47,66 @@ class SellViewModel extends ChangeNotifier { * =============================== */ - final List _soldBooks = []; - List get soldBooks => _soldBooks; + /// Books in the sell + final List _booksInSell = []; + /// Books in the sell + List get booksInSell => _booksInSell; + + /// Books scanned on the scan screen final List _scannedBooks = []; + + /// Books scanned on the scan screen List get scannedBooks => _scannedBooks; bool isScanLoaded = false; bool isSendingSell = false; - double minimumAmount = 0; + double minimumAmountToPay = 0; - void sellBook(BookStack addedBook) { - minimumAmount += addedBook.instance.price; - _soldBooks.add(addedBook); + /// Adds a book to the [_booksInSell] + void addBookToSell(BookStack bookToAdd) { + minimumAmountToPay += bookToAdd.instance.price; + _booksInSell.add(bookToAdd); notifyListeners(); } - void sendSell(double givenAmount) async { + /// Sends the sell + void sendSell(double givenMoney) async { isSendingSell = true; notifyListeners(); - List toSend = []; - int nbOfPl = 0; - for (BookStack book in _soldBooks) { + List booksToSend = []; + int numberOfPL = 0; + for (BookStack book in _booksInSell) { if (book.instance.price != 0) { book.instance.soldPrice = book.instance.price; - givenAmount -= book.instance.price; - toSend.add(book.instance); + givenMoney -= book.instance.price; + booksToSend.add(book.instance); } else { - nbOfPl++; + numberOfPL++; } } - if (nbOfPl != 0) { - double amountPerPl = givenAmount / nbOfPl; - for (BookStack book in _soldBooks) { + if (numberOfPL != 0) { + double moneyPerPL = givenMoney / numberOfPL; + for (BookStack book in _booksInSell) { if (book.instance.price == 0) { - book.instance.soldPrice = amountPerPl; - toSend.add(book.instance); + book.instance.soldPrice = moneyPerPL; + booksToSend.add(book.instance); } } } - await _bookInstanceRepository.sellBooks(toSend); - _soldBooks.clear(); + await _bookInstanceRepository.sellBooks(booksToSend); + _booksInSell.clear(); isSendingSell = false; notifyListeners(); } - void deleteBook(int id) { - _soldBooks.removeWhere((book) => book.instance.id == id); + /// Removes a book from the sell + void removeBookFromSell(int bookId) { + _booksInSell.removeWhere((book) => book.instance.id == bookId); notifyListeners(); } + /// Search a book by [title] or [author] Future searchBook(String title, String author) async { Bal? bal = await _balRepository.ongoingBal(); isScanLoaded = false; @@ -106,15 +119,21 @@ class SellViewModel extends ChangeNotifier { ); switch (result) { case Ok(): + // For each result value, you need to complete some values for (SearchResult searchResult in result.value) { + // In case you get a book that's actually not available if (searchResult.instance.available == false) { continue; } - if (_soldBooks + + // In case the instance is already in the sell + if (_booksInSell .where((book) => book.instance.id == searchResult.instance.id) .isNotEmpty) { continue; } + + // Search for the owner Owner owner; final result2 = await _ownerRepository.getOwnerById( searchResult.instance.ownerId, @@ -126,6 +145,7 @@ class SellViewModel extends ChangeNotifier { case Error(): continue; } + _scannedBooks.add( BookStack(searchResult.book, searchResult.instance, owner), ); @@ -140,18 +160,19 @@ class SellViewModel extends ChangeNotifier { return; } + /// Gets [BookInstance]s from its ean in a [barcode] Future scanBook(BarcodeCapture barcode) async { isScanLoaded = false; int ean = int.parse(barcode.barcodes.first.rawValue!); - Bal? bal = await _balRepository.ongoingBal(); + Bal? ongoingBal = await _balRepository.ongoingBal(); _scannedBooks.clear(); - final result = await _bookInstanceRepository.getByEan(bal!.id, ean); - switch (result) { + final result1 = await _bookInstanceRepository.getByEan(ongoingBal!.id, ean); + switch (result1) { case Ok(): Book book; final result2 = await _bookRepository.getBookById( - result.value.first.bookId, + result1.value.first.bookId, ); switch (result2) { case Ok(): @@ -160,15 +181,22 @@ class SellViewModel extends ChangeNotifier { case Error(): return; } - for (BookInstance instance in result.value) { + + // For each result value, you need to complete some values + for (BookInstance instance in result1.value) { + // In case you get a book that's actually not available if (instance.available == false) { continue; } - if (_soldBooks + + // In case the instance is already in the sell + if (_booksInSell .where((book) => book.instance.id == instance.id) .isNotEmpty) { continue; } + + // Search for the owner Owner owner; final result3 = await _ownerRepository.getOwnerById(instance.ownerId); switch (result3) { @@ -178,6 +206,7 @@ class SellViewModel extends ChangeNotifier { case Error(): continue; } + _scannedBooks.add(BookStack(book, instance, owner)); } break; @@ -196,8 +225,11 @@ class SellViewModel extends ChangeNotifier { * ================= */ - Bal? _currentBal; - get currentBal => _currentBal; + /// The currently ongoing [Bal] + Bal? _ongoingBal; + + /// The currently ongoing [Bal] + get ongoingBal => _ongoingBal; /* * ================================= @@ -205,9 +237,11 @@ class SellViewModel extends ChangeNotifier { * ================================= */ + /// Command to load necessary data late final Command0 load; bool isLoaded = false; + /// Manages loaders Future> _load() async { final result1 = await _loadBal(); switch (result1) { @@ -221,11 +255,12 @@ class SellViewModel extends ChangeNotifier { return result1; } + /// Loads information about [Bal] Future> _loadBal() async { final result = await _balRepository.getBals(); switch (result) { case Ok(): - _currentBal = result.value + _ongoingBal = result.value .where((bal) => bal.state == BalState.ongoing) .firstOrNull; break; diff --git a/lib/ui/sell_page/widgets/scan_screen.dart b/lib/ui/sell_page/widgets/scan_screen.dart index ceb9920..3036fb8 100644 --- a/lib/ui/sell_page/widgets/scan_screen.dart +++ b/lib/ui/sell_page/widgets/scan_screen.dart @@ -65,7 +65,7 @@ class _ScanScreenState extends State { children: [ IconButton( onPressed: () { - widget.viewModel.showScan = false; + widget.viewModel.showScanScreen = false; }, icon: Icon(Icons.arrow_back), ), diff --git a/lib/ui/sell_page/widgets/sell_choice_popup.dart b/lib/ui/sell_page/widgets/sell_choice_popup.dart index 1931a86..f96bb93 100644 --- a/lib/ui/sell_page/widgets/sell_choice_popup.dart +++ b/lib/ui/sell_page/widgets/sell_choice_popup.dart @@ -34,9 +34,9 @@ class SellChoicePopup extends StatelessWidget { child: Card( child: InkWell( onTap: () { - viewModel.sellBook(book); + viewModel.addBookToSell(book); Navigator.of(context).pop(); - viewModel.showScan = false; + viewModel.showScanScreen = false; }, child: ListTile( leading: Text( diff --git a/lib/ui/sell_page/widgets/sell_page.dart b/lib/ui/sell_page/widgets/sell_page.dart index b04ae81..784ff23 100644 --- a/lib/ui/sell_page/widgets/sell_page.dart +++ b/lib/ui/sell_page/widgets/sell_page.dart @@ -28,7 +28,7 @@ class _SellPageState extends State { builder: (context, child) { return switch (widget.viewModel.isLoaded) { false => AwaitLoading(), - true => switch (widget.viewModel.currentBal) { + true => switch (widget.viewModel.ongoingBal) { null => Center( child: SizedBox( width: 300, @@ -80,7 +80,7 @@ class _SellPageState extends State { ? Center(child: Text("Aucun")) : SizedBox(), for (BookStack book - in widget.viewModel.soldBooks) + in widget.viewModel.booksInSell) Padding( padding: const EdgeInsets.symmetric( horizontal: 15, @@ -99,9 +99,10 @@ class _SellPageState extends State { ), trailing: IconButton( onPressed: () { - widget.viewModel.deleteBook( - book.instance.id, - ); + widget.viewModel + .removeBookFromSell( + book.instance.id, + ); }, icon: Icon(Icons.delete), ), @@ -113,7 +114,7 @@ class _SellPageState extends State { ), SizedBox(height: 40), Text( - "Montant minimum à payer : ${widget.viewModel.minimumAmount.toString()}€", + "Montant minimum à payer : ${widget.viewModel.minimumAmountToPay.toString()}€", ), Padding( padding: const EdgeInsets.symmetric(horizontal: 60.0), @@ -167,7 +168,7 @@ class _SellPageState extends State { } else if (double.parse( price.text.replaceFirst(",", "."), ) < - widget.viewModel.minimumAmount) { + widget.viewModel.minimumAmountToPay) { ScaffoldMessenger.of(context).showSnackBar( SnackBar( content: Text( @@ -202,7 +203,7 @@ class _SellPageState extends State { SizedBox(width: 70), IconButton( onPressed: () { - widget.viewModel.showScan = true; + widget.viewModel.showScanScreen = true; }, icon: Icon(Icons.add), style: ButtonStyle( @@ -216,7 +217,7 @@ class _SellPageState extends State { ], ), ), - (widget.viewModel.showScan) + (widget.viewModel.showScanScreen) ? ScanScreen(viewModel: widget.viewModel) : SizedBox(), ], From 6bcc3a7e882bb81179a276d423b82fafa0559751 Mon Sep 17 00:00:00 2001 From: Alzalia Date: Sat, 23 Aug 2025 15:52:51 +0200 Subject: [PATCH 28/29] fix: some error managment and a whole feature missing --- .../add_page/view_model/add_view_model.dart | 3 +- lib/ui/add_page/widgets/add_page.dart | 148 +++++++++++------- .../add_page/widgets/confirmation_popup.dart | 100 +++++++----- lib/ui/add_page/widgets/form_popup.dart | 75 +++++---- lib/ui/add_page/widgets/owner_popup.dart | 6 +- 5 files changed, 192 insertions(+), 140 deletions(-) diff --git a/lib/ui/add_page/view_model/add_view_model.dart b/lib/ui/add_page/view_model/add_view_model.dart index 9334ea7..3b0be3a 100644 --- a/lib/ui/add_page/view_model/add_view_model.dart +++ b/lib/ui/add_page/view_model/add_view_model.dart @@ -125,8 +125,7 @@ class AddViewModel extends ChangeNotifier { */ /// Retrieves the book associated with an ean through a [barcode] - Future> scanBook(BarcodeCapture barcode) async { - var ean = barcode.barcodes.first.rawValue!; + Future> scanBook(String ean) async { var result = await _bookRepository.getBookByEAN(ean); return result; } diff --git a/lib/ui/add_page/widgets/add_page.dart b/lib/ui/add_page/widgets/add_page.dart index 4d22005..e0e68ec 100644 --- a/lib/ui/add_page/widgets/add_page.dart +++ b/lib/ui/add_page/widgets/add_page.dart @@ -22,23 +22,20 @@ class AddPage extends StatefulWidget { } class _AddPageState extends State { - num? price; - final MobileScannerController controller = MobileScannerController( + final MobileScannerController scannerController = MobileScannerController( formats: [BarcodeFormat.ean13], detectionTimeoutMs: 1000, ); @override void dispose() { - controller.dispose(); + scannerController.dispose(); super.dispose(); } @override Widget build(BuildContext context) { final theme = Theme.of(context); - // return Consumer( - // builder: (context, screen, child) { return Scaffold( bottomNavigationBar: AppNavigationBar(startIndex: 1), body: ListenableBuilder( @@ -78,60 +75,37 @@ class _AddPageState extends State { children: [ ColoredBox(color: Colors.black), MobileScanner( - controller: controller, + controller: scannerController, onDetect: (barcodes) async { if (barcodes.barcodes.isEmpty) { return; } + if (widget.viewModel.currentOwner == null) { - ScaffoldMessenger.of(context).showSnackBar( - SnackBar( - content: Text( - "Attention : vous devez choisir un·e propriétaire", - ), - behavior: SnackBarBehavior.floating, - ), + _showMissingOwnerSnackBar( + context, + scannerController, + widget.viewModel, ); return; } - void setPrice(num newPrice) async { - setState(() { - price = newPrice; - }); - } - - Result result = await widget.viewModel.scanBook( - barcodes, + _scanEan( + context, + widget.viewModel, + barcodes.barcodes.first.rawValue!, + scannerController, ); - - switch (result) { - case Ok(): - await _confirmationDialogBuilder( - context, - setPrice, - controller, - widget.viewModel, - result.value, - ); - break; - case Error(): - ScaffoldMessenger.of(context).showSnackBar( - SnackBar( - content: Text("Erreur : ${result.error}"), - behavior: SnackBarBehavior.floating, - ), - ); - break; - } }, ), + Center( child: SvgPicture.asset( 'assets/scan-overlay.svg', height: (MediaQuery.sizeOf(context).height / 5) * 2, ), ), + SafeArea( child: SingleChildScrollView( child: Column( @@ -155,7 +129,7 @@ class _AddPageState extends State { ), onPressed: () => _ownerDialogBuilder( context, - controller, + scannerController, widget.viewModel, ), ), @@ -184,6 +158,7 @@ class _AddPageState extends State { ), ), ), + SafeArea( child: Column( mainAxisAlignment: MainAxisAlignment.end, @@ -197,19 +172,17 @@ class _AddPageState extends State { ), onPressed: () { if (widget.viewModel.currentOwner == null) { - ScaffoldMessenger.of(context).showSnackBar( - SnackBar( - content: Text( - "Attention : vous devez choisir un·e propriétaire", - ), - behavior: SnackBarBehavior.floating, - ), + _showMissingOwnerSnackBar( + context, + scannerController, + widget.viewModel, ); + return; } _formDialogBuilder( context, - controller, + scannerController, widget.viewModel, ); }, @@ -231,18 +204,71 @@ class _AddPageState extends State { } } +void _scanEan( + BuildContext context, + AddViewModel viewModel, + String ean, + MobileScannerController scannerController, { + Function(BuildContext)? leaveLastPopup, +}) async { + Result result = await viewModel.scanBook(ean); + + if (context.mounted) { + if (leaveLastPopup != null) { + leaveLastPopup(context); + } + switch (result) { + case Ok(): + await _confirmationDialogBuilder( + context, + scannerController, + viewModel, + result.value, + ); + break; + case Error(): + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text("Erreur : ${result.error}"), + behavior: SnackBarBehavior.floating, + ), + ); + break; + } + } +} + +void _showMissingOwnerSnackBar( + BuildContext context, + MobileScannerController scannerController, + AddViewModel viewModel, +) { + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text("Attention : vous devez choisir un·e propriétaire"), + duration: Duration(seconds: 4), + action: SnackBarAction( + label: "Choisir", + onPressed: () => + _ownerDialogBuilder(context, scannerController, viewModel), + ), + behavior: SnackBarBehavior.floating, + ), + ); +} + Future _confirmationDialogBuilder( BuildContext context, - Function(num) setPrice, - MobileScannerController controller, + MobileScannerController scannerController, AddViewModel viewModel, Book book, ) { - controller.stop(); + scannerController.stop(); + // Utility function to pass to downwards widgets void exitPopup(BuildContext localContext) { Navigator.of(localContext).pop(); - controller.start(); + scannerController.start(); } return showDialog( @@ -250,7 +276,6 @@ Future _confirmationDialogBuilder( barrierDismissible: false, builder: (context) => ConfirmationPopup( exitPopup: exitPopup, - setPrice: setPrice, viewModel: viewModel, book: book, ), @@ -264,6 +289,7 @@ Future _formDialogBuilder( ) { controller.stop(); + // Utility function to pass to downwards widgets void exitPopup(BuildContext localContext) { Navigator.of(localContext).pop(); controller.start(); @@ -272,7 +298,12 @@ Future _formDialogBuilder( return showDialog( context: context, barrierDismissible: false, - builder: (context) => FormPopup(viewModel: viewModel, exitPopup: exitPopup), + builder: (context) => FormPopup( + viewModel: viewModel, + exitPopup: exitPopup, + scannerController: controller, + scanEan: _scanEan, + ), ); } @@ -283,7 +314,8 @@ Future _ownerDialogBuilder( ) { controller.stop(); - void onPressAccept(BuildContext localContext) { + // Utility function to pass to downwards widgets + void exitPopup(BuildContext localContext) { Navigator.of(localContext).pop(); controller.start(); } @@ -292,6 +324,6 @@ Future _ownerDialogBuilder( context: context, barrierDismissible: false, builder: (context) => - OwnerPopup(viewModel: viewModel, onPressAccept: onPressAccept), + OwnerPopup(viewModel: viewModel, exitPopup: exitPopup), ); } diff --git a/lib/ui/add_page/widgets/confirmation_popup.dart b/lib/ui/add_page/widgets/confirmation_popup.dart index e57f02c..998e3ea 100644 --- a/lib/ui/add_page/widgets/confirmation_popup.dart +++ b/lib/ui/add_page/widgets/confirmation_popup.dart @@ -7,13 +7,11 @@ class ConfirmationPopup extends StatefulWidget { const ConfirmationPopup({ super.key, required this.exitPopup, - required this.setPrice, required this.viewModel, required this.book, }); final Function(BuildContext) exitPopup; - final Function(num) setPrice; final AddViewModel viewModel; final Book book; @@ -24,9 +22,11 @@ class ConfirmationPopup extends StatefulWidget { class _ConfirmationPopupState extends State { final GlobalKey _formKey = GlobalKey(); double price = 0; + @override Widget build(BuildContext context) { final theme = Theme.of(context); + return AlertDialog( title: Text("Prix"), content: Form( @@ -72,6 +72,7 @@ class _ConfirmationPopupState extends State { ], ), ), + SizedBox(height: 10), (widget.viewModel.askPrice) ? TextFormField( @@ -87,15 +88,21 @@ class _ConfirmationPopupState extends State { validator: (value) { if (value == null || value.isEmpty) { return "Indiquez un prix"; - } else if (num.tryParse(value) == null) { + } else if (double.tryParse( + value.replaceAll(",", "."), + ) == + null) { return "Le prix doit être un nombre"; - } else if (num.parse(value) < 0) { + } else if (double.parse(value.replaceAll(",", ".")) < + 0) { return "Le prix doit être positif ou nul"; } return null; }, onSaved: (newValue) { - price = double.parse(newValue!); + price = double.parse( + newValue?.replaceAll(",", ".") ?? "0", + ); }, ) : SizedBox(), @@ -105,6 +112,7 @@ class _ConfirmationPopupState extends State { ), actions: [ TextButton( + child: Text("Annuler"), onPressed: () { ScaffoldMessenger.of(context).showSnackBar( SnackBar( @@ -114,13 +122,15 @@ class _ConfirmationPopupState extends State { ); widget.exitPopup(context); }, - child: Text("Annuler"), ), TextButton( + child: Text("Valider"), onPressed: () async { if (widget.viewModel.askPrice && _formKey.currentState!.validate()) { _formKey.currentState!.save(); + } else { + return; } var result = await widget.viewModel.sendNewBookInstance( @@ -136,49 +146,61 @@ class _ConfirmationPopupState extends State { Navigator.of(context).pop(); showDialog( context: context, - builder: (context) => AlertDialog( - title: Text( - "ID : ${widget.viewModel.currentOwner!.firstName[0].toUpperCase()}${widget.viewModel.currentOwner!.lastName[0].toUpperCase()}${(price == 0) ? "PL" : price.toString()}", - ), - content: Text( - (widget.viewModel.currentOwner!.id == - widget.viewModel.ownerOfUser!.id) - ? "Ce livre appartient à la section. Vous pouvez mettre le code, ou poser une gomette, ..." - : "Identifiant propriétaire de ce livre. Pensez à l'écrire pour retrouver lae propriétaire du livre lors de la vente ou du retour !", - ), - actions: [ - TextButton( - onPressed: () { - widget.exitPopup(context); - ScaffoldMessenger.of(context).showSnackBar( - SnackBar( - content: Text( - "Enregistré: ${widget.book.title}", - ), - behavior: SnackBarBehavior.floating, - ), - ); - }, - child: Text("Ok"), - ), - ], - ), + barrierDismissible: false, + builder: (context) => + RegisteredBookPopup(widget: widget, price: price), ); } break; case Error(): if (context.mounted) { ScaffoldMessenger.of(context).showSnackBar( - SnackBar( - content: Text( - "Une erreur est survenue : ${result.error}", - ), - ), + SnackBar(content: Text("Erreur : ${result.error}")), ); } } }, - child: Text("Valider"), + ), + ], + ); + } +} + +class RegisteredBookPopup extends StatelessWidget { + const RegisteredBookPopup({ + super.key, + required this.widget, + required this.price, + }); + + final ConfirmationPopup widget; + final double price; + + @override + Widget build(BuildContext context) { + return AlertDialog( + // This thing is the BookInstance's short ID + title: Text( + "ID : ${widget.viewModel.currentOwner!.firstName[0].toUpperCase()}${widget.viewModel.currentOwner!.lastName[0].toUpperCase()}${(price == 0) ? "PL" : price.toString()}", + ), + content: Text( + (widget.viewModel.currentOwner!.id == widget.viewModel.ownerOfUser!.id) + ? "Pensez à la gomette ! Ce livre appartient au syndicat." + : "Identifiant propriétaire de ce livre. Pensez à l'écrire pour retrouver lae propriétaire du livre lors de la vente ou du retour !", + ), + actions: [ + TextButton( + child: Text("Ok"), + onPressed: () { + widget.exitPopup(context); + ScaffoldMessenger.of(context).showSnackBar( + SnackBar( + content: Text("Livre enregistré: ${widget.book.title}"), + behavior: SnackBarBehavior.floating, + duration: Duration(seconds: 2), + ), + ); + }, ), ], ); diff --git a/lib/ui/add_page/widgets/form_popup.dart b/lib/ui/add_page/widgets/form_popup.dart index 2a3f1de..47a3104 100644 --- a/lib/ui/add_page/widgets/form_popup.dart +++ b/lib/ui/add_page/widgets/form_popup.dart @@ -1,4 +1,5 @@ import 'package:flutter/material.dart'; +import 'package:mobile_scanner/mobile_scanner.dart'; import 'package:seshat/ui/add_page/view_model/add_view_model.dart'; class FormPopup extends StatelessWidget { @@ -6,10 +7,14 @@ class FormPopup extends StatelessWidget { super.key, required this.viewModel, required this.exitPopup, + required this.scannerController, + required this.scanEan, }); final AddViewModel viewModel; final Function(BuildContext) exitPopup; + final MobileScannerController scannerController; + final Function scanEan; @override Widget build(BuildContext context) { @@ -31,6 +36,8 @@ class FormPopup extends StatelessWidget { return _ManualEANPopup( exitPopup: exitPopup, viewModel: viewModel, + scannerController: scannerController, + scanEan: scanEan, ); }, ); @@ -51,6 +58,7 @@ class FormPopup extends StatelessWidget { ), ), ), + Card( clipBehavior: Clip.hardEdge, child: InkWell( @@ -93,11 +101,24 @@ class FormPopup extends StatelessWidget { } } +/* + * ====================== + * ====< MANUAL EAN >==== + * ====================== +*/ + class _ManualEANPopup extends StatefulWidget { - const _ManualEANPopup({required this.exitPopup, required this.viewModel}); + const _ManualEANPopup({ + required this.exitPopup, + required this.viewModel, + required this.scannerController, + required this.scanEan, + }); final Function(BuildContext) exitPopup; final AddViewModel viewModel; + final MobileScannerController scannerController; + final Function scanEan; @override State<_ManualEANPopup> createState() => _ManualEANPopupState(); @@ -106,11 +127,11 @@ class _ManualEANPopup extends StatefulWidget { class _ManualEANPopupState extends State<_ManualEANPopup> { final GlobalKey _formKey = GlobalKey(); String? ean; - num? price; + @override Widget build(BuildContext context) { return AlertDialog( - title: Text("Recherche par EAN"), + title: Text("Entrée manuelle par EAN"), content: Form( key: _formKey, child: Column( @@ -128,61 +149,39 @@ class _ManualEANPopupState extends State<_ManualEANPopup> { validator: (value) { if (value == null || value.length != 13 || - int.tryParse(value) == null) { + int.tryParse(value) == null || + int.parse(value) < 0) { return "L'entrée n'est pas un code EAN-13 valide"; } return null; }, ), - SizedBox(height: 10), - ListenableBuilder( - listenable: widget.viewModel, - builder: (context, child) { - return (widget.viewModel.askPrice) - ? TextFormField( - decoration: InputDecoration( - labelText: "Prix", - border: OutlineInputBorder(), - suffixText: "€", - ), - keyboardType: TextInputType.numberWithOptions( - decimal: true, - ), - validator: (value) { - if (value == null || value.isEmpty) { - return "Indiquez un prix"; - } else if (num.tryParse(value) == null) { - return "Le prix doit être un nombre"; - } else if (num.parse(value) < 0) { - return "Le prix doit être positif ou nul"; - } - return null; - }, - onSaved: (newValue) { - price = num.parse(newValue!); - }, - ) - : SizedBox(); - }, - ), ], ), ), actions: [ TextButton( + child: Text("Annuler"), onPressed: () { widget.exitPopup(context); }, - child: Text("Annuler"), ), TextButton( + child: Text("Valider"), onPressed: () { if (_formKey.currentState!.validate()) { _formKey.currentState!.save(); - widget.exitPopup(context); + widget.scanEan( + context, + widget.viewModel, + ean!, + widget.scannerController, + leaveLastPopup: (context) { + Navigator.of(context).pop(); + }, + ); } }, - child: Text("Valider"), ), ], ); diff --git a/lib/ui/add_page/widgets/owner_popup.dart b/lib/ui/add_page/widgets/owner_popup.dart index 35fe4b3..8627c13 100644 --- a/lib/ui/add_page/widgets/owner_popup.dart +++ b/lib/ui/add_page/widgets/owner_popup.dart @@ -6,11 +6,11 @@ class OwnerPopup extends StatefulWidget { const OwnerPopup({ super.key, required this.viewModel, - required this.onPressAccept, + required this.exitPopup, }); final AddViewModel viewModel; - final Function(BuildContext) onPressAccept; + final Function(BuildContext) exitPopup; @override State createState() => _OwnerPopupState(); @@ -166,7 +166,7 @@ class _OwnerPopupState extends State { }); } } - widget.onPressAccept(context); + widget.exitPopup(context); }, child: Text( (!showNewOwner && searchController.text == "") From 7b62d9781ab43e7433ac2f1f78234be767212e97 Mon Sep 17 00:00:00 2001 From: alzalia Date: Mon, 25 Aug 2025 15:05:15 +0200 Subject: [PATCH 29/29] Update README.md --- README.md | 3 +++ 1 file changed, 3 insertions(+) diff --git a/README.md b/README.md index 304117b..a5a46b7 100644 --- a/README.md +++ b/README.md @@ -1,3 +1,6 @@ +> [!WARNING] +> This repo has been moved to [illes](https://git.illes.fr/UEAuvergne/Seshat). + # seshat Client android/iOS/web, écrit en dart x flutter, pour Alexandria.