1111
This commit is contained in:
26
.vscode/launch.json
vendored
Normal file
26
.vscode/launch.json
vendored
Normal 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"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
||||||
@@ -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: 'خطأ غير متوقع');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -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,
|
||||||
|
|||||||
16
pubspec.lock
16
pubspec.lock
@@ -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:
|
||||||
|
|||||||
Reference in New Issue
Block a user