This commit is contained in:
Mohammed Al-Samarraie
2026-01-16 14:42:43 +03:00
parent 56e2c0ffaa
commit 2fd5aff0c2
4 changed files with 363 additions and 56 deletions

26
.vscode/launch.json vendored Normal file
View File

@@ -0,0 +1,26 @@
{
// Use IntelliSense to learn about possible attributes.
// Hover to view descriptions of existing attributes.
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
"version": "0.2.0",
"configurations": [
{
"name": "finger_print_app",
"request": "launch",
"type": "dart"
},
{
"name": "finger_print_app (profile mode)",
"request": "launch",
"type": "dart",
"flutterMode": "profile"
},
{
"name": "finger_print_app (release mode)",
"request": "launch",
"type": "dart",
"flutterMode": "release"
}
]
}

View File

@@ -1,5 +1,6 @@
import 'dart:io'; import 'dart:io';
import 'package:dio/dio.dart'; import 'package:dio/dio.dart';
import '../../core/error/exceptions.dart';
import '../../core/network/api_client.dart'; import '../../core/network/api_client.dart';
import '../dto/attendance_response_dto.dart'; import '../dto/attendance_response_dto.dart';
@@ -24,6 +25,7 @@ class AttendanceRemoteDataSourceImpl implements AttendanceRemoteDataSource {
required String employeeId, required String employeeId,
required File faceImage, required File faceImage,
}) async { }) async {
try {
final formData = FormData.fromMap({ final formData = FormData.fromMap({
'EmployeeId': employeeId, 'EmployeeId': employeeId,
'FaceImage': await MultipartFile.fromFile(faceImage.path), 'FaceImage': await MultipartFile.fromFile(faceImage.path),
@@ -34,7 +36,51 @@ class AttendanceRemoteDataSourceImpl implements AttendanceRemoteDataSource {
data: formData, data: formData,
options: Options(contentType: 'multipart/form-data'), options: Options(contentType: 'multipart/form-data'),
); );
return AttendanceResponseDto.fromJson(response.data);
if (response.statusCode == 200 || response.statusCode == 201) {
final responseData = response.data;
if (responseData is Map<String, dynamic>) {
return AttendanceResponseDto.fromJson(responseData);
} else {
throw ServerException(
message: 'استجابة غير صحيحة من الخادم',
statusCode: response.statusCode,
);
}
} else {
throw ServerException(
message: 'فشل تسجيل الدخول',
statusCode: response.statusCode,
);
}
} on DioException catch (e) {
if (e.type == DioExceptionType.connectionTimeout ||
e.type == DioExceptionType.receiveTimeout) {
throw NetworkException(message: 'انتهت مهلة الاتصال');
} else if (e.type == DioExceptionType.connectionError) {
throw NetworkException(message: 'لا يوجد اتصال بالانترنيت');
} else if (e.response?.statusCode == 500) {
throw ServerException(message: 'خطأ في الخادم يرجى المحاولة لاحقا');
} else if (e.response != null) {
final message =
e.response?.data?['message'] ??
e.response?.data?['error'] ??
'فشل تسجيل الدخول';
throw ServerException(
message: message.toString(),
statusCode: e.response?.statusCode,
);
} else {
throw NetworkException(message: 'خطأ في الانترنيت يرجى المحاولة لاحقا');
}
} catch (e) {
if (e is ServerException || e is NetworkException) {
rethrow;
}
throw ServerException(message: 'خطأ غير متوقع');
}
} }
@override @override
@@ -42,6 +88,7 @@ class AttendanceRemoteDataSourceImpl implements AttendanceRemoteDataSource {
required String employeeId, required String employeeId,
required File faceImage, required File faceImage,
}) async { }) async {
try {
final formData = FormData.fromMap({ final formData = FormData.fromMap({
'EmployeeId': employeeId, 'EmployeeId': employeeId,
'FaceImage': await MultipartFile.fromFile(faceImage.path), 'FaceImage': await MultipartFile.fromFile(faceImage.path),
@@ -53,6 +100,49 @@ class AttendanceRemoteDataSourceImpl implements AttendanceRemoteDataSource {
options: Options(contentType: 'multipart/form-data'), options: Options(contentType: 'multipart/form-data'),
); );
return AttendanceResponseDto.fromJson(response.data); if (response.statusCode == 200 || response.statusCode == 201) {
final responseData = response.data;
if (responseData is Map<String, dynamic>) {
return AttendanceResponseDto.fromJson(responseData);
} else {
throw ServerException(
message: 'استجابة غير صحيحة من الخادم',
statusCode: response.statusCode,
);
}
} else {
throw ServerException(
message: 'فشل تسجيل الخروج',
statusCode: response.statusCode,
);
}
} on DioException catch (e) {
if (e.type == DioExceptionType.connectionTimeout ||
e.type == DioExceptionType.receiveTimeout) {
throw NetworkException(message: 'انتهت مهلة الاتصال');
} else if (e.type == DioExceptionType.connectionError) {
throw NetworkException(message: 'لا يوجد اتصال بالانترنيت');
} else if (e.response?.statusCode == 500) {
throw ServerException(message: 'خطأ في الخادم يرجى المحاولة لاحقا');
} else if (e.response != null) {
final message =
e.response?.data?['message'] ??
e.response?.data?['error'] ??
'فشل تسجيل الخروج';
throw ServerException(
message: message.toString(),
statusCode: e.response?.statusCode,
);
} else {
throw NetworkException(message: 'خطأ في الانترنيت يرجى المحاولة لاحقا');
}
} catch (e) {
if (e is ServerException || e is NetworkException) {
rethrow;
}
throw ServerException(message: 'خطأ غير متوقع');
}
} }
} }

