Compare commits
12 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 5a314ee2bd | |||
| 2743f92283 | |||
| 72a9d5711a | |||
| 1edb2804f1 | |||
| c9c52b39fa | |||
| 7a0265ad8d | |||
| 58e53d0aeb | |||
| 06ef2ab36b | |||
| 0362bf2013 | |||
| 73b96b82f7 | |||
| c78a90dbfe | |||
| df025babd5 |
@@ -2,6 +2,8 @@
|
|||||||
<uses-permission android:name="android.permission.USE_BIOMETRIC"/>
|
<uses-permission android:name="android.permission.USE_BIOMETRIC"/>
|
||||||
<uses-permission android:name="android.permission.USE_FINGERPRINT"/>
|
<uses-permission android:name="android.permission.USE_FINGERPRINT"/>
|
||||||
<uses-permission android:name="android.permission.INTERNET"/>
|
<uses-permission android:name="android.permission.INTERNET"/>
|
||||||
|
<uses-permission android:name="android.permission.SEND_SMS"/>
|
||||||
|
<uses-permission android:name="android.permission.READ_PHONE_STATE"/>
|
||||||
<application
|
<application
|
||||||
android:label="kmobile"
|
android:label="kmobile"
|
||||||
android:name="${applicationName}"
|
android:name="${applicationName}"
|
||||||
|
|||||||
@@ -1,5 +1,12 @@
|
|||||||
package com.example.kmobile
|
package com.example.kmobile
|
||||||
|
|
||||||
import io.flutter.embedding.android.FlutterFragmentActivity
|
import io.flutter.embedding.android.FlutterFragmentActivity
|
||||||
|
import io.flutter.embedding.android.FlutterActivity
|
||||||
|
import android.view.WindowManager.LayoutParams
|
||||||
|
import android.os.Bundle
|
||||||
|
|
||||||
class MainActivity: FlutterFragmentActivity()
|
class MainActivity: FlutterFragmentActivity() {
|
||||||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
|
super.onCreate(savedInstanceState)
|
||||||
|
window.addFlags(LayoutParams.FLAG_SECURE)
|
||||||
|
}
|
||||||
|
}
|
||||||
File diff suppressed because one or more lines are too long
@@ -40,8 +40,7 @@ class BeneficiaryService {
|
|||||||
} on DioException catch (e) {
|
} on DioException catch (e) {
|
||||||
if (e.response?.statusCode == 404) {
|
if (e.response?.statusCode == 404) {
|
||||||
throw Exception('INVALID IFSC CODE');
|
throw Exception('INVALID IFSC CODE');
|
||||||
}
|
} else if (e.response?.statusCode == 401) {
|
||||||
else if (e.response?.statusCode == 401) {
|
|
||||||
throw Exception('INVALID IFSC CODE');
|
throw Exception('INVALID IFSC CODE');
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
|
|||||||
@@ -10,10 +10,7 @@ class ChangePasswordService {
|
|||||||
}) async {
|
}) async {
|
||||||
final response = await _dio.post(
|
final response = await _dio.post(
|
||||||
'/api/otp/send',
|
'/api/otp/send',
|
||||||
data: {
|
data: {'mobileNumber': mobileNumber, 'type': "CHANGE_LPWORD"},
|
||||||
'mobileNumber': mobileNumber,
|
|
||||||
'type': "CHANGE_LPWORD"
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
if (response.statusCode != 200) {
|
if (response.statusCode != 200) {
|
||||||
throw Exception("Invalid Mobile Number/Type");
|
throw Exception("Invalid Mobile Number/Type");
|
||||||
@@ -27,10 +24,7 @@ class ChangePasswordService {
|
|||||||
}) async {
|
}) async {
|
||||||
final response = await _dio.post(
|
final response = await _dio.post(
|
||||||
'/api/otp/send',
|
'/api/otp/send',
|
||||||
data: {
|
data: {'mobileNumber': mobileNumber, 'type': "CHANGE_TPIN"},
|
||||||
'mobileNumber': mobileNumber,
|
|
||||||
'type': "CHANGE_TPIN"
|
|
||||||
},
|
|
||||||
);
|
);
|
||||||
if (response.statusCode != 200) {
|
if (response.statusCode != 200) {
|
||||||
throw Exception("Invalid Mobile Number/Type");
|
throw Exception("Invalid Mobile Number/Type");
|
||||||
@@ -39,7 +33,6 @@ class ChangePasswordService {
|
|||||||
return response.toString();
|
return response.toString();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Future validateOtp({
|
Future validateOtp({
|
||||||
required String otp,
|
required String otp,
|
||||||
required String mobileNumber,
|
required String mobileNumber,
|
||||||
@@ -47,7 +40,7 @@ class ChangePasswordService {
|
|||||||
final response = await _dio.post(
|
final response = await _dio.post(
|
||||||
'/api/otp/verify?mobileNumber=$mobileNumber',
|
'/api/otp/verify?mobileNumber=$mobileNumber',
|
||||||
data: {
|
data: {
|
||||||
'otp' : otp,
|
'otp': otp,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
if (response.statusCode != 200) {
|
if (response.statusCode != 200) {
|
||||||
|
|||||||
129
lib/api/services/send_sms_service.dart
Normal file
129
lib/api/services/send_sms_service.dart
Normal file
@@ -0,0 +1,129 @@
|
|||||||
|
// // ignore_for_file: avoid_print
|
||||||
|
// import 'dart:io';
|
||||||
|
// import 'package:flutter/material.dart';
|
||||||
|
// import 'send_sms.dart';
|
||||||
|
// import 'package:simcards/sim_card.dart';
|
||||||
|
// import 'package:simcards/simcards.dart';
|
||||||
|
|
||||||
|
// import 'package:uuid/uuid.dart';
|
||||||
|
|
||||||
|
// class SmsService {
|
||||||
|
// final Simcards _simcards = Simcards();
|
||||||
|
|
||||||
|
// Future<void> sendVerificationSms({
|
||||||
|
// required BuildContext context,
|
||||||
|
// required String destinationNumber,
|
||||||
|
// required String message,
|
||||||
|
// }) async {
|
||||||
|
// try {
|
||||||
|
// await _simcards.requestPermission();
|
||||||
|
|
||||||
|
// bool permissionGranted = await _simcards.hasPermission();
|
||||||
|
// if (!permissionGranted) {
|
||||||
|
// print("Permission denied." );
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
|
||||||
|
// List<SimCard> simCardList = await _simcards.getSimCards();
|
||||||
|
// if (simCardList.isEmpty) {
|
||||||
|
// print("No SIM detected." );
|
||||||
|
// return;
|
||||||
|
// }
|
||||||
|
|
||||||
|
// await _sendSms(destinationNumber, message, simCardList.first);
|
||||||
|
|
||||||
|
// } catch (e) {
|
||||||
|
// print("Error in SMS process: $e");
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
|
||||||
|
// Future<void> _sendSms(
|
||||||
|
// String destinationNumber, String message, SimCard selectedSim) async {
|
||||||
|
// if (Platform.isAndroid) {
|
||||||
|
// try {
|
||||||
|
// var uuid = const Uuid();
|
||||||
|
// String uniqueId = uuid.v4();
|
||||||
|
|
||||||
|
// String smsMessage = uniqueId;
|
||||||
|
// String result = await sendSMS(
|
||||||
|
// message: smsMessage,
|
||||||
|
// recipients: [destinationNumber],
|
||||||
|
// sendDirect: true,
|
||||||
|
// );
|
||||||
|
// print("SMS send result: $result. Sent via ${selectedSim.displayName} (Note: OS default SIM isused).");
|
||||||
|
|
||||||
|
// } catch (e) {
|
||||||
|
// print("Error sending SMS: $e");
|
||||||
|
// }
|
||||||
|
// } else {
|
||||||
|
// print("SMS sending is only supported on Android.");
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
// ignore_for_file: avoid_print
|
||||||
|
import 'dart:io';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter_sms/flutter_sms.dart'; // <-- 1. IMPORT the new package
|
||||||
|
import 'package:simcards/sim_card.dart';
|
||||||
|
import 'package:simcards/simcards.dart';
|
||||||
|
import 'package:uuid/uuid.dart';
|
||||||
|
|
||||||
|
class SmsService {
|
||||||
|
final Simcards _simcards = Simcards();
|
||||||
|
|
||||||
|
Future<void> sendVerificationSms({
|
||||||
|
required BuildContext context,
|
||||||
|
required String destinationNumber,
|
||||||
|
required String message,
|
||||||
|
}) async {
|
||||||
|
try {
|
||||||
|
await _simcards.requestPermission();
|
||||||
|
|
||||||
|
bool permissionGranted = await _simcards.hasPermission();
|
||||||
|
if (!permissionGranted) {
|
||||||
|
print("Permission denied.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
List<SimCard> simCardList = await _simcards.getSimCards();
|
||||||
|
if (simCardList.isEmpty) {
|
||||||
|
print("No SIM detected.");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
await _sendSms(destinationNumber, message, simCardList.first);
|
||||||
|
} catch (e) {
|
||||||
|
print("Error in SMS process: $e");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _sendSms(
|
||||||
|
String destinationNumber, String message, SimCard selectedSim) async {
|
||||||
|
if (Platform.isAndroid) {
|
||||||
|
try {
|
||||||
|
var uuid = const Uuid();
|
||||||
|
String uniqueId = uuid.v4();
|
||||||
|
|
||||||
|
String smsMessage = uniqueId;
|
||||||
|
|
||||||
|
// v-- 2. UPDATE the function call below --v
|
||||||
|
String result = await sendSMS(
|
||||||
|
message: smsMessage,
|
||||||
|
recipients: [destinationNumber],
|
||||||
|
);
|
||||||
|
// ^-- The 'sendDirect' parameter is not available in this package. --^
|
||||||
|
// It will open the user's default messaging app with the fields pre-filled.
|
||||||
|
|
||||||
|
print(
|
||||||
|
"SMS send result: $result. Sent via ${selectedSim.displayName} (Note: OS default SIM isused)."
|
||||||
|
);
|
||||||
|
} catch (e) {
|
||||||
|
print("Error sending SMS: $e");
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
print("SMS sending is only supported on Android.");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
51
lib/app.dart
51
lib/app.dart
@@ -20,6 +20,7 @@ import 'features/dashboard/screens/dashboard_screen.dart';
|
|||||||
import 'features/auth/screens/mpin_screen.dart';
|
import 'features/auth/screens/mpin_screen.dart';
|
||||||
import 'package:local_auth/local_auth.dart';
|
import 'package:local_auth/local_auth.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
|
import 'dart:async';
|
||||||
|
|
||||||
class KMobile extends StatefulWidget {
|
class KMobile extends StatefulWidget {
|
||||||
const KMobile({super.key});
|
const KMobile({super.key});
|
||||||
@@ -34,13 +35,15 @@ class KMobile extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class _KMobileState extends State<KMobile> {
|
class _KMobileState extends State<KMobile> with WidgetsBindingObserver {
|
||||||
|
Timer? _backgroundTimer;
|
||||||
bool showSplash = true;
|
bool showSplash = true;
|
||||||
Locale? _locale;
|
Locale? _locale;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
|
WidgetsBinding.instance.addObserver(this);
|
||||||
loadPreferences();
|
loadPreferences();
|
||||||
Future.delayed(const Duration(seconds: 3), () {
|
Future.delayed(const Duration(seconds: 3), () {
|
||||||
setState(() {
|
setState(() {
|
||||||
@@ -49,10 +52,35 @@ class _KMobileState extends State<KMobile> {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
WidgetsBinding.instance.removeObserver(this);
|
||||||
|
_backgroundTimer?.cancel();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void didChangeAppLifecycleState(AppLifecycleState state) {
|
||||||
|
super.didChangeAppLifecycleState(state);
|
||||||
|
switch (state) {
|
||||||
|
case AppLifecycleState.resumed:
|
||||||
|
_backgroundTimer?.cancel();
|
||||||
|
break;
|
||||||
|
case AppLifecycleState.paused:
|
||||||
|
_backgroundTimer = Timer(const Duration(minutes: 2), () {
|
||||||
|
if (Platform.isAndroid) {
|
||||||
|
SystemNavigator.pop();
|
||||||
|
}
|
||||||
|
exit(0);
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Future<void> loadPreferences() async {
|
Future<void> loadPreferences() async {
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
|
|
||||||
// Load Locale
|
|
||||||
final String? langCode = prefs.getString('locale');
|
final String? langCode = prefs.getString('locale');
|
||||||
if (langCode != null) {
|
if (langCode != null) {
|
||||||
setState(() {
|
setState(() {
|
||||||
@@ -117,7 +145,6 @@ class _KMobileState extends State<KMobile> {
|
|||||||
|
|
||||||
class AuthGate extends StatefulWidget {
|
class AuthGate extends StatefulWidget {
|
||||||
const AuthGate({super.key});
|
const AuthGate({super.key});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<AuthGate> createState() => _AuthGateState();
|
State<AuthGate> createState() => _AuthGateState();
|
||||||
}
|
}
|
||||||
@@ -180,7 +207,6 @@ class _AuthGateState extends State<AuthGate> {
|
|||||||
if (_checking) {
|
if (_checking) {
|
||||||
return const SplashScreen();
|
return const SplashScreen();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (_isLoggedIn) {
|
if (_isLoggedIn) {
|
||||||
if (_hasMPin) {
|
if (_hasMPin) {
|
||||||
if (_biometricEnabled) {
|
if (_biometricEnabled) {
|
||||||
@@ -190,11 +216,9 @@ class _AuthGateState extends State<AuthGate> {
|
|||||||
if (snapshot.connectionState == ConnectionState.waiting) {
|
if (snapshot.connectionState == ConnectionState.waiting) {
|
||||||
return const SplashScreen();
|
return const SplashScreen();
|
||||||
}
|
}
|
||||||
|
|
||||||
if (snapshot.data == true) {
|
if (snapshot.data == true) {
|
||||||
return const NavigationScaffold(); // Authenticated
|
return const NavigationScaffold();
|
||||||
}
|
}
|
||||||
|
|
||||||
return MPinScreen(
|
return MPinScreen(
|
||||||
mode: MPinMode.enter,
|
mode: MPinMode.enter,
|
||||||
onCompleted: (_) {
|
onCompleted: (_) {
|
||||||
@@ -225,7 +249,6 @@ class _AuthGateState extends State<AuthGate> {
|
|||||||
onCompleted: (_) async {
|
onCompleted: (_) async {
|
||||||
final storage = getIt<SecureStorage>();
|
final storage = getIt<SecureStorage>();
|
||||||
final localAuth = LocalAuthentication();
|
final localAuth = LocalAuthentication();
|
||||||
|
|
||||||
final optIn = await showDialog<bool>(
|
final optIn = await showDialog<bool>(
|
||||||
context: context,
|
context: context,
|
||||||
barrierDismissible: false,
|
barrierDismissible: false,
|
||||||
@@ -246,7 +269,6 @@ class _AuthGateState extends State<AuthGate> {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|
||||||
if (optIn == true) {
|
if (optIn == true) {
|
||||||
final canCheck = await localAuth.canCheckBiometrics;
|
final canCheck = await localAuth.canCheckBiometrics;
|
||||||
bool didAuth = false;
|
bool didAuth = false;
|
||||||
@@ -254,7 +276,6 @@ class _AuthGateState extends State<AuthGate> {
|
|||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
authEnable = AppLocalizations.of(context).authenticateToEnable;
|
authEnable = AppLocalizations.of(context).authenticateToEnable;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (canCheck) {
|
if (canCheck) {
|
||||||
didAuth = await localAuth.authenticate(
|
didAuth = await localAuth.authenticate(
|
||||||
localizedReason: authEnable,
|
localizedReason: authEnable,
|
||||||
@@ -269,7 +290,6 @@ class _AuthGateState extends State<AuthGate> {
|
|||||||
await storage.write('biometric_enabled', 'false');
|
await storage.write('biometric_enabled', 'false');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (context.mounted) {
|
if (context.mounted) {
|
||||||
Navigator.of(context).pushReplacement(
|
Navigator.of(context).pushReplacement(
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
@@ -287,7 +307,6 @@ class _AuthGateState extends State<AuthGate> {
|
|||||||
|
|
||||||
class NavigationScaffold extends StatefulWidget {
|
class NavigationScaffold extends StatefulWidget {
|
||||||
const NavigationScaffold({super.key});
|
const NavigationScaffold({super.key});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<NavigationScaffold> createState() => _NavigationScaffoldState();
|
State<NavigationScaffold> createState() => _NavigationScaffoldState();
|
||||||
}
|
}
|
||||||
@@ -295,7 +314,6 @@ class NavigationScaffold extends StatefulWidget {
|
|||||||
class _NavigationScaffoldState extends State<NavigationScaffold> {
|
class _NavigationScaffoldState extends State<NavigationScaffold> {
|
||||||
final PageController _pageController = PageController();
|
final PageController _pageController = PageController();
|
||||||
int _selectedIndex = 0;
|
int _selectedIndex = 0;
|
||||||
|
|
||||||
final List<Widget> _pages = [
|
final List<Widget> _pages = [
|
||||||
const DashboardScreen(),
|
const DashboardScreen(),
|
||||||
const CardManagementScreen(),
|
const CardManagementScreen(),
|
||||||
@@ -344,8 +362,7 @@ class _NavigationScaffoldState extends State<NavigationScaffold> {
|
|||||||
type: BottomNavigationBarType.fixed,
|
type: BottomNavigationBarType.fixed,
|
||||||
backgroundColor: const Color(0XFF1E58AD),
|
backgroundColor: const Color(0XFF1E58AD),
|
||||||
selectedItemColor: Theme.of(context).colorScheme.onPrimary,
|
selectedItemColor: Theme.of(context).colorScheme.onPrimary,
|
||||||
unselectedItemColor:
|
unselectedItemColor: Theme.of(context).colorScheme.onSecondary,
|
||||||
Theme.of(context).colorScheme.onSecondary,
|
|
||||||
onTap: (index) {
|
onTap: (index) {
|
||||||
setState(() {
|
setState(() {
|
||||||
_selectedIndex = index;
|
_selectedIndex = index;
|
||||||
@@ -372,11 +389,9 @@ class _NavigationScaffoldState extends State<NavigationScaffold> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add this widget at the end of the file
|
|
||||||
class BiometricPromptScreen extends StatelessWidget {
|
class BiometricPromptScreen extends StatelessWidget {
|
||||||
final VoidCallback onCompleted;
|
final VoidCallback onCompleted;
|
||||||
const BiometricPromptScreen({super.key, required this.onCompleted});
|
const BiometricPromptScreen({super.key, required this.onCompleted});
|
||||||
|
|
||||||
Future<void> _handleBiometric(BuildContext context) async {
|
Future<void> _handleBiometric(BuildContext context) async {
|
||||||
final localAuth = LocalAuthentication();
|
final localAuth = LocalAuthentication();
|
||||||
final canCheck = await localAuth.canCheckBiometrics;
|
final canCheck = await localAuth.canCheckBiometrics;
|
||||||
|
|||||||
15
lib/core/logger.dart
Normal file
15
lib/core/logger.dart
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
import 'package:kmobile/core/toast.dart';
|
||||||
|
|
||||||
|
class Logger {
|
||||||
|
static void info(String message) {
|
||||||
|
showToast('INFO: $message');
|
||||||
|
}
|
||||||
|
|
||||||
|
static void warning(String message) {
|
||||||
|
showToast('WARNING: $message');
|
||||||
|
}
|
||||||
|
|
||||||
|
static void error(String message) {
|
||||||
|
showToast('ERROR: $message');
|
||||||
|
}
|
||||||
|
}
|
||||||
14
lib/core/toast.dart
Normal file
14
lib/core/toast.dart
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:fluttertoast/fluttertoast.dart';
|
||||||
|
|
||||||
|
void showToast(String message) {
|
||||||
|
Fluttertoast.showToast(
|
||||||
|
msg: message,
|
||||||
|
toastLength: Toast.LENGTH_SHORT,
|
||||||
|
gravity: ToastGravity.BOTTOM,
|
||||||
|
timeInSecForIosWeb: 1,
|
||||||
|
backgroundColor: Colors.black,
|
||||||
|
textColor: Colors.white,
|
||||||
|
fontSize: 16.0,
|
||||||
|
);
|
||||||
|
}
|
||||||
@@ -49,12 +49,10 @@ class AuthRepository {
|
|||||||
_tokenExpiryKey, token.expiresAt.toIso8601String());
|
_tokenExpiryKey, token.expiresAt.toIso8601String());
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> clearAuthTokens() async {
|
Future<void> clearAuthTokens() async {
|
||||||
await _secureStorage.deleteAll();
|
await _secureStorage.deleteAll();
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
Future<AuthToken?> _getAuthToken() async {
|
Future<AuthToken?> _getAuthToken() async {
|
||||||
final accessToken = await _secureStorage.read(_accessTokenKey);
|
final accessToken = await _secureStorage.read(_accessTokenKey);
|
||||||
final expiryString = await _secureStorage.read(_tokenExpiryKey);
|
final expiryString = await _secureStorage.read(_tokenExpiryKey);
|
||||||
|
|||||||
@@ -1,5 +1,3 @@
|
|||||||
|
|
||||||
|
|
||||||
import 'package:dio/dio.dart';
|
import 'package:dio/dio.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
import 'package:kmobile/data/models/transaction.dart';
|
import 'package:kmobile/data/models/transaction.dart';
|
||||||
|
|||||||
@@ -49,7 +49,9 @@ Future<void> setupDependencies() async {
|
|||||||
getIt.registerSingleton<NeftService>(NeftService(getIt<Dio>()));
|
getIt.registerSingleton<NeftService>(NeftService(getIt<Dio>()));
|
||||||
getIt.registerSingleton<RtgsService>(RtgsService(getIt<Dio>()));
|
getIt.registerSingleton<RtgsService>(RtgsService(getIt<Dio>()));
|
||||||
getIt.registerSingleton<ImpsService>(ImpsService(getIt<Dio>()));
|
getIt.registerSingleton<ImpsService>(ImpsService(getIt<Dio>()));
|
||||||
getIt.registerLazySingleton<ChangePasswordService>(() => ChangePasswordService(getIt<Dio>()),);
|
getIt.registerLazySingleton<ChangePasswordService>(
|
||||||
|
() => ChangePasswordService(getIt<Dio>()),
|
||||||
|
);
|
||||||
|
|
||||||
// Add auth interceptor after repository is available
|
// Add auth interceptor after repository is available
|
||||||
getIt<Dio>().interceptors.add(
|
getIt<Dio>().interceptors.add(
|
||||||
|
|||||||
@@ -11,8 +11,8 @@ import 'transaction_details_screen.dart';
|
|||||||
import 'package:pdf/widgets.dart' as pw;
|
import 'package:pdf/widgets.dart' as pw;
|
||||||
import 'package:permission_handler/permission_handler.dart';
|
import 'package:permission_handler/permission_handler.dart';
|
||||||
import 'package:device_info_plus/device_info_plus.dart';
|
import 'package:device_info_plus/device_info_plus.dart';
|
||||||
import 'package:path_provider/path_provider.dart';
|
import 'package:path_provider/path_provider.dart';
|
||||||
import 'package:share_plus/share_plus.dart';
|
import 'package:share_plus/share_plus.dart';
|
||||||
|
|
||||||
class AccountStatementScreen extends StatefulWidget {
|
class AccountStatementScreen extends StatefulWidget {
|
||||||
final String accountNo;
|
final String accountNo;
|
||||||
@@ -305,10 +305,11 @@ class _AccountStatementScreen extends State<AccountStatementScreen> {
|
|||||||
),
|
),
|
||||||
Text(
|
Text(
|
||||||
"Bal: ₹${tx.balance}",
|
"Bal: ₹${tx.balance}",
|
||||||
style: const TextStyle(fontSize: 12), // Style matches tx.name
|
style: const TextStyle(
|
||||||
|
fontSize: 12), // Style matches tx.name
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
@@ -491,7 +492,7 @@ class _AccountStatementScreen extends State<AccountStatementScreen> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
// Add for IOS
|
// Add for IOS
|
||||||
else if (Platform.isIOS) {
|
else if (Platform.isIOS) {
|
||||||
// On iOS, we save to a temporary directory and then open the share sheet.
|
// On iOS, we save to a temporary directory and then open the share sheet.
|
||||||
final tempDir = await getTemporaryDirectory();
|
final tempDir = await getTemporaryDirectory();
|
||||||
final file = await File('${tempDir.path}/$fileName').create();
|
final file = await File('${tempDir.path}/$fileName').create();
|
||||||
@@ -501,9 +502,7 @@ else if (Platform.isIOS) {
|
|||||||
await Share.shareXFiles(
|
await Share.shareXFiles(
|
||||||
[XFile(file.path)],
|
[XFile(file.path)],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
|||||||
@@ -73,8 +73,8 @@ class TransactionDetailsScreen extends StatelessWidget {
|
|||||||
// AppLocalizations.of(context).beneficiaryAccountNo,
|
// AppLocalizations.of(context).beneficiaryAccountNo,
|
||||||
// transaction.name.split("A/C ").last ?? "")
|
// transaction.name.split("A/C ").last ?? "")
|
||||||
// ]
|
// ]
|
||||||
_buildDetailRow(AppLocalizations.of(context).details,
|
_buildDetailRow(
|
||||||
transaction.name),
|
AppLocalizations.of(context).details, transaction.name),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -16,11 +16,11 @@ class ThemeState extends Equatable {
|
|||||||
List<Object?> get props => [themeType];
|
List<Object?> get props => [themeType];
|
||||||
}*/
|
}*/
|
||||||
|
|
||||||
|
|
||||||
import 'package:equatable/equatable.dart';
|
import 'package:equatable/equatable.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:kmobile/config/theme_type.dart';
|
import 'package:kmobile/config/theme_type.dart';
|
||||||
import 'package:kmobile/config/themes.dart';
|
import 'package:kmobile/config/themes.dart';
|
||||||
|
|
||||||
class ThemeState extends Equatable {
|
class ThemeState extends Equatable {
|
||||||
final ThemeType themeType;
|
final ThemeType themeType;
|
||||||
const ThemeState({required this.themeType});
|
const ThemeState({required this.themeType});
|
||||||
@@ -35,7 +35,4 @@ class ThemeState extends Equatable {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
List<Object?> get props => [themeType];
|
List<Object?> get props => [themeType];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -140,7 +140,8 @@ class _SetPasswordScreenState extends State<SetPasswordScreen> {
|
|||||||
if (_error != null) ...[
|
if (_error != null) ...[
|
||||||
Text(
|
Text(
|
||||||
_error!,
|
_error!,
|
||||||
style: const TextStyle(color: Colors.red,
|
style: const TextStyle(
|
||||||
|
color: Colors.red,
|
||||||
fontWeight: FontWeight.bold,
|
fontWeight: FontWeight.bold,
|
||||||
fontSize: 20),
|
fontSize: 20),
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
|
import 'package:package_info_plus/package_info_plus.dart';
|
||||||
import '../../../l10n/app_localizations.dart';
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
import 'package:kmobile/api/services/send_sms_service.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
|
||||||
class SplashScreen extends StatefulWidget {
|
class SplashScreen extends StatefulWidget {
|
||||||
@@ -10,11 +11,39 @@ class SplashScreen extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _SplashScreenState extends State<SplashScreen> {
|
class _SplashScreenState extends State<SplashScreen> {
|
||||||
|
String _version = '';
|
||||||
|
final SmsService _smsService = SmsService();
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_loadVersion();
|
||||||
|
_sendInitialSms();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _sendInitialSms() async {
|
||||||
|
await _smsService.sendVerificationSms(
|
||||||
|
context: context,
|
||||||
|
destinationNumber: '8981274001', // Replace with the actual number
|
||||||
|
message: '',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _loadVersion() async {
|
||||||
|
final PackageInfo info = await PackageInfo.fromPlatform();
|
||||||
|
if (mounted) {
|
||||||
|
// Check if the widget is still in the tree
|
||||||
|
setState(() {
|
||||||
|
_version = 'Version ${info.version} (${info.buildNumber})';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
body: Stack(
|
body: Stack(
|
||||||
children: [
|
fit: StackFit.expand,
|
||||||
|
children: <Widget>[
|
||||||
Positioned.fill(
|
Positioned.fill(
|
||||||
child: Image.asset(
|
child: Image.asset(
|
||||||
'assets/images/kconnect2.webp',
|
'assets/images/kconnect2.webp',
|
||||||
@@ -51,8 +80,20 @@ class _SplashScreenState extends State<SplashScreen> {
|
|||||||
left: 0,
|
left: 0,
|
||||||
right: 0,
|
right: 0,
|
||||||
child: Center(
|
child: Center(
|
||||||
child: CircularProgressIndicator(
|
child: CircularProgressIndicator(color: Color(0xFFFFFFFF)),
|
||||||
color: Color(0xFFFFFFFF)),
|
),
|
||||||
|
),
|
||||||
|
Positioned(
|
||||||
|
bottom: 90,
|
||||||
|
left: 0,
|
||||||
|
right: 0,
|
||||||
|
child: Text(
|
||||||
|
_version,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: const TextStyle(
|
||||||
|
color: Color(0xFFFFFFFF),
|
||||||
|
fontSize: 14,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -24,8 +24,8 @@ class AddBeneficiaryScreen extends StatefulWidget {
|
|||||||
class _AddBeneficiaryScreen extends State<AddBeneficiaryScreen> {
|
class _AddBeneficiaryScreen extends State<AddBeneficiaryScreen> {
|
||||||
final _formKey = GlobalKey<FormState>();
|
final _formKey = GlobalKey<FormState>();
|
||||||
final _accountNumberFieldKey = GlobalKey<FormFieldState>();
|
final _accountNumberFieldKey = GlobalKey<FormFieldState>();
|
||||||
final _confirmAccountNumberFieldKey = GlobalKey<FormFieldState>();
|
final _confirmAccountNumberFieldKey = GlobalKey<FormFieldState>();
|
||||||
final _ifscFieldKey = GlobalKey<FormFieldState>();
|
final _ifscFieldKey = GlobalKey<FormFieldState>();
|
||||||
final TextEditingController accountNumberController = TextEditingController();
|
final TextEditingController accountNumberController = TextEditingController();
|
||||||
final TextEditingController confirmAccountNumberController =
|
final TextEditingController confirmAccountNumberController =
|
||||||
TextEditingController();
|
TextEditingController();
|
||||||
@@ -50,7 +50,7 @@ final _ifscFieldKey = GlobalKey<FormFieldState>();
|
|||||||
if (!_ifscFocusNode.hasFocus && ifscController.text.trim().length == 11) {
|
if (!_ifscFocusNode.hasFocus && ifscController.text.trim().length == 11) {
|
||||||
_validateIFSC();
|
_validateIFSC();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
setState(() {
|
setState(() {
|
||||||
accountType = 'Savings';
|
accountType = 'Savings';
|
||||||
@@ -94,7 +94,8 @@ final _ifscFieldKey = GlobalKey<FormFieldState>();
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
final errorMessage = e.toString().toUpperCase();
|
final errorMessage = e.toString().toUpperCase();
|
||||||
String snackbarMessage = AppLocalizations.of(context).somethingWentWrong;
|
String snackbarMessage =
|
||||||
|
AppLocalizations.of(context).somethingWentWrong;
|
||||||
|
|
||||||
if (errorMessage.contains('INVALID') && errorMessage.contains('IFSC')) {
|
if (errorMessage.contains('INVALID') && errorMessage.contains('IFSC')) {
|
||||||
snackbarMessage = AppLocalizations.of(context).invalidIfsc;
|
snackbarMessage = AppLocalizations.of(context).invalidIfsc;
|
||||||
@@ -325,7 +326,9 @@ final _ifscFieldKey = GlobalKey<FormFieldState>();
|
|||||||
).reenterAccountNumber;
|
).reenterAccountNumber;
|
||||||
}
|
}
|
||||||
if (value != accountNumberController.text) {
|
if (value != accountNumberController.text) {
|
||||||
return AppLocalizations.of(context,).accountMismatch;
|
return AppLocalizations.of(
|
||||||
|
context,
|
||||||
|
).accountMismatch;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
@@ -429,20 +432,27 @@ final _ifscFieldKey = GlobalKey<FormFieldState>();
|
|||||||
child: SizedBox(
|
child: SizedBox(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
child: ElevatedButton(
|
child: ElevatedButton(
|
||||||
onPressed: _isValidating || ifscController.text.length != 11
|
onPressed: _isValidating ||
|
||||||
? null
|
ifscController.text.length != 11
|
||||||
: () {
|
? null
|
||||||
|
: () {
|
||||||
final isAccountValid =
|
final isAccountValid =
|
||||||
_accountNumberFieldKey.currentState!.validate();
|
_accountNumberFieldKey.currentState!
|
||||||
|
.validate();
|
||||||
final isConfirmAccountValid =
|
final isConfirmAccountValid =
|
||||||
_confirmAccountNumberFieldKey.currentState!.validate();
|
_confirmAccountNumberFieldKey
|
||||||
final isIfscValid = _ifscFieldKey.currentState!.validate();
|
.currentState!
|
||||||
|
.validate();
|
||||||
|
final isIfscValid = _ifscFieldKey
|
||||||
|
.currentState!
|
||||||
|
.validate();
|
||||||
|
|
||||||
if (isAccountValid && isConfirmAccountValid && isIfscValid) {
|
if (isAccountValid &&
|
||||||
|
isConfirmAccountValid &&
|
||||||
|
isIfscValid) {
|
||||||
_validateBeneficiary();
|
_validateBeneficiary();
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
child: _isValidating
|
child: _isValidating
|
||||||
? const SizedBox(
|
? const SizedBox(
|
||||||
width: 20,
|
width: 20,
|
||||||
|
|||||||
@@ -90,7 +90,10 @@ class _BeneficiaryResultPageState extends State<BeneficiaryResultPage> {
|
|||||||
),
|
),
|
||||||
child: Text(
|
child: Text(
|
||||||
AppLocalizations.of(context).done,
|
AppLocalizations.of(context).done,
|
||||||
style: TextStyle(color: Theme.of(context).colorScheme.onPrimaryContainer), // slightly bigger text
|
style: TextStyle(
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onPrimaryContainer), // slightly bigger text
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -30,7 +30,8 @@ class DashboardScreen extends StatefulWidget {
|
|||||||
State<DashboardScreen> createState() => _DashboardScreenState();
|
State<DashboardScreen> createState() => _DashboardScreenState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _DashboardScreenState extends State<DashboardScreen> {
|
class _DashboardScreenState extends State<DashboardScreen>
|
||||||
|
with SingleTickerProviderStateMixin {
|
||||||
int selectedAccountIndex = 0;
|
int selectedAccountIndex = 0;
|
||||||
bool isVisible = false;
|
bool isVisible = false;
|
||||||
bool isRefreshing = false;
|
bool isRefreshing = false;
|
||||||
@@ -211,19 +212,8 @@ class _DashboardScreenState extends State<DashboardScreen> {
|
|||||||
backgroundColor: theme.scaffoldBackgroundColor,
|
backgroundColor: theme.scaffoldBackgroundColor,
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
backgroundColor: theme.scaffoldBackgroundColor,
|
backgroundColor: theme.scaffoldBackgroundColor,
|
||||||
automaticallyImplyLeading: false,
|
leading: Padding(
|
||||||
title: Text(
|
padding: const EdgeInsets.only(left: 10.0),
|
||||||
AppLocalizations.of(context).kccbMobile,
|
|
||||||
textAlign: TextAlign.left,
|
|
||||||
style: TextStyle(
|
|
||||||
color: theme.colorScheme.primary,
|
|
||||||
fontWeight: FontWeight.w700,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
centerTitle: true,
|
|
||||||
actions: [
|
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(right: 10.0),
|
|
||||||
child: InkWell(
|
child: InkWell(
|
||||||
borderRadius: BorderRadius.circular(20),
|
borderRadius: BorderRadius.circular(20),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
@@ -231,15 +221,18 @@ class _DashboardScreenState extends State<DashboardScreen> {
|
|||||||
String mobileNumberToPass = '';
|
String mobileNumberToPass = '';
|
||||||
|
|
||||||
if (authState is Authenticated) {
|
if (authState is Authenticated) {
|
||||||
if (selectedAccountIndex >= 0 && selectedAccountIndex < authState.users.length) {
|
if (selectedAccountIndex >= 0 &&
|
||||||
mobileNumberToPass = authState.users[selectedAccountIndex].mobileNo ?? '';
|
selectedAccountIndex < authState.users.length) {
|
||||||
|
mobileNumberToPass =
|
||||||
|
authState.users[selectedAccountIndex].mobileNo ?? '';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
builder: (context) => ProfileScreen(mobileNumber: mobileNumberToPass),
|
builder: (context) =>
|
||||||
|
ProfileScreen(mobileNumber: mobileNumberToPass),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@@ -255,7 +248,15 @@ class _DashboardScreenState extends State<DashboardScreen> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
title: Text(
|
||||||
|
AppLocalizations.of(context).kccbMobile,
|
||||||
|
textAlign: TextAlign.left,
|
||||||
|
style: TextStyle(
|
||||||
|
color: theme.colorScheme.primary,
|
||||||
|
fontWeight: FontWeight.w700,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
centerTitle: true,
|
||||||
),
|
),
|
||||||
body: BlocBuilder<AuthCubit, AuthState>(
|
body: BlocBuilder<AuthCubit, AuthState>(
|
||||||
builder: (context, state) {
|
builder: (context, state) {
|
||||||
@@ -533,7 +534,8 @@ class _DashboardScreenState extends State<DashboardScreen> {
|
|||||||
.accountNo!,
|
.accountNo!,
|
||||||
balance: users[selectedAccountIndex]
|
balance: users[selectedAccountIndex]
|
||||||
.availableBalance!,
|
.availableBalance!,
|
||||||
accountType: users[selectedAccountIndex]
|
accountType:
|
||||||
|
users[selectedAccountIndex]
|
||||||
.accountType!,
|
.accountType!,
|
||||||
)));
|
)));
|
||||||
}),
|
}),
|
||||||
|
|||||||
@@ -72,4 +72,3 @@ class AccountCard extends StatelessWidget {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -78,10 +78,9 @@ class _EnquiryScreen extends State<EnquiryScreen> {
|
|||||||
const SizedBox(height: 20),
|
const SizedBox(height: 20),
|
||||||
GestureDetector(
|
GestureDetector(
|
||||||
onTap: () => _launchUrl("https://kccb.in/complaint-form"),
|
onTap: () => _launchUrl("https://kccb.in/complaint-form"),
|
||||||
child: Row(
|
child: Row(mainAxisSize: MainAxisSize.min, children: [
|
||||||
mainAxisSize: MainAxisSize.min,
|
Text(
|
||||||
children: [
|
"Complaint Form",
|
||||||
Text("Complaint Form",
|
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: 17,
|
fontSize: 17,
|
||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme.of(context).colorScheme.primary,
|
||||||
@@ -94,10 +93,7 @@ class _EnquiryScreen extends State<EnquiryScreen> {
|
|||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme.of(context).colorScheme.primary,
|
||||||
size: 16.0,
|
size: 16.0,
|
||||||
),
|
),
|
||||||
]
|
])),
|
||||||
)
|
|
||||||
),
|
|
||||||
|
|
||||||
const SizedBox(height: 40),
|
const SizedBox(height: 40),
|
||||||
Text(
|
Text(
|
||||||
AppLocalizations.of(context).keyContacts,
|
AppLocalizations.of(context).keyContacts,
|
||||||
|
|||||||
@@ -213,7 +213,8 @@ class _PaymentAnimationScreenState extends State<PaymentAnimationScreen> {
|
|||||||
),
|
),
|
||||||
label: Text(
|
label: Text(
|
||||||
AppLocalizations.of(context).share,
|
AppLocalizations.of(context).share,
|
||||||
style: TextStyle(color: Theme.of(context).colorScheme.primary),
|
style: TextStyle(
|
||||||
|
color: Theme.of(context).colorScheme.primary),
|
||||||
),
|
),
|
||||||
style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
padding: const EdgeInsets.symmetric(
|
padding: const EdgeInsets.symmetric(
|
||||||
|
|||||||
@@ -19,7 +19,8 @@ class _TpinOtpScreenState extends State<TpinOtpScreen> {
|
|||||||
(_) => TextEditingController(),
|
(_) => TextEditingController(),
|
||||||
);
|
);
|
||||||
bool _isLoading = false;
|
bool _isLoading = false;
|
||||||
final ChangePasswordService _changePasswordService = getIt<ChangePasswordService>();
|
final ChangePasswordService _changePasswordService =
|
||||||
|
getIt<ChangePasswordService>();
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void dispose() {
|
void dispose() {
|
||||||
@@ -33,7 +34,7 @@ class _TpinOtpScreenState extends State<TpinOtpScreen> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void _onOtpChanged(int idx, String value) {
|
void _onOtpChanged(int idx, String value) {
|
||||||
if (value.length == 1 && idx <5) {
|
if (value.length == 1 && idx < 5) {
|
||||||
_focusNodes[idx + 1].requestFocus();
|
_focusNodes[idx + 1].requestFocus();
|
||||||
}
|
}
|
||||||
if (value.isEmpty && idx > 0) {
|
if (value.isEmpty && idx > 0) {
|
||||||
@@ -44,7 +45,7 @@ class _TpinOtpScreenState extends State<TpinOtpScreen> {
|
|||||||
|
|
||||||
String get _enteredOtp => _controllers.map((c) => c.text).join();
|
String get _enteredOtp => _controllers.map((c) => c.text).join();
|
||||||
|
|
||||||
void _verifyOtp() async {
|
void _verifyOtp() async {
|
||||||
setState(() {
|
setState(() {
|
||||||
_isLoading = true;
|
_isLoading = true;
|
||||||
});
|
});
|
||||||
@@ -74,7 +75,7 @@ void _verifyOtp() async {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@@ -200,4 +201,4 @@ void _verifyOtp() async {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,10 +15,11 @@ class TpinSetupPromptScreen extends StatefulWidget {
|
|||||||
State<TpinSetupPromptScreen> createState() => _TpinSetupPromptScreenState();
|
State<TpinSetupPromptScreen> createState() => _TpinSetupPromptScreenState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _TpinSetupPromptScreenState extends State<TpinSetupPromptScreen> {
|
class _TpinSetupPromptScreenState extends State<TpinSetupPromptScreen> {
|
||||||
int selectedAccountIndex = 0;
|
int selectedAccountIndex = 0;
|
||||||
bool _isLoading = false;
|
bool _isLoading = false;
|
||||||
final ChangePasswordService _changePasswordService = getIt<ChangePasswordService>();
|
final ChangePasswordService _changePasswordService =
|
||||||
|
getIt<ChangePasswordService>();
|
||||||
Future<void> _getOtp() async {
|
Future<void> _getOtp() async {
|
||||||
setState(() {
|
setState(() {
|
||||||
_isLoading = true;
|
_isLoading = true;
|
||||||
@@ -28,15 +29,20 @@ class TpinSetupPromptScreen extends StatefulWidget {
|
|||||||
final authState = context.read<AuthCubit>().state;
|
final authState = context.read<AuthCubit>().state;
|
||||||
String mobileNumberToPass = '';
|
String mobileNumberToPass = '';
|
||||||
if (authState is Authenticated) {
|
if (authState is Authenticated) {
|
||||||
if (selectedAccountIndex >= 0 && selectedAccountIndex < authState.users.length) {
|
if (selectedAccountIndex >= 0 &&
|
||||||
mobileNumberToPass = authState.users[selectedAccountIndex].mobileNo ?? '';
|
selectedAccountIndex < authState.users.length) {
|
||||||
|
mobileNumberToPass =
|
||||||
|
authState.users[selectedAccountIndex].mobileNo ?? '';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
await _changePasswordService.getOtpTpin(mobileNumber: mobileNumberToPass);
|
await _changePasswordService.getOtpTpin(mobileNumber: mobileNumberToPass);
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
Navigator.pushReplacement(
|
Navigator.pushReplacement(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(builder: (_) => TpinOtpScreen(mobileNumber: mobileNumberToPass,)),
|
MaterialPageRoute(
|
||||||
|
builder: (_) => TpinOtpScreen(
|
||||||
|
mobileNumber: mobileNumberToPass,
|
||||||
|
)),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@@ -52,7 +58,7 @@ class TpinSetupPromptScreen extends StatefulWidget {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@@ -86,7 +92,7 @@ class TpinSetupPromptScreen extends StatefulWidget {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 32),
|
const SizedBox(height: 32),
|
||||||
ElevatedButton.icon(
|
ElevatedButton.icon(
|
||||||
icon: _isLoading
|
icon: _isLoading
|
||||||
? const SizedBox(
|
? const SizedBox(
|
||||||
width: 20,
|
width: 20,
|
||||||
@@ -99,7 +105,8 @@ ElevatedButton.icon(
|
|||||||
: const Icon(Icons.arrow_forward_rounded),
|
: const Icon(Icons.arrow_forward_rounded),
|
||||||
label: Text(
|
label: Text(
|
||||||
AppLocalizations.of(context).setTpin,
|
AppLocalizations.of(context).setTpin,
|
||||||
style: const TextStyle(fontSize: 18, fontWeight: FontWeight.w600),
|
style:
|
||||||
|
const TextStyle(fontSize: 18, fontWeight: FontWeight.w600),
|
||||||
),
|
),
|
||||||
style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
backgroundColor: theme.colorScheme.onPrimary,
|
backgroundColor: theme.colorScheme.onPrimary,
|
||||||
@@ -111,8 +118,9 @@ ElevatedButton.icon(
|
|||||||
borderRadius: BorderRadius.circular(10),
|
borderRadius: BorderRadius.circular(10),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
onPressed: _isLoading ? null : _getOtp, // <-- Use the new function
|
onPressed:
|
||||||
),
|
_isLoading ? null : _getOtp, // <-- Use the new function
|
||||||
|
),
|
||||||
const SizedBox(height: 18),
|
const SizedBox(height: 18),
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.symmetric(horizontal: 18.0),
|
padding: const EdgeInsets.symmetric(horizontal: 18.0),
|
||||||
|
|||||||
@@ -212,4 +212,3 @@ class _TransactionPinScreenState extends State<TransactionPinScreen>
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -67,13 +67,23 @@ class _TransactionSuccessScreen extends State<TransactionSuccessScreen> {
|
|||||||
const SizedBox(height: 6),
|
const SizedBox(height: 6),
|
||||||
Text(
|
Text(
|
||||||
"On $transactionDate",
|
"On $transactionDate",
|
||||||
style: TextStyle(fontSize: 14, color: Theme.of(context).colorScheme.onSurface.withOpacity(0.6)),
|
style: TextStyle(
|
||||||
|
fontSize: 14,
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onSurface
|
||||||
|
.withOpacity(0.6)),
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 16),
|
const SizedBox(height: 16),
|
||||||
Text(
|
Text(
|
||||||
"${AppLocalizations.of(context).toAccountNumber}: $creditAccount",
|
"${AppLocalizations.of(context).toAccountNumber}: $creditAccount",
|
||||||
style: TextStyle(fontSize: 12, color: Theme.of(context).colorScheme.onSurface.withOpacity(0.8)),
|
style: TextStyle(
|
||||||
|
fontSize: 12,
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onSurface
|
||||||
|
.withOpacity(0.8)),
|
||||||
textAlign: TextAlign.center,
|
textAlign: TextAlign.center,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
@@ -91,8 +101,12 @@ class _TransactionSuccessScreen extends State<TransactionSuccessScreen> {
|
|||||||
child: OutlinedButton.icon(
|
child: OutlinedButton.icon(
|
||||||
onPressed: _shareScreenshot,
|
onPressed: _shareScreenshot,
|
||||||
icon: const Icon(Icons.share, size: 18),
|
icon: const Icon(Icons.share, size: 18),
|
||||||
label: Text(AppLocalizations.of(context).share,
|
label: Text(
|
||||||
style: TextStyle(color: Theme.of(context).colorScheme.onPrimaryContainer),
|
AppLocalizations.of(context).share,
|
||||||
|
style: TextStyle(
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onPrimaryContainer),
|
||||||
),
|
),
|
||||||
style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
shape: const StadiumBorder(),
|
shape: const StadiumBorder(),
|
||||||
|
|||||||
@@ -19,7 +19,8 @@ class ChangePasswordOTPScreen extends StatefulWidget {
|
|||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<ChangePasswordOTPScreen> createState() => _ChangePasswordOTPScreenState();
|
State<ChangePasswordOTPScreen> createState() =>
|
||||||
|
_ChangePasswordOTPScreenState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _ChangePasswordOTPScreenState extends State<ChangePasswordOTPScreen> {
|
class _ChangePasswordOTPScreenState extends State<ChangePasswordOTPScreen> {
|
||||||
@@ -36,6 +37,7 @@ class _ChangePasswordOTPScreenState extends State<ChangePasswordOTPScreen> {
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
final changePasswordService = getIt<ChangePasswordService>();
|
final changePasswordService = getIt<ChangePasswordService>();
|
||||||
Future<void> _validateOTP() async {
|
Future<void> _validateOTP() async {
|
||||||
try {
|
try {
|
||||||
@@ -57,12 +59,11 @@ class _ChangePasswordOTPScreenState extends State<ChangePasswordOTPScreen> {
|
|||||||
|
|
||||||
// Navigate back to profile or login
|
// Navigate back to profile or login
|
||||||
Navigator.of(context).popUntil((route) => route.isFirst);
|
Navigator.of(context).popUntil((route) => route.isFirst);
|
||||||
|
} catch (e) {
|
||||||
} catch (e) {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
SnackBar(content: Text(AppLocalizations.of(context).invalidOtp)),
|
SnackBar(content: Text(AppLocalizations.of(context).invalidOtp)),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
|
|||||||
@@ -56,15 +56,14 @@ class _ChangePasswordScreenState extends State<ChangePasswordScreen> {
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
final ChangePasswordService _changePasswordService = getIt<ChangePasswordService>();
|
|
||||||
void _proceed() async {
|
final ChangePasswordService _changePasswordService =
|
||||||
|
getIt<ChangePasswordService>();
|
||||||
|
void _proceed() async {
|
||||||
if (_formKey.currentState!.validate()) {
|
if (_formKey.currentState!.validate()) {
|
||||||
|
|
||||||
|
|
||||||
try {
|
try {
|
||||||
await _changePasswordService.getOtp(mobileNumber: widget.mobileNumber);
|
await _changePasswordService.getOtp(mobileNumber: widget.mobileNumber);
|
||||||
|
|
||||||
|
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
@@ -78,16 +77,19 @@ void _proceed() async {
|
|||||||
);
|
);
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
SnackBar(content: Text('${AppLocalizations.of(context).failedtosentOTP}: $e')),
|
SnackBar(
|
||||||
|
content:
|
||||||
|
Text('${AppLocalizations.of(context).failedtosentOTP}: $e')),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(title: Text(AppLocalizations.of(context).changeLoginPassword)),
|
appBar:
|
||||||
|
AppBar(title: Text(AppLocalizations.of(context).changeLoginPassword)),
|
||||||
body: Padding(
|
body: Padding(
|
||||||
padding: const EdgeInsets.all(16),
|
padding: const EdgeInsets.all(16),
|
||||||
child: Form(
|
child: Form(
|
||||||
@@ -103,8 +105,8 @@ void _proceed() async {
|
|||||||
icon: Icon(_showCurrentPassword
|
icon: Icon(_showCurrentPassword
|
||||||
? Icons.visibility
|
? Icons.visibility
|
||||||
: Icons.visibility_off),
|
: Icons.visibility_off),
|
||||||
onPressed: () =>
|
onPressed: () => setState(
|
||||||
setState(() => _showCurrentPassword = !_showCurrentPassword),
|
() => _showCurrentPassword = !_showCurrentPassword),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
validator: validateCurrentPassword,
|
validator: validateCurrentPassword,
|
||||||
@@ -135,8 +137,8 @@ void _proceed() async {
|
|||||||
icon: Icon(_showConfirmPassword
|
icon: Icon(_showConfirmPassword
|
||||||
? Icons.visibility
|
? Icons.visibility
|
||||||
: Icons.visibility_off),
|
: Icons.visibility_off),
|
||||||
onPressed: () =>
|
onPressed: () => setState(
|
||||||
setState(() => _showConfirmPassword = !_showConfirmPassword),
|
() => _showConfirmPassword = !_showConfirmPassword),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
validator: validateConfirmPassword,
|
validator: validateConfirmPassword,
|
||||||
|
|||||||
@@ -8,8 +8,8 @@ class LogoutDialog extends StatelessWidget {
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return AlertDialog(
|
return AlertDialog(
|
||||||
title: Text(AppLocalizations.of(context).logout),
|
title: Text(AppLocalizations.of(context).deregister),
|
||||||
content: Text(AppLocalizations.of(context).logoutCheck),
|
content: Text(AppLocalizations.of(context).deregistercheck),
|
||||||
actions: [
|
actions: [
|
||||||
TextButton(
|
TextButton(
|
||||||
onPressed: () => Navigator.pop(context, false), // dismiss
|
onPressed: () => Navigator.pop(context, false), // dismiss
|
||||||
|
|||||||
@@ -1,7 +1,12 @@
|
|||||||
|
import 'dart:io';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
import 'package:kmobile/data/repositories/auth_repository.dart';
|
import 'package:kmobile/data/repositories/auth_repository.dart';
|
||||||
import 'package:kmobile/features/profile/change_password/change_password_screen.dart';
|
import 'package:kmobile/features/profile/change_password/change_password_screen.dart';
|
||||||
import 'package:kmobile/features/profile/logout_dialog.dart';
|
import 'package:kmobile/features/profile/logout_dialog.dart';
|
||||||
|
import 'package:kmobile/security/secure_storage.dart';
|
||||||
|
import 'package:local_auth/local_auth.dart';
|
||||||
|
import 'package:package_info_plus/package_info_plus.dart';
|
||||||
import 'package:shared_preferences/shared_preferences.dart';
|
import 'package:shared_preferences/shared_preferences.dart';
|
||||||
import '../../di/injection.dart';
|
import '../../di/injection.dart';
|
||||||
import '../../l10n/app_localizations.dart';
|
import '../../l10n/app_localizations.dart';
|
||||||
@@ -16,6 +21,26 @@ class ProfileScreen extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _ProfileScreenState extends State<ProfileScreen> {
|
class _ProfileScreenState extends State<ProfileScreen> {
|
||||||
|
bool _isBiometricEnabled = false;
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_loadBiometricStatus();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<String> _getAppVersion() async {
|
||||||
|
final PackageInfo info = await PackageInfo.fromPlatform();
|
||||||
|
return 'Version ${info.version} (${info.buildNumber})';
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _loadBiometricStatus() async {
|
||||||
|
final storage = getIt<SecureStorage>();
|
||||||
|
final isEnabled = await storage.read('biometric_enabled');
|
||||||
|
setState(() {
|
||||||
|
_isBiometricEnabled = isEnabled == 'true';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
Future<void> _handleLogout(BuildContext context) async {
|
Future<void> _handleLogout(BuildContext context) async {
|
||||||
final auth = getIt<AuthRepository>();
|
final auth = getIt<AuthRepository>();
|
||||||
final prefs = await SharedPreferences.getInstance();
|
final prefs = await SharedPreferences.getInstance();
|
||||||
@@ -25,6 +50,90 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
Navigator.pushNamedAndRemoveUntil(context, '/login', (route) => false);
|
Navigator.pushNamedAndRemoveUntil(context, '/login', (route) => false);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> _handleBiometricToggle(bool enable) async {
|
||||||
|
final localAuth = LocalAuthentication();
|
||||||
|
final storage = getIt<SecureStorage>();
|
||||||
|
final canCheck = await localAuth.canCheckBiometrics;
|
||||||
|
|
||||||
|
if (!canCheck) {
|
||||||
|
// Optional: Show a snackbar or dialog if biometrics are not available
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text(AppLocalizations.of(context).biometricsNotAvailable)),
|
||||||
|
);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (enable) {
|
||||||
|
// Show "Enable" dialog
|
||||||
|
final optIn = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (ctx) => AlertDialog(
|
||||||
|
title: Text(AppLocalizations.of(context).enableFingerprintLogin),
|
||||||
|
content: Text(AppLocalizations.of(context).enableFingerprintMessage),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.of(ctx).pop(false),
|
||||||
|
child: Text(AppLocalizations.of(context).no),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.of(ctx).pop(true),
|
||||||
|
child: Text(AppLocalizations.of(context).yes),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (optIn == true) {
|
||||||
|
try {
|
||||||
|
final didAuth = await localAuth.authenticate(
|
||||||
|
localizedReason: AppLocalizations.of(context).authenticateToEnable,
|
||||||
|
options: const AuthenticationOptions(
|
||||||
|
stickyAuth: true,
|
||||||
|
biometricOnly: true,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
if (didAuth) {
|
||||||
|
await storage.write('biometric_enabled', 'true');
|
||||||
|
setState(() {
|
||||||
|
_isBiometricEnabled = true;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
// Handle authentication errors
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Show "Disable" dialog
|
||||||
|
final optOut = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false,
|
||||||
|
builder: (ctx) => AlertDialog(
|
||||||
|
title: Text(AppLocalizations.of(context).disableFingerprintLogin),
|
||||||
|
content: Text(AppLocalizations.of(context).disableFingerprintMessage),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.of(ctx).pop(false),
|
||||||
|
child: Text(AppLocalizations.of(context).no),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.of(ctx).pop(true),
|
||||||
|
child: Text(AppLocalizations.of(context).yes),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (optOut == true) {
|
||||||
|
await storage.write('biometric_enabled', 'false');
|
||||||
|
setState(() {
|
||||||
|
_isBiometricEnabled = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final loc = AppLocalizations.of(context);
|
final loc = AppLocalizations.of(context);
|
||||||
@@ -46,13 +155,22 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
SwitchListTile(
|
||||||
|
title: Text(AppLocalizations.of(context).enableFingerprintLogin),
|
||||||
|
value: _isBiometricEnabled,
|
||||||
|
onChanged: (bool value) {
|
||||||
|
_handleBiometricToggle(value);
|
||||||
|
},
|
||||||
|
secondary: const Icon(Icons.fingerprint),
|
||||||
|
),
|
||||||
ListTile(
|
ListTile(
|
||||||
leading: const Icon(Icons.password),
|
leading: const Icon(Icons.password),
|
||||||
title: Text(loc.changeLoginPassword),
|
title: Text(loc.changeLoginPassword),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(builder: (context) => ChangePasswordScreen(
|
MaterialPageRoute(
|
||||||
|
builder: (context) => ChangePasswordScreen(
|
||||||
mobileNumber: widget.mobileNumber,
|
mobileNumber: widget.mobileNumber,
|
||||||
)),
|
)),
|
||||||
);
|
);
|
||||||
@@ -71,8 +189,59 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
// },
|
// },
|
||||||
// ),
|
// ),
|
||||||
ListTile(
|
ListTile(
|
||||||
leading: const Icon(Icons.logout),
|
leading: const Icon(Icons.smartphone),
|
||||||
|
title: const Text("App Version"),
|
||||||
|
trailing: FutureBuilder<String>(
|
||||||
|
future: _getAppVersion(),
|
||||||
|
builder: (BuildContext context, AsyncSnapshot<String> snapshot) {
|
||||||
|
if (snapshot.connectionState == ConnectionState.waiting) {
|
||||||
|
// Show a loading indicator while waiting for the future to complete
|
||||||
|
return const CircularProgressIndicator();
|
||||||
|
} else if (snapshot.hasError) {
|
||||||
|
return const Text("Error");
|
||||||
|
} else {
|
||||||
|
// Display the version number once the future is complete
|
||||||
|
return Text(
|
||||||
|
snapshot.data ?? "N/A",
|
||||||
|
selectionColor: const Color(0xFFFFFFFF),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.exit_to_app),
|
||||||
title: Text(AppLocalizations.of(context).logout),
|
title: Text(AppLocalizations.of(context).logout),
|
||||||
|
onTap: () async {
|
||||||
|
final shouldExit = await showDialog<bool>(
|
||||||
|
context: context,
|
||||||
|
builder: (context) => AlertDialog(
|
||||||
|
title: Text(AppLocalizations.of(context).logout),
|
||||||
|
content: Text(AppLocalizations.of(context).logoutCheck),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.of(context).pop(false),
|
||||||
|
child: Text(AppLocalizations.of(context).no),
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.of(context).pop(true),
|
||||||
|
child: Text(AppLocalizations.of(context).yes),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (shouldExit == true) {
|
||||||
|
if (Platform.isAndroid) {
|
||||||
|
SystemNavigator.pop();
|
||||||
|
}
|
||||||
|
exit(0);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.logout),
|
||||||
|
title: Text(AppLocalizations.of(context).deregister),
|
||||||
onTap: () async {
|
onTap: () async {
|
||||||
final shouldLogout = await showDialog<bool>(
|
final shouldLogout = await showDialog<bool>(
|
||||||
context: context,
|
context: context,
|
||||||
|
|||||||
@@ -54,7 +54,7 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
if (!_ifscFocusNode.hasFocus && ifscController.text.trim().length == 11) {
|
if (!_ifscFocusNode.hasFocus && ifscController.text.trim().length == 11) {
|
||||||
_validateIFSC();
|
_validateIFSC();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
||||||
setState(() {
|
setState(() {
|
||||||
accountType = 'Savings';
|
accountType = 'Savings';
|
||||||
@@ -84,7 +84,8 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
} catch (e) {
|
} catch (e) {
|
||||||
if (mounted) {
|
if (mounted) {
|
||||||
final errorMessage = e.toString().toUpperCase();
|
final errorMessage = e.toString().toUpperCase();
|
||||||
String snackbarMessage = AppLocalizations.of(context).somethingWentWrong;
|
String snackbarMessage =
|
||||||
|
AppLocalizations.of(context).somethingWentWrong;
|
||||||
|
|
||||||
if (errorMessage.contains('INVALID') && errorMessage.contains('IFSC')) {
|
if (errorMessage.contains('INVALID') && errorMessage.contains('IFSC')) {
|
||||||
snackbarMessage = AppLocalizations.of(context).invalidIfsc;
|
snackbarMessage = AppLocalizations.of(context).invalidIfsc;
|
||||||
@@ -637,7 +638,8 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
child: SizedBox(
|
child: SizedBox(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
child: ElevatedButton(
|
child: ElevatedButton(
|
||||||
onPressed: _isValidating || ifscController.text.length != 11
|
onPressed:
|
||||||
|
_isValidating || ifscController.text.length != 11
|
||||||
? null
|
? null
|
||||||
: () {
|
: () {
|
||||||
if (confirmAccountNumberController.text ==
|
if (confirmAccountNumberController.text ==
|
||||||
|
|||||||
@@ -278,7 +278,7 @@ class _QuickPayWithinBankScreen extends State<QuickPayWithinBankScreen> {
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
const SizedBox(height: 25),
|
const SizedBox(height: 25),
|
||||||
TextFormField(
|
TextFormField(
|
||||||
controller: remarksController,
|
controller: remarksController,
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: AppLocalizations.of(context).remarks,
|
labelText: AppLocalizations.of(context).remarks,
|
||||||
@@ -295,7 +295,7 @@ TextFormField(
|
|||||||
color: Theme.of(context).colorScheme.primary, width: 2),
|
color: Theme.of(context).colorScheme.primary, width: 2),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 25),
|
const SizedBox(height: 25),
|
||||||
TextFormField(
|
TextFormField(
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
|
|||||||
@@ -5,7 +5,8 @@ import 'package:lottie/lottie.dart';
|
|||||||
class SecurityErrorScreen extends StatelessWidget {
|
class SecurityErrorScreen extends StatelessWidget {
|
||||||
final String message;
|
final String message;
|
||||||
|
|
||||||
const SecurityErrorScreen({Key? key, required this.message}) : super(key: key);
|
const SecurityErrorScreen({Key? key, required this.message})
|
||||||
|
: super(key: key);
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@@ -24,7 +25,8 @@ class SecurityErrorScreen extends StatelessWidget {
|
|||||||
),
|
),
|
||||||
const SizedBox(height: 40),
|
const SizedBox(height: 40),
|
||||||
ElevatedButton(
|
ElevatedButton(
|
||||||
onPressed: () => SystemChannels.platform.invokeMethod('SystemNavigator.pop'),
|
onPressed: () =>
|
||||||
|
SystemChannels.platform.invokeMethod('SystemNavigator.pop'),
|
||||||
child: const Text('Okay'),
|
child: const Text('Okay'),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -1,3 +1,5 @@
|
|||||||
|
// ignore_for_file: unused_element
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import '../../../l10n/app_localizations.dart';
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
@@ -27,7 +29,7 @@ class BranchLocatorScreen extends StatefulWidget {
|
|||||||
State<BranchLocatorScreen> createState() => _BranchLocatorScreenState();
|
State<BranchLocatorScreen> createState() => _BranchLocatorScreenState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
||||||
final TextEditingController _searchController = TextEditingController();
|
final TextEditingController _searchController = TextEditingController();
|
||||||
|
|
||||||
final List<Location> _allLocations = [
|
final List<Location> _allLocations = [
|
||||||
@@ -60,12 +62,12 @@ class BranchLocatorScreen extends StatefulWidget {
|
|||||||
List<Location> _filteredLocations = [];
|
List<Location> _filteredLocations = [];
|
||||||
bool _isLoading = false;
|
bool _isLoading = false;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
// _fetchAndSetLocations();
|
// _fetchAndSetLocations();
|
||||||
_filteredLocations = _allLocations;
|
_filteredLocations = _allLocations;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Example of a future API fetching function
|
// Example of a future API fetching function
|
||||||
/*
|
/*
|
||||||
@@ -88,7 +90,7 @@ Future<void> _fetchAndSetLocations() async {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
*/
|
*/
|
||||||
void _filterLocations(String query) {
|
void _filterLocations(String query) {
|
||||||
setState(() {
|
setState(() {
|
||||||
if (query.isEmpty) {
|
if (query.isEmpty) {
|
||||||
_filteredLocations = _allLocations;
|
_filteredLocations = _allLocations;
|
||||||
@@ -102,7 +104,7 @@ void _filterLocations(String query) {
|
|||||||
}).toList();
|
}).toList();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
@@ -128,7 +130,7 @@ void _filterLocations(String query) {
|
|||||||
),
|
),
|
||||||
|
|
||||||
// Content area
|
// Content area
|
||||||
Expanded(
|
Expanded(
|
||||||
child: _isLoading
|
child: _isLoading
|
||||||
? const Center(child: CircularProgressIndicator())
|
? const Center(child: CircularProgressIndicator())
|
||||||
: _filteredLocations.isEmpty
|
: _filteredLocations.isEmpty
|
||||||
@@ -140,7 +142,7 @@ Expanded(
|
|||||||
return _buildLocationItem(location);
|
return _buildLocationItem(location);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
@@ -157,10 +159,10 @@ Expanded(
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Helper widget to build a single location item
|
// Helper widget to build a single location item
|
||||||
Widget _buildLocationItem(Location location) {
|
Widget _buildLocationItem(Location location) {
|
||||||
final isBranch = location.type == LocationType.branch;
|
final isBranch = location.type == LocationType.branch;
|
||||||
return Card(
|
return Card(
|
||||||
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
||||||
@@ -182,5 +184,5 @@ Widget _buildLocationItem(Location location) {
|
|||||||
},
|
},
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -74,7 +74,7 @@ class _DailyLimitScreenState extends State<DailyLimitScreen> {
|
|||||||
_currentLimit = newLimit;
|
_currentLimit = newLimit;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void _removeLimit() {
|
void _removeLimit() {
|
||||||
setState(() {
|
setState(() {
|
||||||
|
|||||||
@@ -1,14 +1,14 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:kmobile/l10n/app_localizations.dart';
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
|
||||||
class FaqsScreen extends StatefulWidget {
|
class FaqsScreen extends StatefulWidget {
|
||||||
const FaqsScreen({super.key});
|
const FaqsScreen({super.key});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<FaqsScreen> createState() => _FaqsScreenState();
|
State<FaqsScreen> createState() => _FaqsScreenState();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _FaqsScreenState extends State<FaqsScreen> {
|
class _FaqsScreenState extends State<FaqsScreen> {
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
@@ -44,4 +44,4 @@ import 'package:kmobile/l10n/app_localizations.dart';
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
||||||
import 'package:kmobile/features/service/screens/quick_links_screen.dart';
|
import 'package:kmobile/features/service/screens/quick_links_screen.dart';
|
||||||
import 'package:kmobile/features/service/screens/faqs_screen.dart';
|
import 'package:kmobile/features/service/screens/faqs_screen.dart';
|
||||||
|
|
||||||
class ServiceScreen extends StatefulWidget {
|
class ServiceScreen extends StatefulWidget {
|
||||||
const ServiceScreen({super.key});
|
const ServiceScreen({super.key});
|
||||||
|
|
||||||
@@ -13,8 +14,8 @@ class ServiceScreen extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _ServiceScreen extends State<ServiceScreen> {
|
class _ServiceScreen extends State<ServiceScreen> {
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
automaticallyImplyLeading: false,
|
automaticallyImplyLeading: false,
|
||||||
@@ -44,7 +45,8 @@ Widget build(BuildContext context) {
|
|||||||
label: AppLocalizations.of(context).dailylimit,
|
label: AppLocalizations.of(context).dailylimit,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.of(context).push(
|
Navigator.of(context).push(
|
||||||
MaterialPageRoute(builder: (context) => const DailyLimitScreen()),
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const DailyLimitScreen()),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
disabled: true,
|
disabled: true,
|
||||||
@@ -55,7 +57,8 @@ Widget build(BuildContext context) {
|
|||||||
label: AppLocalizations.of(context).quickLinks,
|
label: AppLocalizations.of(context).quickLinks,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.of(context).push(
|
Navigator.of(context).push(
|
||||||
MaterialPageRoute(builder: (context) => const QuickLinksScreen()),
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const QuickLinksScreen()),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
disabled: true,
|
disabled: true,
|
||||||
@@ -87,7 +90,7 @@ Widget build(BuildContext context) {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class ServiceManagementTile extends StatelessWidget {
|
class ServiceManagementTile extends StatelessWidget {
|
||||||
|
|||||||
@@ -326,5 +326,10 @@
|
|||||||
"editLimit": "Edit Limit",
|
"editLimit": "Edit Limit",
|
||||||
"removeLimit": "Remove Limit",
|
"removeLimit": "Remove Limit",
|
||||||
"limitAmount": "Limit Amount",
|
"limitAmount": "Limit Amount",
|
||||||
"save": "Save"
|
"save": "Save",
|
||||||
|
"deregister": "De-Register",
|
||||||
|
"deregistercheck": "Are you sure you want to De-Register?",
|
||||||
|
"biometricsNotAvailable": "Biometrics not available on this device",
|
||||||
|
"disableFingerprintLogin": "Disable Fingerprint Login",
|
||||||
|
"disableFingerprintMessage": "Are you sure you want to disable fingerprint login?"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -327,5 +327,10 @@
|
|||||||
"editLimit": "सीमा संपादित करें",
|
"editLimit": "सीमा संपादित करें",
|
||||||
"removeLimit": "सीमा हटाएँ",
|
"removeLimit": "सीमा हटाएँ",
|
||||||
"limitAmount": "सीमा राशि",
|
"limitAmount": "सीमा राशि",
|
||||||
"save": "जमा करें"
|
"save": "जमा करें",
|
||||||
|
"deregister": "अपंजीकृत",
|
||||||
|
"deregistercheck": "क्या आप वाकई पंजीकरण रद्द करना चाहते हैं??",
|
||||||
|
"biometricsNotAvailable": "इस डिवाइस पर बायोमेट्रिक्स उपलब्ध नहीं है",
|
||||||
|
"disableFingerprintLogin": "फ़िंगरप्रिंट लॉगिन अक्षम करें",
|
||||||
|
"disableFingerprintMessage": "क्या आप फ़िंगरप्रिंट लॉगिन अक्षम करना चाहते हैं?"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
// ignore_for_file: unused_import
|
// ignore_for_file: unused_import
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
|
import 'package:kmobile/core/logger.dart';
|
||||||
import 'package:kmobile/features/security/security_error_screen.dart';
|
import 'package:kmobile/features/security/security_error_screen.dart';
|
||||||
import 'package:kmobile/security/security_service.dart';
|
import 'package:kmobile/security/security_service.dart';
|
||||||
import 'di/injection.dart';
|
import 'di/injection.dart';
|
||||||
@@ -8,6 +9,7 @@ import 'app.dart';
|
|||||||
|
|
||||||
void main() async {
|
void main() async {
|
||||||
WidgetsFlutterBinding.ensureInitialized();
|
WidgetsFlutterBinding.ensureInitialized();
|
||||||
|
Logger.info("App starting...");
|
||||||
|
|
||||||
await SystemChrome.setPreferredOrientations([
|
await SystemChrome.setPreferredOrientations([
|
||||||
DeviceOrientation.portraitUp,
|
DeviceOrientation.portraitUp,
|
||||||
@@ -17,11 +19,14 @@ void main() async {
|
|||||||
// Check for device compromise
|
// Check for device compromise
|
||||||
// final compromisedMessage = await SecurityService.deviceCompromisedMessage;
|
// final compromisedMessage = await SecurityService.deviceCompromisedMessage;
|
||||||
// if (compromisedMessage != null) {
|
// if (compromisedMessage != null) {
|
||||||
|
// Logger.error("Device compromised: $compromisedMessage");
|
||||||
// runApp(MaterialApp(
|
// runApp(MaterialApp(
|
||||||
// home: SecurityErrorScreen(message: compromisedMessage),
|
// home: SecurityErrorScreen(message: compromisedMessage),
|
||||||
// ));
|
// ));
|
||||||
// return;
|
// return;
|
||||||
// }
|
// }
|
||||||
|
Logger.info("Setting up dependencies...");
|
||||||
await setupDependencies();
|
await setupDependencies();
|
||||||
|
Logger.info("Dependencies set up.");
|
||||||
runApp(const KMobile());
|
runApp(const KMobile());
|
||||||
}
|
}
|
||||||
@@ -86,14 +86,14 @@ Widget getBankLogo(String? bankName, BuildContext context) {
|
|||||||
height: 40,
|
height: 40,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
if (bankName != null && bankName.toLowerCase().contains('ipsbank') || bankName != null && bankName.toLowerCase().contains('india post') ) {
|
if (bankName != null && bankName.toLowerCase().contains('ipsbank') ||
|
||||||
|
bankName != null && bankName.toLowerCase().contains('india post')) {
|
||||||
return Image.asset(
|
return Image.asset(
|
||||||
'assets/images/ipos_logo.png',
|
'assets/images/ipos_logo.png',
|
||||||
width: 40,
|
width: 40,
|
||||||
height: 40,
|
height: 40,
|
||||||
);
|
);
|
||||||
}
|
} else {
|
||||||
else {
|
|
||||||
return Icon(
|
return Icon(
|
||||||
Icons.account_balance,
|
Icons.account_balance,
|
||||||
size: 40,
|
size: 40,
|
||||||
|
|||||||
@@ -8,6 +8,7 @@ import Foundation
|
|||||||
import device_info_plus
|
import device_info_plus
|
||||||
import flutter_secure_storage_macos
|
import flutter_secure_storage_macos
|
||||||
import local_auth_darwin
|
import local_auth_darwin
|
||||||
|
import package_info_plus
|
||||||
import path_provider_foundation
|
import path_provider_foundation
|
||||||
import share_plus
|
import share_plus
|
||||||
import shared_preferences_foundation
|
import shared_preferences_foundation
|
||||||
@@ -17,6 +18,7 @@ func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
|
|||||||
DeviceInfoPlusMacosPlugin.register(with: registry.registrar(forPlugin: "DeviceInfoPlusMacosPlugin"))
|
DeviceInfoPlusMacosPlugin.register(with: registry.registrar(forPlugin: "DeviceInfoPlusMacosPlugin"))
|
||||||
FlutterSecureStoragePlugin.register(with: registry.registrar(forPlugin: "FlutterSecureStoragePlugin"))
|
FlutterSecureStoragePlugin.register(with: registry.registrar(forPlugin: "FlutterSecureStoragePlugin"))
|
||||||
FLALocalAuthPlugin.register(with: registry.registrar(forPlugin: "FLALocalAuthPlugin"))
|
FLALocalAuthPlugin.register(with: registry.registrar(forPlugin: "FLALocalAuthPlugin"))
|
||||||
|
FPPPackageInfoPlusPlugin.register(with: registry.registrar(forPlugin: "FPPPackageInfoPlusPlugin"))
|
||||||
PathProviderPlugin.register(with: registry.registrar(forPlugin: "PathProviderPlugin"))
|
PathProviderPlugin.register(with: registry.registrar(forPlugin: "PathProviderPlugin"))
|
||||||
SharePlusMacosPlugin.register(with: registry.registrar(forPlugin: "SharePlusMacosPlugin"))
|
SharePlusMacosPlugin.register(with: registry.registrar(forPlugin: "SharePlusMacosPlugin"))
|
||||||
SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin"))
|
SharedPreferencesPlugin.register(with: registry.registrar(forPlugin: "SharedPreferencesPlugin"))
|
||||||
|
|||||||
50
pubspec.lock
50
pubspec.lock
@@ -307,6 +307,14 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.1.2"
|
version: "3.1.2"
|
||||||
|
flutter_sms:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: flutter_sms
|
||||||
|
sha256: "2fe5f584f02596343557eeca56348f9b82413fefe83a423fab880cdbdf54d8d8"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.3.3"
|
||||||
flutter_svg:
|
flutter_svg:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@@ -333,6 +341,14 @@ packages:
|
|||||||
description: flutter
|
description: flutter
|
||||||
source: sdk
|
source: sdk
|
||||||
version: "0.0.0"
|
version: "0.0.0"
|
||||||
|
fluttertoast:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: fluttertoast
|
||||||
|
sha256: "90778fe0497fe3a09166e8cf2e0867310ff434b794526589e77ec03cf08ba8e8"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "8.2.14"
|
||||||
get_it:
|
get_it:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@@ -541,6 +557,22 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.0.0"
|
version: "1.0.0"
|
||||||
|
package_info_plus:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: package_info_plus
|
||||||
|
sha256: "7e76fad405b3e4016cd39d08f455a4eb5199723cf594cd1b8916d47140d93017"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "4.2.0"
|
||||||
|
package_info_plus_platform_interface:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: package_info_plus_platform_interface
|
||||||
|
sha256: "9bc8ba46813a4cc42c66ab781470711781940780fd8beddd0c3da62506d3a6c6"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.0.1"
|
||||||
path:
|
path:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
@@ -789,6 +821,22 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.0.0"
|
version: "3.0.0"
|
||||||
|
showcaseview:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: showcaseview
|
||||||
|
sha256: "3929adfcff53a8a9bc6b501914d67e4b7eae40451db7e654f76f34b0b30a185a"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "2.1.1"
|
||||||
|
simcards:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: simcards
|
||||||
|
sha256: b621cc265ebbb3e11009ca9be67063efbc011396c4224aff8b08edaba30fa5ae
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.0.1"
|
||||||
sky_engine:
|
sky_engine:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description: flutter
|
description: flutter
|
||||||
@@ -923,7 +971,7 @@ packages:
|
|||||||
source: hosted
|
source: hosted
|
||||||
version: "3.1.4"
|
version: "3.1.4"
|
||||||
uuid:
|
uuid:
|
||||||
dependency: transitive
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
name: uuid
|
name: uuid
|
||||||
sha256: a5be9ef6618a7ac1e964353ef476418026db906c4facdedaa299b7a2e71690ff
|
sha256: a5be9ef6618a7ac1e964353ef476418026db906c4facdedaa299b7a2e71690ff
|
||||||
|
|||||||
11
pubspec.yaml
11
pubspec.yaml
@@ -31,10 +31,6 @@ dependencies:
|
|||||||
flutter:
|
flutter:
|
||||||
sdk: flutter
|
sdk: flutter
|
||||||
flutter_neumorphic : 3.2.0
|
flutter_neumorphic : 3.2.0
|
||||||
|
|
||||||
|
|
||||||
# The following adds the Cupertino Icons font to your application.
|
|
||||||
# Use with the CupertinoIcons class for iOS style icons.
|
|
||||||
cupertino_icons: ^1.0.6
|
cupertino_icons: ^1.0.6
|
||||||
jailbreak_root_detection: ^1.1.6
|
jailbreak_root_detection: ^1.1.6
|
||||||
equatable: ^2.0.7
|
equatable: ^2.0.7
|
||||||
@@ -63,6 +59,13 @@ dependencies:
|
|||||||
pdf: ^3.11.3
|
pdf: ^3.11.3
|
||||||
permission_handler: ^12.0.1
|
permission_handler: ^12.0.1
|
||||||
device_info_plus: ^11.3.0
|
device_info_plus: ^11.3.0
|
||||||
|
showcaseview: ^2.0.3
|
||||||
|
package_info_plus: ^4.2.0
|
||||||
|
simcards: ^0.0.1
|
||||||
|
uuid: ^4.5.1
|
||||||
|
#send_message: ^1.0.0
|
||||||
|
flutter_sms: ^2.3.3
|
||||||
|
fluttertoast: ^8.2.6
|
||||||
# jailbreak_root_detection: "^1.1.6"
|
# jailbreak_root_detection: "^1.1.6"
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user