import 'dart:convert'; import 'dart:io'; import 'dart:ui'; import 'package:device_info_plus/device_info_plus.dart'; import 'package:flutter/cupertino.dart'; import 'package:flutter/material.dart'; import 'package:get/get.dart'; import 'package:jailbreak_root_detection/jailbreak_root_detection.dart'; import 'package:package_info_plus/package_info_plus.dart'; import 'package:url_launcher/url_launcher.dart'; import '../../constant/box_name.dart'; import '../../constant/colors.dart'; import '../../main.dart'; import '../../print.dart'; import 'encrypt_decrypt.dart'; Future checkForUpdate(BuildContext context) async { final packageInfo = await PackageInfo.fromPlatform(); final currentVersion = packageInfo.buildNumber; final version = packageInfo.version; print('currentVersion is : $currentVersion'); // Fetch the latest version from your server String latestVersion = box.read(BoxName.package); box.write(BoxName.packagInfo, version); if (latestVersion.isNotEmpty && latestVersion != currentVersion) { showUpdateDialog(context); } } checkForBounusInvitation() { if (box.read(BoxName.inviteCode) != null) {} } // Future getPackageInfo() async { // final response = await CRUD().get(link: AppLink.packageInfo, payload: { // "platform": Platform.isAndroid ? 'android' : 'ios', // "appName": AppInformation.appName, // }); // if (response != 'failure') { // return jsonDecode(response)['message'][0]['version']; // } // return ''; // } // getDeviceFingerprint() async { // final deviceInfo = await DeviceInfoPlugin().deviceInfo; // var deviceData; // if (Platform.isAndroid) { // deviceData = deviceInfo.data; // Log.print('deviceData: ${jsonEncode(deviceData)}'); // } else if (Platform.isIOS) { // deviceData = deviceInfo.data; // } // final String deviceId = // deviceData['device'] ?? deviceData['identifierForVendor']; // final String deviceModel = deviceData['model']; // final String osVersion = deviceData['systemVersion']; // Log.print('fingr: ${'${deviceId}_${deviceModel}_$osVersion'}'); // Log.print('deviceModel: ${deviceModel}'); // Log.print('osVersion: ${osVersion}'); // return EncryptionHelper.instance // .encryptData('${deviceId}_${deviceModel}_$osVersion'); // } void showUpdateDialog(BuildContext context) { final String storeUrl = Platform.isAndroid ? 'https://play.google.com/store/apps/details?id=com.mobileapp.store.ride' : 'https://apps.apple.com/ae/app/sefer/id6458734951'; showGeneralDialog( context: context, barrierDismissible: false, barrierColor: Colors.black.withOpacity(0.5), pageBuilder: (_, __, ___) { return BackdropFilter( filter: ImageFilter.blur(sigmaX: 5, sigmaY: 5), child: Center( child: AlertDialog( // Using AlertDialog for a more Material Design look shape: RoundedRectangleBorder( borderRadius: BorderRadius.circular(16)), // More rounded corners elevation: 4, // Add a bit more elevation contentPadding: EdgeInsets.zero, // Remove default content padding content: Column( mainAxisSize: MainAxisSize.min, children: [ Padding( padding: const EdgeInsets.only(top: 20.0), child: Image.asset( 'assets/images/logo.png', height: 72, // Slightly larger logo width: 72, ), ), const SizedBox(height: 16), Padding( padding: const EdgeInsets.symmetric(horizontal: 24.0), child: Text( 'Update Available'.tr, textAlign: TextAlign.center, style: Theme.of(context).textTheme.titleLarge?.copyWith( // Use theme's title style fontWeight: FontWeight.bold, ), ), ), Padding( padding: const EdgeInsets.all(24.0), child: Text( 'A new version of the app is available. Please update to the latest version.' .tr, // More encouraging message textAlign: TextAlign.center, style: Theme.of(context).textTheme.bodyMedium?.copyWith( // Use theme's body style color: Colors.black87, ), ), ), const Divider(height: 0), Row( children: [ Expanded( child: TextButton( // Using TextButton for "Cancel" onPressed: () => Navigator.pop(context), style: TextButton.styleFrom( foregroundColor: Colors.grey, shape: const RoundedRectangleBorder( borderRadius: BorderRadius.only( bottomLeft: Radius.circular(16), ), ), ), child: Text('Cancel'.tr), ), ), const SizedBox( height: 48, child: VerticalDivider(width: 0), // Using VerticalDivider ), Expanded( child: ElevatedButton( // Using ElevatedButton for "Update" onPressed: () async { if (await canLaunchUrl(Uri.parse(storeUrl))) { await launchUrl(Uri.parse(storeUrl)); } if (context.mounted) Navigator.pop(context); }, style: ElevatedButton.styleFrom( backgroundColor: AppColor .primaryColor, // Use theme's primary color foregroundColor: Theme.of(context) .colorScheme .onPrimary, // Use theme's onPrimary color shape: const RoundedRectangleBorder( borderRadius: BorderRadius.only( bottomRight: Radius.circular(16), ), ), ), child: Text('Update'.tr), ), ), ], ), ], ), ), ), ); }, transitionBuilder: (_, animation, __, child) { return ScaleTransition( scale: CurvedAnimation( parent: animation, curve: Curves.easeOutCubic, // More natural curve ), child: child, ); }, ); } class SecurityHelper { /// Performs security checks and handles potential risks static Future performSecurityChecks() async { bool isNotTrust = false; bool isJailBroken = false; bool isRealDevice = true; bool isOnExternalStorage = false; bool checkForIssues = false; bool isDevMode = false; bool isTampered = false; String bundleId = ""; try { isNotTrust = await JailbreakRootDetection.instance.isNotTrust; isJailBroken = await JailbreakRootDetection.instance.isJailBroken; isRealDevice = await JailbreakRootDetection.instance.isRealDevice; isOnExternalStorage = await JailbreakRootDetection.instance.isOnExternalStorage; List issues = await JailbreakRootDetection.instance.checkForIssues; checkForIssues = issues.isNotEmpty; isDevMode = await JailbreakRootDetection.instance.isDevMode; // Get Bundle ID PackageInfo packageInfo = await PackageInfo.fromPlatform(); bundleId = packageInfo.packageName; if (bundleId.isNotEmpty) { // Pass the CORRECT bundle ID to isTampered isTampered = await JailbreakRootDetection.instance.isTampered(bundleId); } } catch (e) { debugPrint("Error during security checks: $e"); // Consider handling specific exceptions, not just general errors. } // Save values to storage (using GetStorage) await box.write('isNotTrust', isNotTrust); // Use await for write operations await box.write('isTampered', isTampered); // Use await await box.write('isJailBroken', isJailBroken); // Use await // debugPrint("Security Check Results:"); // debugPrint("isNotTrust: $isNotTrust"); // debugPrint("isJailBroken: $isJailBroken"); // debugPrint("isRealDevice: $isRealDevice"); // debugPrint("isOnExternalStorage: $isOnExternalStorage"); // debugPrint("checkForIssues: $checkForIssues"); // debugPrint("isDevMode: $isDevMode"); // debugPrint("isTampered: $isTampered"); // debugPrint("Bundle ID: $bundleId"); // Print the bundle ID // Check for security risks and potentially show a warning if (isJailBroken || isRealDevice == false || isTampered) { // print("security_warning".tr); //using easy_localization // Use a more robust approach to show a warning, like a dialog: _showSecurityWarning(); } } /// Deletes all app data static void _showSecurityWarning() { // Use an RxInt to track the remaining seconds. This is the KEY! RxInt secondsRemaining = 10.obs; Get.dialog( CupertinoAlertDialog( title: Text("Security Warning".tr), content: Column( mainAxisSize: MainAxisSize.min, children: [ Obx(() => Text( "Potential security risks detected. The application will close in @seconds seconds." .trParams({ // Use trParams for placeholders 'seconds': secondsRemaining.value.toString(), }), // Wrap the Text widget in Obx )), SizedBox(height: 24), // More spacing before the progress bar Obx(() => SizedBox( width: double.infinity, // Make progress bar full width child: CupertinoActivityIndicator( // in case of loading radius: 15, animating: true, ))), SizedBox(height: 8), Obx(() => ClipRRect( borderRadius: BorderRadius.circular(8), // Rounded corners child: LinearProgressIndicator( value: secondsRemaining.value / 10, backgroundColor: Colors.grey.shade300, // Lighter background valueColor: AlwaysStoppedAnimation( CupertinoColors.systemRed), // iOS-style red minHeight: 8, // Slightly thicker progress bar ), )), ], ), ), barrierDismissible: false, ); Timer.periodic(Duration(seconds: 1), (timer) { secondsRemaining.value--; if (secondsRemaining.value <= 0) { timer.cancel(); // Get.back(); _clearDataAndExit(); } }); } static Future _clearDataAndExit() async { await storage.deleteAll(); await box.erase(); exit(0); // Exit the app print('exit'); } } class DeviceHelper { static Future getDeviceFingerprint() async { final DeviceInfoPlugin deviceInfoPlugin = DeviceInfoPlugin(); var deviceData; try { if (Platform.isAndroid) { // Fetch Android-specific device information AndroidDeviceInfo androidInfo = await deviceInfoPlugin.androidInfo; deviceData = androidInfo.toMap(); // Convert to a map for easier access // Log.print('deviceData: ${jsonEncode(deviceData)}'); } else if (Platform.isIOS) { // Fetch iOS-specific device information IosDeviceInfo iosInfo = await deviceInfoPlugin.iosInfo; deviceData = iosInfo.toMap(); // Convert to a map for easier access } else { throw UnsupportedError('Unsupported platform'); } // Extract relevant device information final String deviceId = Platform.isAndroid ? deviceData['androidId'] ?? deviceData['serialNumber'] ?? 'unknown' : deviceData['identifierForVendor'] ?? 'unknown'; final String deviceModel = deviceData['model'] ?? 'unknown'; final String osVersion = Platform.isAndroid ? deviceData['version']['release'] ?? 'unknown' : deviceData['systemVersion'] ?? 'unknown'; // Log the extracted information // Generate and return the encrypted fingerprint final String fingerprint = '${deviceId}_${deviceModel}_$osVersion'; // print(EncryptionHelper.instance.encryptData(fingerprint)); return EncryptionHelper.instance.encryptData(fingerprint); } catch (e) { throw Exception('Failed to generate device fingerprint'); } } }