View File

@@ -3,6 +3,7 @@ import 'package:flutter/material.dart';
import 'package:flutter_svg/flutter_svg.dart'; import 'package:flutter_svg/flutter_svg.dart';
import 'dart:async'; import 'dart:async';
import 'dart:io'; import 'dart:io';
import '../../core/error/exceptions.dart';
class OvalCameraCapturePage extends StatefulWidget { class OvalCameraCapturePage extends StatefulWidget {
final bool isLogin; final bool isLogin;
@@ -23,6 +24,7 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
bool _isCameraInitialized = false; bool _isCameraInitialized = false;
String? _errorMessage; String? _errorMessage;
bool _isSuccess = false; bool _isSuccess = false;
bool _isLoading = false;
Timer? _timer; Timer? _timer;
@override @override
@@ -33,6 +35,11 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
Future<void> _initializeCamera() async { Future<void> _initializeCamera() async {
try { try {
setState(() {
_errorMessage = null;
_isCameraInitialized = false;
});
// Dispose existing controller if any // Dispose existing controller if any
await _cameraController?.dispose(); await _cameraController?.dispose();
_cameraController = null; _cameraController = null;
@@ -42,8 +49,9 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
// Check if cameras list is available // Check if cameras list is available
if (cameras.isEmpty) { if (cameras.isEmpty) {
if (!mounted) return;
setState(() { setState(() {
_errorMessage = "لا توجد كاميرات متاحة"; _errorMessage = "لا توجد كاميرات متاحة على هذا الجهاز";
_isCameraInitialized = false; _isCameraInitialized = false;
}); });
return; return;
@@ -60,8 +68,9 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
if (cameras.isNotEmpty) { if (cameras.isNotEmpty) {
selectedCamera = cameras.first; selectedCamera = cameras.first;
} else { } else {
if (!mounted) return;
setState(() { setState(() {
_errorMessage = "لا توجد كاميرات متاحة"; _errorMessage = "لا توجد كاميرات متاحة على هذا الجهاز";
_isCameraInitialized = false; _isCameraInitialized = false;
}); });
return; return;
@@ -85,11 +94,33 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
}); });
_startScan(); _startScan();
} on CameraException catch (e) {
if (!mounted) return;
String errorMessage;
switch (e.code) {
case 'CameraAccessDenied':
errorMessage = "تم رفض الوصول إلى الكاميرا. يرجى السماح بالوصول في الإعدادات";
break;
case 'CameraAccessDeniedWithoutPrompt':
errorMessage = "تم رفض الوصول إلى الكاميرا. يرجى السماح بالوصول في الإعدادات";
break;
case 'CameraAccessRestricted':
errorMessage = "الوصول إلى الكاميرا مقيد";
break;
case 'AudioAccessDenied':
errorMessage = "تم رفض الوصول إلى الميكروفون";
break;
default:
errorMessage = "خطأ في تهيئة الكاميرا: ${e.description ?? 'خطأ غير معروف'}";
}
setState(() {
_errorMessage = errorMessage;
_isCameraInitialized = false;
});
} catch (e) { } catch (e) {
if (!mounted) return; if (!mounted) return;
setState(() { setState(() {
_errorMessage = "خطأ في تهيئة الكاميرا: $e"; _errorMessage = "حدث خطأ غير متوقع أثناء تهيئة الكاميرا";
_isCameraInitialized = false; _isCameraInitialized = false;
}); });
print("Error initializing camera: $e"); print("Error initializing camera: $e");
@@ -98,46 +129,104 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
Future<void> _startScan() async { Future<void> _startScan() async {
try { try {
setState(() {
_isLoading = true;
_errorMessage = null;
});
// Simulate scanning delay // Simulate scanning delay
await Future.delayed(const Duration(seconds: 2)); await Future.delayed(const Duration(seconds: 2));
if (!mounted || if (!mounted ||
_cameraController == null || _cameraController == null ||
!_cameraController!.value.isInitialized) { !_cameraController!.value.isInitialized) {
if (mounted) {
setState(() {
_isLoading = false;
_errorMessage = "الكاميرا غير مهيأة. يرجى المحاولة مرة أخرى";
});
}
return; return;
} }
final xFile = await _cameraController!.takePicture(); final xFile = await _cameraController!.takePicture();
final file = File(xFile.path); final file = File(xFile.path);
// Check if file exists and is readable
if (!await file.exists()) {
if (mounted) {
setState(() {
_isLoading = false;
_errorMessage = "فشل حفظ الصورة. يرجى المحاولة مرة أخرى";
});
}
return;
}
// Call the onCapture callback which may throw exceptions
await widget.onCapture(file); await widget.onCapture(file);
if (mounted) { if (mounted) {
setState(() { setState(() {
_isSuccess = true; _isSuccess = true;
_isLoading = false;
}); });
// Auto-close after 2 seconds // Auto-close after 2 seconds
Future.delayed(const Duration(seconds: 2), () { Future.delayed(const Duration(seconds: 2), () {
if (mounted) { if (mounted) {
Navigator.of(context).pop(); Navigator.of(context).pop(true);
} }
}); });
} }
} on ServerException catch (e) {
if (mounted) {
setState(() {
_isLoading = false;
_errorMessage = e.message;
});
}
} on NetworkException catch (e) {
if (mounted) {
setState(() {
_isLoading = false;
_errorMessage = e.message;
});
}
} on CameraException catch (e) {
if (mounted) {
String errorMessage;
switch (e.code) {
case 'captureAlreadyActive':
errorMessage = "جاري التقاط صورة بالفعل. يرجى الانتظار";
break;
case 'pictureTakingInProgress':
errorMessage = "جاري التقاط صورة. يرجى الانتظار";
break;
default:
errorMessage = "فشل التقاط الصورة: ${e.description ?? 'خطأ غير معروف'}";
}
setState(() {
_isLoading = false;
_errorMessage = errorMessage;
});
}
} on FileSystemException catch (e) {
if (mounted) {
setState(() {
_isLoading = false;
_errorMessage = "فشل حفظ الصورة. يرجى التحقق من مساحة التخزين";
});
}
print("File system error: $e");
} catch (e) { } catch (e) {
if (mounted) { if (mounted) {
setState(() { setState(() {
_errorMessage = e.toString(); // Show error to user _isLoading = false;
}); _errorMessage = "حدث خطأ غير متوقع. يرجى المحاولة مرة أخرى";
// Auto-close or let user retry? For now let's just show error.
// If we want to auto-close on error:
Future.delayed(const Duration(seconds: 3), () {
if (mounted) {
Navigator.of(context).pop();
}
}); });
} }
print("Unexpected error in _startScan: $e");
} }
} }
@@ -154,7 +243,8 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
backgroundColor: Color(0xff000000), backgroundColor: Color(0xff000000),
body: body:
_errorMessage != null // Show error screen only if camera failed to initialize
_errorMessage != null && !_isCameraInitialized
? Center( ? Center(
child: Padding( child: Padding(
padding: const EdgeInsets.all(20.0), padding: const EdgeInsets.all(20.0),
@@ -177,6 +267,9 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
textAlign: TextAlign.center, textAlign: TextAlign.center,
), ),
SizedBox(height: 24), SizedBox(height: 24),
Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
ElevatedButton( ElevatedButton(
onPressed: _initializeCamera, onPressed: _initializeCamera,
style: ElevatedButton.styleFrom( style: ElevatedButton.styleFrom(
@@ -189,6 +282,25 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
), ),
child: Text("إعادة المحاولة"), child: Text("إعادة المحاولة"),
), ),
SizedBox(width: 16),
OutlinedButton(
onPressed: () {
if (mounted) {
Navigator.of(context).pop(false);
}
},
style: OutlinedButton.styleFrom(
foregroundColor: Colors.white,
side: BorderSide(color: Colors.white70),
padding: EdgeInsets.symmetric(
horizontal: 32,
vertical: 12,
),
),
child: Text("إلغاء"),
),
],
),
], ],
), ),
), ),
@@ -239,9 +351,13 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
? (widget.isLogin ? (widget.isLogin
? "تم تسجيل دخولك بنجاح" ? "تم تسجيل دخولك بنجاح"
: "تم تسجيل خروجك بنجاح") : "تم تسجيل خروجك بنجاح")
: (widget.isLogin : _isLoading
? (widget.isLogin
? "يتم تسجيل الدخول ..." ? "يتم تسجيل الدخول ..."
: "يتم تسجيل الخروج ..."), : "يتم تسجيل الخروج ...")
: (widget.isLogin
? "جاهز للتقاط الصورة"
: "جاهز للتقاط الصورة"),
style: const TextStyle( style: const TextStyle(
color: Colors.white, color: Colors.white,
fontSize: 18, fontSize: 18,
@@ -266,6 +382,81 @@ class _OvalCameraCapturePageState extends State<OvalCameraCapturePage> {
), ),
), ),
// Error overlay (shown when error occurs during scanning)
if (_errorMessage != null && _isCameraInitialized)
Positioned.fill(
child: Container(
color: Colors.black.withOpacity(0.7),
child: Center(
child: Padding(
padding: const EdgeInsets.all(20.0),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
mainAxisSize: MainAxisSize.min,
children: [
Icon(
Icons.error_outline,
size: 64,
color: Colors.red,
),
SizedBox(height: 16),
Text(
_errorMessage!,
style: TextStyle(
color: Colors.white,
fontSize: 18,
fontWeight: FontWeight.bold,
),
textAlign: TextAlign.center,
),
SizedBox(height: 24),
Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [
ElevatedButton(
onPressed: () {
setState(() {
_errorMessage = null;
_isLoading = false;
});
_startScan();
},
style: ElevatedButton.styleFrom(
backgroundColor: Color(0xffE8001A),
foregroundColor: Colors.white,
padding: EdgeInsets.symmetric(
horizontal: 32,
vertical: 12,
),
),
child: Text("إعادة المحاولة"),
),
SizedBox(width: 16),
OutlinedButton(
onPressed: () {
if (mounted) {
Navigator.of(context).pop(false);
}
},
style: OutlinedButton.styleFrom(
foregroundColor: Colors.white,
side: BorderSide(color: Colors.white70),
padding: EdgeInsets.symmetric(
horizontal: 32,
vertical: 12,
),
),
child: Text("إلغاء"),
),
],
),
],
),
),
),
),
),
// // Capture button // // Capture button
// Positioned( // Positioned(
// bottom: 60, // bottom: 60,

View File

@@ -29,7 +29,7 @@ packages:
dependency: transitive dependency: transitive
description: description:
name: async name: async
sha256: d2872f9c19731c2e5f10444b14686eb7cc85c76274bd6c16e1816bff9a3bab63 sha256: "758e6d74e971c3e5aceb4110bfd6698efc7f501675bcfe0c775459a8140750eb"
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "2.13.0" version: "2.13.0"
@@ -189,10 +189,10 @@ packages:
dependency: transitive dependency: transitive
description: description:
name: fake_async name: fake_async
sha256: "6a95e56b2449df2273fd8c45a662d6947ce1ebb7aafe80e550a3f68297f3cacc" sha256: "5368f224a74523e8d2e7399ea1638b37aecfca824a3cc4dfdf77bf1fa905ac44"
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "1.3.2" version: "1.3.3"
ffi: ffi:
dependency: transitive dependency: transitive
description: description:
@@ -289,7 +289,7 @@ packages:
source: hosted source: hosted
version: "0.15.6" version: "0.15.6"
http: http:
dependency: "direct main" dependency: transitive
description: description:
name: http name: http
sha256: "87721a4a50b19c7f1d49001e51409bddc46303966ce89a65af4f4e6004896412" sha256: "87721a4a50b19c7f1d49001e51409bddc46303966ce89a65af4f4e6004896412"
@@ -324,10 +324,10 @@ packages:
dependency: transitive dependency: transitive
description: description:
name: leak_tracker name: leak_tracker
sha256: c35baad643ba394b40aac41080300150a4f08fd0fd6a10378f8f7c6bc161acec sha256: "6bb818ecbdffe216e81182c2f0714a2e62b593f4a4f13098713ff1685dfb6ab0"
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "10.0.8" version: "10.0.9"
leak_tracker_flutter_testing: leak_tracker_flutter_testing:
dependency: transitive dependency: transitive
description: description:
@@ -641,10 +641,10 @@ packages:
dependency: transitive dependency: transitive
description: description:
name: vm_service name: vm_service
sha256: "0968250880a6c5fe7edc067ed0a13d4bae1577fe2771dcf3010d52c4a9d3ca14" sha256: ddfa8d30d89985b96407efce8acbdd124701f96741f2d981ca860662f1c0dc02
url: "https://pub.dev" url: "https://pub.dev"
source: hosted source: hosted
version: "14.3.1" version: "15.0.0"
web: web:
dependency: transitive dependency: transitive
description: description: