Compare commits
22 Commits
b5acae85d5
...
testing
| Author | SHA1 | Date | |
|---|---|---|---|
| f15b8ac3f7 | |||
| 8f8fdb70e6 | |||
| d86ff2c427 | |||
| 527111c1de | |||
| dfbdb3238d | |||
| 3d13edf676 | |||
| 32e8b85cee | |||
| 58e53d0aeb | |||
| 06ef2ab36b | |||
| 0362bf2013 | |||
| 73b96b82f7 | |||
| c78a90dbfe | |||
| df025babd5 | |||
| 4d19bf6146 | |||
| d36cad31c1 | |||
| 39e7a02ca5 | |||
| 0c7470d74b | |||
| 32463680e8 | |||
| 00cb98ae83 | |||
| 18844495c2 | |||
| 3f7869677c | |||
| 0e4072fe8f |
@@ -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)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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,
|
||||||
|
|||||||
57
lib/api/services/limit_service.dart
Normal file
57
lib/api/services/limit_service.dart
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
// ignore_for_file: collection_methods_unrelated_type
|
||||||
|
import 'dart:developer';
|
||||||
|
import 'package:dio/dio.dart';
|
||||||
|
|
||||||
|
class Limit {
|
||||||
|
final double dailyLimit;
|
||||||
|
final double usedLimit;
|
||||||
|
|
||||||
|
Limit({
|
||||||
|
required this.dailyLimit,
|
||||||
|
required this.usedLimit,
|
||||||
|
});
|
||||||
|
|
||||||
|
factory Limit.fromJson(Map<String, dynamic> json) {
|
||||||
|
return Limit(
|
||||||
|
dailyLimit: json['dailyLimit']!,
|
||||||
|
usedLimit: json['usedLimit']!,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class LimitService {
|
||||||
|
final Dio _dio;
|
||||||
|
LimitService(this._dio);
|
||||||
|
|
||||||
|
Future<Limit> getLimit() async {
|
||||||
|
try {
|
||||||
|
final response = await _dio.get('/api/customer/daily-limit');
|
||||||
|
if (response.statusCode == 200) {
|
||||||
|
log('Response: ${response.data}');
|
||||||
|
return Limit.fromJson(response.data);
|
||||||
|
} else {
|
||||||
|
throw Exception('Failed to load');
|
||||||
|
}
|
||||||
|
} on DioException catch (e) {
|
||||||
|
throw Exception('Network error: ${e.message}');
|
||||||
|
} catch (e) {
|
||||||
|
throw Exception('Unexpected error: ${e.toString()}');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void editLimit( double newLimit) async {
|
||||||
|
try {
|
||||||
|
final response = await _dio.post('/api/customer/daily-limit',
|
||||||
|
data: '{"amount": $newLimit}');
|
||||||
|
if (response.statusCode == 200) {
|
||||||
|
log('Response: ${response.data}');
|
||||||
|
} else {
|
||||||
|
throw Exception('Failed to load');
|
||||||
|
}
|
||||||
|
} on DioException catch (e) {
|
||||||
|
throw Exception('Network error: ${e.message}');
|
||||||
|
} catch (e) {
|
||||||
|
throw Exception('Unexpected error: ${e.toString()}');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -10,7 +10,7 @@ class UserService {
|
|||||||
|
|
||||||
Future<List<User>> getUserDetails() async {
|
Future<List<User>> getUserDetails() async {
|
||||||
try {
|
try {
|
||||||
final response = await _dio.get('/api/customer/details');
|
final response = await _dio.get('/api/customer');
|
||||||
if (response.statusCode == 200) {
|
if (response.statusCode == 200) {
|
||||||
log('Response: ${response.data}');
|
log('Response: ${response.data}');
|
||||||
return (response.data as List)
|
return (response.data as List)
|
||||||
|
|||||||
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;
|
||||||
|
|||||||
@@ -51,10 +51,8 @@ class AuthRepository {
|
|||||||
|
|
||||||
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';
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import 'package:kmobile/api/services/limit_service.dart';
|
||||||
import 'package:kmobile/api/services/rtgs_service.dart';
|
import 'package:kmobile/api/services/rtgs_service.dart';
|
||||||
import 'package:kmobile/api/services/neft_service.dart';
|
import 'package:kmobile/api/services/neft_service.dart';
|
||||||
import 'package:kmobile/api/services/imps_service.dart';
|
import 'package:kmobile/api/services/imps_service.dart';
|
||||||
@@ -46,10 +47,13 @@ Future<void> setupDependencies() async {
|
|||||||
|
|
||||||
getIt.registerSingleton<PaymentService>(PaymentService(getIt<Dio>()));
|
getIt.registerSingleton<PaymentService>(PaymentService(getIt<Dio>()));
|
||||||
getIt.registerSingleton<BeneficiaryService>(BeneficiaryService(getIt<Dio>()));
|
getIt.registerSingleton<BeneficiaryService>(BeneficiaryService(getIt<Dio>()));
|
||||||
|
getIt.registerSingleton<LimitService>(LimitService(getIt<Dio>()));
|
||||||
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(
|
||||||
@@ -67,12 +71,13 @@ Dio _createDioClient() {
|
|||||||
baseUrl:
|
baseUrl:
|
||||||
// 'http://lb-test-mobile-banking-app-192209417.ap-south-1.elb.amazonaws.com:8080', //test
|
// 'http://lb-test-mobile-banking-app-192209417.ap-south-1.elb.amazonaws.com:8080', //test
|
||||||
//'http://lb-kccb-mobile-banking-app-848675342.ap-south-1.elb.amazonaws.com', //prod
|
//'http://lb-kccb-mobile-banking-app-848675342.ap-south-1.elb.amazonaws.com', //prod
|
||||||
'https://kccbmbnk.net',
|
'https://kccbmbnk.net', //prod small
|
||||||
connectTimeout: const Duration(seconds: 60),
|
connectTimeout: const Duration(seconds: 60),
|
||||||
receiveTimeout: const Duration(seconds: 60),
|
receiveTimeout: const Duration(seconds: 60),
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': 'application/json',
|
'Content-Type': 'application/json',
|
||||||
'Accept': 'application/json',
|
'Accept': 'application/json',
|
||||||
|
'X-Login-Type': 'MB',
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -305,7 +305,8 @@ 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
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@@ -502,8 +503,6 @@ else if (Platform.isIOS) {
|
|||||||
[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});
|
||||||
@@ -36,6 +36,3 @@ 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,3 +1,5 @@
|
|||||||
|
import 'package:package_info_plus/package_info_plus.dart';
|
||||||
|
|
||||||
import '../../../l10n/app_localizations.dart';
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
@@ -10,11 +12,29 @@ class SplashScreen extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _SplashScreenState extends State<SplashScreen> {
|
class _SplashScreenState extends State<SplashScreen> {
|
||||||
|
String _version = '';
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_loadVersion();
|
||||||
|
}
|
||||||
|
|
||||||
|
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 +71,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,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -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 ||
|
||||||
|
ifscController.text.length != 11
|
||||||
? null
|
? 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,
|
||||||
|
|||||||
@@ -65,7 +65,6 @@ class BeneficiaryDetailsScreen extends StatelessWidget {
|
|||||||
TextButton(
|
TextButton(
|
||||||
child: const Text('Delete'),
|
child: const Text('Delete'),
|
||||||
onPressed: () {
|
onPressed: () {
|
||||||
//Navigator.of(context).pop();
|
|
||||||
_deleteBeneficiary(context);
|
_deleteBeneficiary(context);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -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
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -13,6 +13,15 @@ class CustomerInfoScreen extends StatefulWidget {
|
|||||||
|
|
||||||
class _CustomerInfoScreenState extends State<CustomerInfoScreen> {
|
class _CustomerInfoScreenState extends State<CustomerInfoScreen> {
|
||||||
late final User user = widget.user;
|
late final User user = widget.user;
|
||||||
|
|
||||||
|
String _maskPrimaryId(String? primaryId) {
|
||||||
|
if (primaryId == null || primaryId.length <= 4) {
|
||||||
|
return primaryId ?? 'N/A';
|
||||||
|
}
|
||||||
|
final lastFour = primaryId.substring(primaryId.length - 4);
|
||||||
|
return '*' * (primaryId.length - 4) + lastFour;
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final theme = Theme.of(context);
|
final theme = Theme.of(context);
|
||||||
@@ -85,7 +94,7 @@ class _CustomerInfoScreenState extends State<CustomerInfoScreen> {
|
|||||||
), // Replace with Aadhar if available
|
), // Replace with Aadhar if available
|
||||||
InfoField(
|
InfoField(
|
||||||
label: AppLocalizations.of(context).primaryId,
|
label: AppLocalizations.of(context).primaryId,
|
||||||
value: user.primaryId ?? 'N/A',
|
value: _maskPrimaryId(user.primaryId),
|
||||||
), // Replace with PAN if available
|
), // Replace with PAN if available
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -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 {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -30,6 +30,16 @@ class _EnquiryScreen extends State<EnquiryScreen> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future<void> _launchUrl(String url) async {
|
||||||
|
final Uri uri = Uri.parse(url);
|
||||||
|
if (await canLaunchUrl(uri)) {
|
||||||
|
await launchUrl(uri);
|
||||||
|
} else {
|
||||||
|
// Consider adding a 'urlLaunchError' key to your AppLocalizations
|
||||||
|
debugPrint('Could not launch $url');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
Widget _buildContactItem(String role, String email, String phone) {
|
Widget _buildContactItem(String role, String email, String phone) {
|
||||||
return Column(
|
return Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
@@ -65,18 +75,26 @@ class _EnquiryScreen extends State<EnquiryScreen> {
|
|||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
// … existing Mail us / Call us / Write to us …
|
|
||||||
const SizedBox(height: 20),
|
const SizedBox(height: 20),
|
||||||
|
GestureDetector(
|
||||||
|
onTap: () => _launchUrl("https://kccb.in/complaint-form"),
|
||||||
|
child: Row(mainAxisSize: MainAxisSize.min, children: [
|
||||||
Text(
|
Text(
|
||||||
AppLocalizations.of(context).writeToUs,
|
"Complaint Form",
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 17,
|
||||||
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
decorationColor: Theme.of(context).colorScheme.primary,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 4),
|
|
||||||
Text(
|
|
||||||
"complaint@kccb.in",
|
|
||||||
style: TextStyle(color: Theme.of(context).colorScheme.primary),
|
|
||||||
),
|
),
|
||||||
|
const SizedBox(width: 4),
|
||||||
const SizedBox(height: 20),
|
Icon(
|
||||||
|
Icons.open_in_new,
|
||||||
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
size: 16.0,
|
||||||
|
),
|
||||||
|
])),
|
||||||
|
const SizedBox(height: 40),
|
||||||
Text(
|
Text(
|
||||||
AppLocalizations.of(context).keyContacts,
|
AppLocalizations.of(context).keyContacts,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
|
|||||||
@@ -2,6 +2,8 @@ import 'dart:async';
|
|||||||
import 'dart:convert';
|
import 'dart:convert';
|
||||||
import 'package:dio/dio.dart';
|
import 'package:dio/dio.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:intl/intl.dart';
|
||||||
|
import 'package:kmobile/api/services/limit_service.dart';
|
||||||
import 'package:kmobile/api/services/neft_service.dart';
|
import 'package:kmobile/api/services/neft_service.dart';
|
||||||
import 'package:kmobile/api/services/rtgs_service.dart';
|
import 'package:kmobile/api/services/rtgs_service.dart';
|
||||||
import 'package:kmobile/api/services/imps_service.dart';
|
import 'package:kmobile/api/services/imps_service.dart';
|
||||||
@@ -40,13 +42,67 @@ class FundTransferAmountScreen extends StatefulWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
class _FundTransferAmountScreenState extends State<FundTransferAmountScreen> {
|
class _FundTransferAmountScreenState extends State<FundTransferAmountScreen> {
|
||||||
|
final _limitService = getIt<LimitService>();
|
||||||
|
Limit? _limit;
|
||||||
|
bool _isLoadingLimit = true;
|
||||||
|
bool _isAmountOverLimit = false;
|
||||||
|
final _formatCurrency = NumberFormat.currency(locale: 'en_IN', symbol: '₹');
|
||||||
final _amountController = TextEditingController();
|
final _amountController = TextEditingController();
|
||||||
final _remarksController = TextEditingController();
|
final _remarksController = TextEditingController();
|
||||||
final _formKey = GlobalKey<FormState>();
|
final _formKey = GlobalKey<FormState>();
|
||||||
TransactionMode _selectedMode = TransactionMode.neft;
|
TransactionMode _selectedMode = TransactionMode.neft;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_loadLimit(); // Call the new method
|
||||||
|
_amountController.addListener(_checkAmountLimit);
|
||||||
|
}
|
||||||
|
Future<void> _loadLimit() async {
|
||||||
|
setState(() {
|
||||||
|
_isLoadingLimit = true;
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
final limitData = await _limitService.getLimit();
|
||||||
|
setState(() {
|
||||||
|
_limit = limitData;
|
||||||
|
_isLoadingLimit = false;
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
// Handle error if needed
|
||||||
|
setState(() {
|
||||||
|
_isLoadingLimit = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add this method to check the amount against the limit
|
||||||
|
void _checkAmountLimit() {
|
||||||
|
if (_limit == null) return;
|
||||||
|
|
||||||
|
final amount = double.tryParse(_amountController.text) ?? 0;
|
||||||
|
final remainingLimit = _limit!.dailyLimit - _limit!.usedLimit;
|
||||||
|
final bool isOverLimit = amount > remainingLimit;
|
||||||
|
|
||||||
|
if (isOverLimit) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text('Amount exceeds remaining daily limit of ${_formatCurrency.format(remainingLimit)}'),
|
||||||
|
backgroundColor: Colors.red,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_isAmountOverLimit != isOverLimit) {
|
||||||
|
setState(() {
|
||||||
|
_isAmountOverLimit = isOverLimit;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void dispose() {
|
void dispose() {
|
||||||
|
_amountController.removeListener(_checkAmountLimit);
|
||||||
_amountController.dispose();
|
_amountController.dispose();
|
||||||
_remarksController.dispose();
|
_remarksController.dispose();
|
||||||
super.dispose();
|
super.dispose();
|
||||||
@@ -429,6 +485,14 @@ class _FundTransferAmountScreenState extends State<FundTransferAmountScreen> {
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
if (_isLoadingLimit)
|
||||||
|
const Text('Fetching daily limit...'),
|
||||||
|
if (!_isLoadingLimit && _limit != null)
|
||||||
|
Text(
|
||||||
|
'Remaining Daily Limit: ${_formatCurrency.format(_limit!.dailyLimit - _limit!.usedLimit)}',
|
||||||
|
style: Theme.of(context).textTheme.bodySmall,
|
||||||
),
|
),
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
|
|
||||||
@@ -436,7 +500,7 @@ class _FundTransferAmountScreenState extends State<FundTransferAmountScreen> {
|
|||||||
SizedBox(
|
SizedBox(
|
||||||
width: double.infinity,
|
width: double.infinity,
|
||||||
child: ElevatedButton(
|
child: ElevatedButton(
|
||||||
onPressed: _onProceed,
|
onPressed: _isAmountOverLimit ? null : _onProceed,
|
||||||
style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
padding: const EdgeInsets.symmetric(vertical: 16),
|
padding: const EdgeInsets.symmetric(vertical: 16),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -39,9 +39,8 @@ class _FundTransferBeneficiaryScreenState
|
|||||||
setState(() {
|
setState(() {
|
||||||
_beneficiaries = data
|
_beneficiaries = data
|
||||||
.where((b) => widget.isOwnBank
|
.where((b) => widget.isOwnBank
|
||||||
? b.bankName ==
|
? b.bankName!.toLowerCase().contains('kangra central')
|
||||||
'THE KANGRA CENTRAL COOPERATIVE BANK LIMITED'
|
: !b.bankName!.toLowerCase().contains('kangra central'))
|
||||||
: b.bankName != 'THE KANGRA CENTRAL COOPERATIVE BANK LIMITED')
|
|
||||||
.toList();
|
.toList();
|
||||||
_isLoading = false;
|
_isLoading = false;
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -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() {
|
||||||
|
|||||||
@@ -18,7 +18,8 @@ class TpinSetupPromptScreen extends StatefulWidget {
|
|||||||
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) {
|
||||||
@@ -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,7 +118,8 @@ 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(
|
||||||
|
|||||||
@@ -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,7 +59,6 @@ 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)),
|
||||||
|
|||||||
@@ -56,15 +56,14 @@ class _ChangePasswordScreenState extends State<ChangePasswordScreen> {
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
}
|
}
|
||||||
final ChangePasswordService _changePasswordService = getIt<ChangePasswordService>();
|
|
||||||
|
final ChangePasswordService _changePasswordService =
|
||||||
|
getIt<ChangePasswordService>();
|
||||||
void _proceed() async {
|
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,7 +77,9 @@ 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')),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -87,7 +88,8 @@ void _proceed() async {
|
|||||||
@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,
|
||||||
|
|||||||
228
lib/features/profile/daily_transaction_limit.dart
Normal file
228
lib/features/profile/daily_transaction_limit.dart
Normal file
@@ -0,0 +1,228 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:flutter/services.dart';
|
||||||
|
import 'package:kmobile/api/services/limit_service.dart';
|
||||||
|
import 'package:kmobile/di/injection.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
import 'package:intl/intl.dart';
|
||||||
|
|
||||||
|
class DailyLimitScreen extends StatefulWidget {
|
||||||
|
const DailyLimitScreen({super.key});
|
||||||
|
@override
|
||||||
|
State<DailyLimitScreen> createState() => _DailyLimitScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _DailyLimitScreenState extends State<DailyLimitScreen> {
|
||||||
|
double? _currentLimit;
|
||||||
|
double? _spentAmount = 0.0;
|
||||||
|
final _limitController = TextEditingController();
|
||||||
|
var service = getIt<LimitService>();
|
||||||
|
Limit? limit;
|
||||||
|
bool _isLoading = true;
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_loadlimits();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _loadlimits() async {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = true;
|
||||||
|
});
|
||||||
|
final limit_data = await service.getLimit();
|
||||||
|
setState(() {
|
||||||
|
limit = limit_data;
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
void dispose() {
|
||||||
|
_limitController.dispose();
|
||||||
|
super.dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _showAddOrEditLimitDialog() async {
|
||||||
|
_limitController.text = _currentLimit?.toStringAsFixed(0) ?? '';
|
||||||
|
final newLimit = await showDialog<double>(
|
||||||
|
context: context,
|
||||||
|
builder: (dialogContext) {
|
||||||
|
final localizations = AppLocalizations.of(dialogContext);
|
||||||
|
final theme = Theme.of(dialogContext);
|
||||||
|
return AlertDialog(
|
||||||
|
title: Text(
|
||||||
|
_currentLimit == null
|
||||||
|
? localizations.addLimit
|
||||||
|
: localizations.editLimit,
|
||||||
|
),
|
||||||
|
content: TextField(
|
||||||
|
controller: _limitController,
|
||||||
|
autofocus: true,
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
inputFormatters: [
|
||||||
|
FilteringTextInputFormatter.allow(RegExp(r'^\d+')),
|
||||||
|
],
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: localizations.limitAmount,
|
||||||
|
prefixText: '₹',
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
actions: [
|
||||||
|
TextButton(
|
||||||
|
onPressed: () => Navigator.of(dialogContext).pop(),
|
||||||
|
child: Text(localizations.cancel),
|
||||||
|
),
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: () {
|
||||||
|
final value = double.tryParse(_limitController.text);
|
||||||
|
if (value == null || value <= 0) return;
|
||||||
|
|
||||||
|
if (value > 200000) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: const Text("Limit To be Set must be less than 200000"),
|
||||||
|
behavior: SnackBarBehavior.floating,
|
||||||
|
backgroundColor: theme.colorScheme.error,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
service.editLimit(value);
|
||||||
|
Navigator.of(dialogContext).pop(value);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Text(localizations.save),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
|
||||||
|
if (newLimit != null) {
|
||||||
|
_loadlimits();
|
||||||
|
if (!mounted) return;
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
const SnackBar(
|
||||||
|
content: Text("Limit Updated"),
|
||||||
|
behavior: SnackBarBehavior.floating,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
void _removeLimit() {
|
||||||
|
setState(() {
|
||||||
|
_currentLimit = null;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
if (_isLoading) {
|
||||||
|
final localizations = AppLocalizations.of(context);
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(localizations.dailylimit),
|
||||||
|
),
|
||||||
|
body: const Center(
|
||||||
|
child: CircularProgressIndicator(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
_currentLimit = limit?.dailyLimit;
|
||||||
|
_spentAmount = limit?.usedLimit;
|
||||||
|
final localizations = AppLocalizations.of(context);
|
||||||
|
final theme = Theme.of(context);
|
||||||
|
final formatCurrency = NumberFormat.currency(locale: 'en_IN', symbol: '₹');
|
||||||
|
final remainingLimit = _currentLimit != null ? _currentLimit! - _spentAmount! : 0.0;
|
||||||
|
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(localizations.dailylimit),
|
||||||
|
),
|
||||||
|
body: Center(
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
localizations.currentDailyLimit,
|
||||||
|
style: theme.textTheme.headlineSmall?.copyWith(
|
||||||
|
color: theme.colorScheme.onSurface.withOpacity(0.7),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
Text(
|
||||||
|
_currentLimit == null
|
||||||
|
? localizations.noLimitSet
|
||||||
|
: formatCurrency.format(_currentLimit),
|
||||||
|
style: theme.textTheme.headlineMedium?.copyWith(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: _currentLimit == null
|
||||||
|
? theme.colorScheme.secondary
|
||||||
|
: theme.colorScheme.primary,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (_currentLimit != null) ...[
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
Text(
|
||||||
|
"Remaining Limit Today", // This should be localized
|
||||||
|
style: theme.textTheme.titleMedium,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Text(
|
||||||
|
formatCurrency.format(remainingLimit),
|
||||||
|
style: theme.textTheme.headlineSmall?.copyWith(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: remainingLimit > 0
|
||||||
|
? Colors.green
|
||||||
|
: theme.colorScheme.error,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
const SizedBox(height: 48),
|
||||||
|
if (_currentLimit == null)
|
||||||
|
ElevatedButton.icon(
|
||||||
|
onPressed: _showAddOrEditLimitDialog,
|
||||||
|
icon: const Icon(Icons.add_circle_outline),
|
||||||
|
label: Text(localizations.addLimit),
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: 24, vertical: 12),
|
||||||
|
textStyle: theme.textTheme.titleMedium,
|
||||||
|
),
|
||||||
|
)
|
||||||
|
else
|
||||||
|
Column(
|
||||||
|
children: [
|
||||||
|
ElevatedButton.icon(
|
||||||
|
onPressed: _showAddOrEditLimitDialog,
|
||||||
|
icon: const Icon(Icons.edit_outlined),
|
||||||
|
label: Text(localizations.editLimit),
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: 24, vertical: 12),
|
||||||
|
textStyle: theme.textTheme.titleMedium,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
// TextButton.icon(
|
||||||
|
// onPressed: _removeLimit,
|
||||||
|
// icon: const Icon(Icons.remove_circle_outline),
|
||||||
|
// label: Text(localizations.removeLimit),
|
||||||
|
// style: TextButton.styleFrom(
|
||||||
|
// foregroundColor: theme.colorScheme.error,
|
||||||
|
// ),
|
||||||
|
// ),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -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
|
||||||
|
|||||||
@@ -22,6 +22,12 @@ class PreferenceScreen extends StatelessWidget {
|
|||||||
builder: (context, state) {
|
builder: (context, state) {
|
||||||
return ListView(
|
return ListView(
|
||||||
children: [
|
children: [
|
||||||
|
//Set Prefered Username
|
||||||
|
// ListTile(
|
||||||
|
// leading: const Icon(Icons.person),
|
||||||
|
// title: const Text("Set Prefered Username"),
|
||||||
|
// onTap: () {
|
||||||
|
// }),
|
||||||
// Language Selection
|
// Language Selection
|
||||||
ListTile(
|
ListTile(
|
||||||
leading: const Icon(Icons.language),
|
leading: const Icon(Icons.language),
|
||||||
|
|||||||
@@ -1,7 +1,13 @@
|
|||||||
|
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/daily_transaction_limit.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 +22,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 +51,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,21 +156,104 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
ListTile(
|
||||||
|
leading: const Icon(Icons.currency_rupee),
|
||||||
|
title: Text(AppLocalizations.of(context).dailylimit),
|
||||||
|
onTap: () {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const DailyLimitScreen()),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
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,
|
||||||
)),
|
)),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
|
// ListTile(
|
||||||
|
// leading: const Icon(Icons.password),
|
||||||
|
// title: const Text("Manage TPIN"),
|
||||||
|
// onTap: () async {
|
||||||
|
// },
|
||||||
|
// ),
|
||||||
|
// ListTile(
|
||||||
|
// leading: const Icon(Icons.password),
|
||||||
|
// title: const Text("Change Login MPIN"),
|
||||||
|
// onTap: () async {
|
||||||
|
// },
|
||||||
|
// ),
|
||||||
|
ListTile(
|
||||||
|
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),
|
||||||
|
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(
|
ListTile(
|
||||||
leading: const Icon(Icons.logout),
|
leading: const Icon(Icons.logout),
|
||||||
title: Text(AppLocalizations.of(context).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,
|
||||||
@@ -72,7 +265,6 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
// You can add more profile options here later
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -2,7 +2,9 @@ import 'dart:async';
|
|||||||
import 'dart:developer';
|
import 'dart:developer';
|
||||||
import 'package:dio/dio.dart';
|
import 'package:dio/dio.dart';
|
||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
|
import 'package:intl/intl.dart';
|
||||||
import 'package:kmobile/api/services/imps_service.dart';
|
import 'package:kmobile/api/services/imps_service.dart';
|
||||||
|
import 'package:kmobile/api/services/limit_service.dart';
|
||||||
import 'package:kmobile/api/services/neft_service.dart';
|
import 'package:kmobile/api/services/neft_service.dart';
|
||||||
import 'package:kmobile/api/services/rtgs_service.dart';
|
import 'package:kmobile/api/services/rtgs_service.dart';
|
||||||
import 'package:kmobile/data/models/imps_transaction.dart';
|
import 'package:kmobile/data/models/imps_transaction.dart';
|
||||||
@@ -28,7 +30,10 @@ class QuickPayOutsideBankScreen extends StatefulWidget {
|
|||||||
|
|
||||||
class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
||||||
final _formKey = GlobalKey<FormState>();
|
final _formKey = GlobalKey<FormState>();
|
||||||
|
final _limitService = getIt<LimitService>();
|
||||||
|
Limit? _limit;
|
||||||
|
bool _isLoadingLimit = true;
|
||||||
|
final _formatCurrency = NumberFormat.currency(locale: 'en_IN', symbol: '₹');
|
||||||
// Controllers
|
// Controllers
|
||||||
final accountNumberController = TextEditingController();
|
final accountNumberController = TextEditingController();
|
||||||
final confirmAccountNumberController = TextEditingController();
|
final confirmAccountNumberController = TextEditingController();
|
||||||
@@ -41,6 +46,7 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
final remarksController = TextEditingController();
|
final remarksController = TextEditingController();
|
||||||
final _ifscFocusNode = FocusNode();
|
final _ifscFocusNode = FocusNode();
|
||||||
final service = getIt<BeneficiaryService>();
|
final service = getIt<BeneficiaryService>();
|
||||||
|
bool _isAmountOverLimit = false;
|
||||||
|
|
||||||
late String accountType;
|
late String accountType;
|
||||||
bool _isValidating = false;
|
bool _isValidating = false;
|
||||||
@@ -50,6 +56,7 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
|
_loadLimit();
|
||||||
_ifscFocusNode.addListener(() {
|
_ifscFocusNode.addListener(() {
|
||||||
if (!_ifscFocusNode.hasFocus && ifscController.text.trim().length == 11) {
|
if (!_ifscFocusNode.hasFocus && ifscController.text.trim().length == 11) {
|
||||||
_validateIFSC();
|
_validateIFSC();
|
||||||
@@ -60,6 +67,49 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
accountType = 'Savings';
|
accountType = 'Savings';
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
amountController.addListener(_checkAmountLimit);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _loadLimit() async {
|
||||||
|
setState(() {
|
||||||
|
_isLoadingLimit = true;
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
final limitData = await _limitService.getLimit();
|
||||||
|
setState(() {
|
||||||
|
_limit = limitData;
|
||||||
|
_isLoadingLimit = false;
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
// Handle error if needed
|
||||||
|
setState(() {
|
||||||
|
_isLoadingLimit = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add this method to check the amount against the limit
|
||||||
|
void _checkAmountLimit() {
|
||||||
|
if (_limit == null) return;
|
||||||
|
|
||||||
|
final amount = double.tryParse(amountController.text) ?? 0;
|
||||||
|
final remainingLimit = _limit!.dailyLimit - _limit!.usedLimit;
|
||||||
|
final bool isOverLimit = amount > remainingLimit;
|
||||||
|
|
||||||
|
if (isOverLimit) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text('Amount exceeds remaining daily limit of ${_formatCurrency.format(remainingLimit)}'),
|
||||||
|
backgroundColor: Colors.red,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (_isAmountOverLimit != isOverLimit) {
|
||||||
|
setState(() {
|
||||||
|
_isAmountOverLimit = isOverLimit;
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void _validateIFSC() async {
|
void _validateIFSC() async {
|
||||||
@@ -84,7 +134,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 +688,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 ==
|
||||||
@@ -716,6 +768,9 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 25),
|
const SizedBox(height: 25),
|
||||||
|
Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
Row(
|
Row(
|
||||||
children: [
|
children: [
|
||||||
Expanded(
|
Expanded(
|
||||||
@@ -780,6 +835,22 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
if (_isLoadingLimit)
|
||||||
|
const Padding(
|
||||||
|
padding: EdgeInsets.only(left: 8.0),
|
||||||
|
child: Text('Fetching daily limit...'),
|
||||||
|
),
|
||||||
|
if (!_isLoadingLimit && _limit != null)
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(left: 8.0),
|
||||||
|
child: Text(
|
||||||
|
'Remaining Daily Limit: ${_formatCurrency.format(_limit!.dailyLimit - _limit!.usedLimit)}',
|
||||||
|
style: Theme.of(context).textTheme.bodySmall,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 30),
|
const SizedBox(height: 30),
|
||||||
Row(
|
Row(
|
||||||
@@ -797,13 +868,20 @@ class _QuickPayOutsideBankScreen extends State<QuickPayOutsideBankScreen> {
|
|||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
child: SwipeButton.expand(
|
child: SwipeButton.expand(
|
||||||
thumb: Icon(Icons.arrow_forward,
|
thumb: Icon(Icons.arrow_forward,
|
||||||
color: Theme.of(context).dialogBackgroundColor),
|
color: _isAmountOverLimit ? Colors.grey : Theme.of(context).dialogBackgroundColor),
|
||||||
activeThumbColor: Theme.of(context).colorScheme.primary,
|
activeThumbColor: _isAmountOverLimit ? Colors.grey.shade700 :
|
||||||
activeTrackColor:
|
Theme.of(context).colorScheme.primary,
|
||||||
Theme.of(context).colorScheme.secondary.withAlpha(100),
|
activeTrackColor: _isAmountOverLimit
|
||||||
|
? Colors.grey.shade300
|
||||||
|
: Theme.of(context).colorScheme.secondary.withAlpha(100),
|
||||||
borderRadius: BorderRadius.circular(30),
|
borderRadius: BorderRadius.circular(30),
|
||||||
height: 56,
|
height: 56,
|
||||||
onSwipe: _onProceedToPay,
|
onSwipe: () {
|
||||||
|
if (_isAmountOverLimit) {
|
||||||
|
return; // Do nothing if amount is over the limit
|
||||||
|
}
|
||||||
|
_onProceedToPay();
|
||||||
|
},
|
||||||
child: Text(
|
child: Text(
|
||||||
AppLocalizations.of(context).swipeToPay,
|
AppLocalizations.of(context).swipeToPay,
|
||||||
style: const TextStyle(
|
style: const TextStyle(
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
import 'dart:async';
|
import 'dart:async';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_swipe_button/flutter_swipe_button.dart';
|
import 'package:flutter_swipe_button/flutter_swipe_button.dart';
|
||||||
|
import 'package:intl/intl.dart';
|
||||||
import 'package:kmobile/api/services/beneficiary_service.dart';
|
import 'package:kmobile/api/services/beneficiary_service.dart';
|
||||||
|
import 'package:kmobile/api/services/limit_service.dart';
|
||||||
import 'package:kmobile/api/services/payment_service.dart';
|
import 'package:kmobile/api/services/payment_service.dart';
|
||||||
import 'package:kmobile/data/models/transfer.dart';
|
import 'package:kmobile/data/models/transfer.dart';
|
||||||
import 'package:kmobile/di/injection.dart';
|
import 'package:kmobile/di/injection.dart';
|
||||||
@@ -19,14 +21,17 @@ class QuickPayWithinBankScreen extends StatefulWidget {
|
|||||||
|
|
||||||
class _QuickPayWithinBankScreen extends State<QuickPayWithinBankScreen> {
|
class _QuickPayWithinBankScreen extends State<QuickPayWithinBankScreen> {
|
||||||
final _formKey = GlobalKey<FormState>();
|
final _formKey = GlobalKey<FormState>();
|
||||||
|
final _limitService = getIt<LimitService>();
|
||||||
|
Limit? _limit;
|
||||||
|
bool _isLoadingLimit = true;
|
||||||
|
final _formatCurrency = NumberFormat.currency(locale: 'en_IN', symbol: '₹');
|
||||||
final TextEditingController accountNumberController = TextEditingController();
|
final TextEditingController accountNumberController = TextEditingController();
|
||||||
final TextEditingController confirmAccountNumberController =
|
final TextEditingController confirmAccountNumberController =
|
||||||
TextEditingController();
|
TextEditingController();
|
||||||
final TextEditingController amountController = TextEditingController();
|
final TextEditingController amountController = TextEditingController();
|
||||||
final TextEditingController remarksController = TextEditingController();
|
final TextEditingController remarksController = TextEditingController();
|
||||||
String? _selectedAccountType;
|
String? _selectedAccountType;
|
||||||
|
bool _isAmountOverLimit = false;
|
||||||
String? _beneficiaryName;
|
String? _beneficiaryName;
|
||||||
bool _isValidating = false;
|
bool _isValidating = false;
|
||||||
bool _isBeneficiaryValidated = false;
|
bool _isBeneficiaryValidated = false;
|
||||||
@@ -35,8 +40,52 @@ class _QuickPayWithinBankScreen extends State<QuickPayWithinBankScreen> {
|
|||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
|
_loadLimit();
|
||||||
accountNumberController.addListener(_resetBeneficiaryValidation);
|
accountNumberController.addListener(_resetBeneficiaryValidation);
|
||||||
confirmAccountNumberController.addListener(_resetBeneficiaryValidation);
|
confirmAccountNumberController.addListener(_resetBeneficiaryValidation);
|
||||||
|
amountController.addListener(_checkAmountLimit);
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _loadLimit() async {
|
||||||
|
setState(() {
|
||||||
|
_isLoadingLimit = true;
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
final limitData = await _limitService.getLimit();
|
||||||
|
setState(() {
|
||||||
|
_limit = limitData;
|
||||||
|
_isLoadingLimit = false;
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
// Handle error if needed
|
||||||
|
setState(() {
|
||||||
|
_isLoadingLimit = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _checkAmountLimit() {
|
||||||
|
if (_limit == null) return;
|
||||||
|
|
||||||
|
final amount = double.tryParse(amountController.text) ?? 0;
|
||||||
|
final remainingLimit = _limit!.dailyLimit - _limit!.usedLimit;
|
||||||
|
final bool isOverLimit = amount > remainingLimit;
|
||||||
|
|
||||||
|
if (isOverLimit) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text('Amount exceeds remaining daily limit of ${_formatCurrency.format(remainingLimit)}'),
|
||||||
|
backgroundColor: Colors.red,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update state only if it changes to avoid unnecessary rebuilds
|
||||||
|
if (_isAmountOverLimit != isOverLimit) {
|
||||||
|
setState(() {
|
||||||
|
_isAmountOverLimit = isOverLimit;
|
||||||
|
});
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void _resetBeneficiaryValidation() {
|
void _resetBeneficiaryValidation() {
|
||||||
@@ -53,6 +102,7 @@ class _QuickPayWithinBankScreen extends State<QuickPayWithinBankScreen> {
|
|||||||
|
|
||||||
@override
|
@override
|
||||||
void dispose() {
|
void dispose() {
|
||||||
|
amountController.removeListener(_checkAmountLimit);
|
||||||
accountNumberController.removeListener(_resetBeneficiaryValidation);
|
accountNumberController.removeListener(_resetBeneficiaryValidation);
|
||||||
confirmAccountNumberController.removeListener(_resetBeneficiaryValidation);
|
confirmAccountNumberController.removeListener(_resetBeneficiaryValidation);
|
||||||
accountNumberController.dispose();
|
accountNumberController.dispose();
|
||||||
@@ -102,6 +152,7 @@ class _QuickPayWithinBankScreen extends State<QuickPayWithinBankScreen> {
|
|||||||
padding: const EdgeInsets.all(16.0),
|
padding: const EdgeInsets.all(16.0),
|
||||||
child: Form(
|
child: Form(
|
||||||
key: _formKey,
|
key: _formKey,
|
||||||
|
child: SingleChildScrollView(
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
const SizedBox(height: 10),
|
const SizedBox(height: 10),
|
||||||
@@ -297,6 +348,7 @@ TextFormField(
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 25),
|
const SizedBox(height: 25),
|
||||||
|
|
||||||
TextFormField(
|
TextFormField(
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: AppLocalizations.of(context).amount,
|
labelText: AppLocalizations.of(context).amount,
|
||||||
@@ -326,15 +378,26 @@ TextFormField(
|
|||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
if (_isLoadingLimit)
|
||||||
|
const Text('Fetching daily limit...'),
|
||||||
|
if (!_isLoadingLimit && _limit != null)
|
||||||
|
Text(
|
||||||
|
'Remaining Daily Limit: ${_formatCurrency.format(_limit!.dailyLimit - _limit!.usedLimit)}',
|
||||||
|
style: Theme.of(context).textTheme.bodySmall,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 45),
|
const SizedBox(height: 45),
|
||||||
Align(
|
Align(
|
||||||
alignment: Alignment.center,
|
alignment: Alignment.center,
|
||||||
child: SwipeButton.expand(
|
child: SwipeButton.expand(
|
||||||
thumb: Icon(Icons.arrow_forward,
|
thumb: Icon(Icons.arrow_forward,
|
||||||
color: Theme.of(context).dialogBackgroundColor),
|
color: _isAmountOverLimit ? Colors.grey : Theme.of(context).dialogBackgroundColor),
|
||||||
activeThumbColor: Theme.of(context).colorScheme.primary,
|
activeThumbColor: _isAmountOverLimit ? Colors.grey.shade700 :
|
||||||
activeTrackColor: Theme.of(
|
Theme.of(context).colorScheme.primary,
|
||||||
|
activeTrackColor: _isAmountOverLimit
|
||||||
|
? Colors.grey.shade300
|
||||||
|
: Theme.of(
|
||||||
context,
|
context,
|
||||||
).colorScheme.secondary.withAlpha(100),
|
).colorScheme.secondary.withAlpha(100),
|
||||||
borderRadius: BorderRadius.circular(30),
|
borderRadius: BorderRadius.circular(30),
|
||||||
@@ -344,6 +407,9 @@ TextFormField(
|
|||||||
style: const TextStyle(fontSize: 16),
|
style: const TextStyle(fontSize: 16),
|
||||||
),
|
),
|
||||||
onSwipe: () {
|
onSwipe: () {
|
||||||
|
if (_isAmountOverLimit) {
|
||||||
|
return; // Do nothing if amount is over limit
|
||||||
|
}
|
||||||
if (_formKey.currentState!.validate()) {
|
if (_formKey.currentState!.validate()) {
|
||||||
if (!_isBeneficiaryValidated) {
|
if (!_isBeneficiaryValidated) {
|
||||||
setState(() {
|
setState(() {
|
||||||
@@ -389,6 +455,7 @@ TextFormField(
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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,17 +1,24 @@
|
|||||||
import '../../../l10n/app_localizations.dart';
|
// ignore_for_file: unused_element
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
class Branch {
|
// Enum to define the type of location
|
||||||
|
enum LocationType { branch, atm }
|
||||||
|
|
||||||
|
class Location {
|
||||||
final String name;
|
final String name;
|
||||||
final String code;
|
final String? code; // Nullable for ATMs
|
||||||
final String ifsc;
|
final String? ifsc; // Nullable for ATMs
|
||||||
final String address;
|
final String address;
|
||||||
|
final LocationType type;
|
||||||
|
|
||||||
Branch({
|
Location({
|
||||||
required this.name,
|
required this.name,
|
||||||
required this.code,
|
this.code,
|
||||||
required this.ifsc,
|
this.ifsc,
|
||||||
required this.address,
|
required this.address,
|
||||||
|
required this.type,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -25,39 +32,75 @@ class BranchLocatorScreen extends StatefulWidget {
|
|||||||
class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
||||||
final TextEditingController _searchController = TextEditingController();
|
final TextEditingController _searchController = TextEditingController();
|
||||||
|
|
||||||
// Static list of 2 branches
|
final List<Location> _allLocations = [
|
||||||
final List<Branch> _branches = [
|
Location(
|
||||||
Branch(
|
|
||||||
name: "Dharamsala - Head Office",
|
name: "Dharamsala - Head Office",
|
||||||
code: "002",
|
code: "002",
|
||||||
ifsc: "KACE0000002",
|
ifsc: "KACE0000002",
|
||||||
address: "Civil Lines Dharmashala, Kangra, HP - 176215"),
|
address: "Civil Lines Dharmashala, Kangra, HP - 176215",
|
||||||
Branch(
|
type: LocationType.branch,
|
||||||
|
),
|
||||||
|
Location(
|
||||||
name: "Kangra",
|
name: "Kangra",
|
||||||
code: "033",
|
code: "033",
|
||||||
ifsc: "KACE0000033",
|
ifsc: "KACE0000033",
|
||||||
address: "Rajput Bhawankangrapo, Kangra, HP "),
|
address: "Rajput Bhawankangrapo, Kangra, HP ",
|
||||||
|
type: LocationType.branch,
|
||||||
|
),
|
||||||
|
Location(
|
||||||
|
name: "Dharamsala ATM",
|
||||||
|
address: "Near Main Square, Dharamsala",
|
||||||
|
type: LocationType.atm,
|
||||||
|
),
|
||||||
|
Location(
|
||||||
|
name: "Kangra ATM",
|
||||||
|
address: "Opposite Bus Stand, Kangra",
|
||||||
|
type: LocationType.atm,
|
||||||
|
),
|
||||||
];
|
];
|
||||||
|
|
||||||
List<Branch> _filteredBranches = [];
|
List<Location> _filteredLocations = [];
|
||||||
|
bool _isLoading = false;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
void initState() {
|
void initState() {
|
||||||
super.initState();
|
super.initState();
|
||||||
_filteredBranches = _branches; // Initially show all branches
|
// _fetchAndSetLocations();
|
||||||
|
_filteredLocations = _allLocations;
|
||||||
}
|
}
|
||||||
|
|
||||||
void _filterBranches(String query) {
|
// Example of a future API fetching function
|
||||||
|
/*
|
||||||
|
Future<void> _fetchAndSetLocations() async {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = true;
|
||||||
|
});
|
||||||
|
try {
|
||||||
|
// final locations = await yourApiService.getLocations();
|
||||||
|
// setState(() {
|
||||||
|
// _allLocations = locations;
|
||||||
|
// _filteredLocations = locations;
|
||||||
|
// });
|
||||||
|
} catch (e) {
|
||||||
|
// Handle error
|
||||||
|
} finally {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
void _filterLocations(String query) {
|
||||||
setState(() {
|
setState(() {
|
||||||
if (query.isEmpty) {
|
if (query.isEmpty) {
|
||||||
_filteredBranches = _branches;
|
_filteredLocations = _allLocations;
|
||||||
} else {
|
} else {
|
||||||
_filteredBranches = _branches.where((branch) {
|
_filteredLocations = _allLocations.where((location) {
|
||||||
final lowerQuery = query.toLowerCase();
|
final lowerQuery = query.toLowerCase();
|
||||||
return branch.name.toLowerCase().contains(lowerQuery) ||
|
return location.name.toLowerCase().contains(lowerQuery) ||
|
||||||
branch.code.toLowerCase().contains(lowerQuery) ||
|
(location.code?.toLowerCase().contains(lowerQuery) ?? false) ||
|
||||||
branch.ifsc.toLowerCase().contains(lowerQuery) ||
|
(location.ifsc?.toLowerCase().contains(lowerQuery) ?? false) ||
|
||||||
branch.address.toLowerCase().contains(lowerQuery);
|
location.address.toLowerCase().contains(lowerQuery);
|
||||||
}).toList();
|
}).toList();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
@@ -71,12 +114,11 @@ class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
|||||||
),
|
),
|
||||||
body: Column(
|
body: Column(
|
||||||
children: [
|
children: [
|
||||||
// Search bar
|
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.all(12.0),
|
padding: const EdgeInsets.all(12.0),
|
||||||
child: TextField(
|
child: TextField(
|
||||||
controller: _searchController,
|
controller: _searchController,
|
||||||
onChanged: _filterBranches,
|
onChanged: _filterLocations,
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
hintText: AppLocalizations.of(context).searchbranchby,
|
hintText: AppLocalizations.of(context).searchbranchby,
|
||||||
prefixIcon: const Icon(Icons.search),
|
prefixIcon: const Icon(Icons.search),
|
||||||
@@ -87,33 +129,17 @@ class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|
||||||
// List of branches
|
// Content area
|
||||||
Expanded(
|
Expanded(
|
||||||
child: _filteredBranches.isEmpty
|
child: _isLoading
|
||||||
? const Center(child: Text("No matching branches found"))
|
? const Center(child: CircularProgressIndicator())
|
||||||
|
: _filteredLocations.isEmpty
|
||||||
|
? const Center(child: Text("No matching locations found"))
|
||||||
: ListView.builder(
|
: ListView.builder(
|
||||||
itemCount: _filteredBranches.length,
|
itemCount: _filteredLocations.length,
|
||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
final branch = _filteredBranches[index];
|
final location = _filteredLocations[index];
|
||||||
return Card(
|
return _buildLocationItem(location);
|
||||||
margin: const EdgeInsets.symmetric(
|
|
||||||
horizontal: 12, vertical: 6),
|
|
||||||
child: ListTile(
|
|
||||||
leading: Icon(Icons.location_city,
|
|
||||||
color: Theme.of(context).colorScheme.primary),
|
|
||||||
title: Text(branch.name,
|
|
||||||
style:
|
|
||||||
const TextStyle(fontWeight: FontWeight.bold)),
|
|
||||||
subtitle: Text(
|
|
||||||
"Code: ${branch.code} | IFSC: ${branch.ifsc} \nBranch Address: ${branch.address}"),
|
|
||||||
onTap: () {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(
|
|
||||||
content: Text("Selected ${branch.name}")),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -121,4 +147,42 @@ class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
|||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Widget _buildHeader(String title) {
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
|
||||||
|
child: Text(
|
||||||
|
title,
|
||||||
|
style: Theme.of(context).textTheme.titleLarge?.copyWith(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Helper widget to build a single location item
|
||||||
|
Widget _buildLocationItem(Location location) {
|
||||||
|
final isBranch = location.type == LocationType.branch;
|
||||||
|
return Card(
|
||||||
|
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
||||||
|
child: ListTile(
|
||||||
|
leading: CircleAvatar(
|
||||||
|
child: Icon(isBranch ? Icons.location_city : Icons.currency_rupee),
|
||||||
|
),
|
||||||
|
title: Text(location.name,
|
||||||
|
style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
subtitle: Text(
|
||||||
|
isBranch
|
||||||
|
? "Code: ${location.code} | IFSC: ${location.ifsc}\nAddress: ${location.address}"
|
||||||
|
: "Address: ${location.address}",
|
||||||
|
),
|
||||||
|
onTap: () {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(content: Text("Selected ${location.name}")),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
47
lib/features/service/screens/faqs_screen.dart
Normal file
47
lib/features/service/screens/faqs_screen.dart
Normal file
@@ -0,0 +1,47 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
|
||||||
|
class FaqsScreen extends StatefulWidget {
|
||||||
|
const FaqsScreen({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<FaqsScreen> createState() => _FaqsScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _FaqsScreenState extends State<FaqsScreen> {
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_getFaqs();
|
||||||
|
}
|
||||||
|
|
||||||
|
// A placeholder for your future API call
|
||||||
|
Future<void> _getFaqs() async {
|
||||||
|
// TODO: Implement API call to fetch FAQs data
|
||||||
|
// For now, simulating a network call with a delay
|
||||||
|
await Future.delayed(const Duration(seconds: 1));
|
||||||
|
// In a real implementation, you would process the API response here
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Row(
|
||||||
|
children: [
|
||||||
|
Flexible(
|
||||||
|
child: Text(
|
||||||
|
AppLocalizations.of(context).faq,
|
||||||
|
softWrap: true,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontSize: 16.5,
|
||||||
|
),
|
||||||
|
textAlign: TextAlign.left,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
34
lib/features/service/screens/quick_links_screen.dart
Normal file
34
lib/features/service/screens/quick_links_screen.dart
Normal file
@@ -0,0 +1,34 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
|
||||||
|
class QuickLinksScreen extends StatefulWidget {
|
||||||
|
const QuickLinksScreen({super.key});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<QuickLinksScreen> createState() => _QuickLinksScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _QuickLinksScreenState extends State<QuickLinksScreen> {
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_getQuickLinks();
|
||||||
|
}
|
||||||
|
|
||||||
|
// A placeholder for your future API call
|
||||||
|
Future<void> _getQuickLinks() async {
|
||||||
|
// TODO: Implement API call to fetch quick links data
|
||||||
|
// For now, simulating a network call with a delay
|
||||||
|
await Future.delayed(const Duration(seconds: 1));
|
||||||
|
// In a real implementation, you would process the API response here
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(AppLocalizations.of(context).quickLinks),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,9 +1,10 @@
|
|||||||
import 'package:kmobile/features/service/screens/branch_locator_screen.dart';
|
import 'package:kmobile/features/service/screens/branch_locator_screen.dart';
|
||||||
|
|
||||||
import '../../../l10n/app_localizations.dart';
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
import 'package:flutter/material.dart';
|
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/faqs_screen.dart';
|
||||||
|
|
||||||
class ServiceScreen extends StatefulWidget {
|
class ServiceScreen extends StatefulWidget {
|
||||||
const ServiceScreen({super.key});
|
const ServiceScreen({super.key});
|
||||||
@@ -29,25 +30,41 @@ Widget build(BuildContext context) {
|
|||||||
icon: Symbols.add,
|
icon: Symbols.add,
|
||||||
label: AppLocalizations.of(context).accountOpeningDeposit,
|
label: AppLocalizations.of(context).accountOpeningDeposit,
|
||||||
onTap: () {},
|
onTap: () {},
|
||||||
disabled: true, // Add this
|
disabled: true,
|
||||||
),
|
),
|
||||||
const Divider(height: 1),
|
const Divider(height: 1),
|
||||||
ServiceManagementTile(
|
ServiceManagementTile(
|
||||||
icon: Symbols.add,
|
icon: Symbols.add,
|
||||||
label: AppLocalizations.of(context).accountOpeningLoan,
|
label: AppLocalizations.of(context).accountOpeningLoan,
|
||||||
onTap: () {},
|
onTap: () {},
|
||||||
disabled: true, // Add this
|
disabled: true,
|
||||||
),
|
),
|
||||||
const Divider(height: 1),
|
const Divider(height: 1),
|
||||||
ServiceManagementTile(
|
ServiceManagementTile(
|
||||||
icon: Symbols.captive_portal,
|
icon: Symbols.captive_portal,
|
||||||
label: AppLocalizations.of(context).quickLinks,
|
label: AppLocalizations.of(context).quickLinks,
|
||||||
onTap: () {},
|
onTap: () {
|
||||||
disabled: true, // Add this
|
Navigator.of(context).push(
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const QuickLinksScreen()),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
disabled: true,
|
||||||
),
|
),
|
||||||
const Divider(height: 1),
|
const Divider(height: 1),
|
||||||
ServiceManagementTile(
|
ServiceManagementTile(
|
||||||
icon: Symbols.missing_controller,
|
icon: Symbols.question_mark,
|
||||||
|
label: AppLocalizations.of(context).faq,
|
||||||
|
onTap: () {
|
||||||
|
Navigator.of(context).push(
|
||||||
|
MaterialPageRoute(builder: (context) => const FaqsScreen()),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
disabled: true,
|
||||||
|
),
|
||||||
|
const Divider(height: 1),
|
||||||
|
ServiceManagementTile(
|
||||||
|
icon: Symbols.location_pin,
|
||||||
label: AppLocalizations.of(context).branchLocator,
|
label: AppLocalizations.of(context).branchLocator,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
@@ -55,7 +72,7 @@ Widget build(BuildContext context) {
|
|||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
builder: (context) => const BranchLocatorScreen()));
|
builder: (context) => const BranchLocatorScreen()));
|
||||||
},
|
},
|
||||||
disabled: true, // Add this
|
disabled: true,
|
||||||
),
|
),
|
||||||
const Divider(height: 1),
|
const Divider(height: 1),
|
||||||
],
|
],
|
||||||
@@ -68,14 +85,14 @@ class ServiceManagementTile extends StatelessWidget {
|
|||||||
final IconData icon;
|
final IconData icon;
|
||||||
final String label;
|
final String label;
|
||||||
final VoidCallback onTap;
|
final VoidCallback onTap;
|
||||||
final bool disabled; // Add this line
|
final bool disabled;
|
||||||
|
|
||||||
const ServiceManagementTile({
|
const ServiceManagementTile({
|
||||||
super.key,
|
super.key,
|
||||||
required this.icon,
|
required this.icon,
|
||||||
required this.label,
|
required this.label,
|
||||||
required this.onTap,
|
required this.onTap,
|
||||||
this.disabled = false, // Add this line
|
this.disabled = false,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
@@ -84,20 +101,20 @@ class ServiceManagementTile extends StatelessWidget {
|
|||||||
return ListTile(
|
return ListTile(
|
||||||
leading: Icon(
|
leading: Icon(
|
||||||
icon,
|
icon,
|
||||||
color: disabled ? theme.disabledColor : null, // Change color when disabled
|
color: disabled ? theme.disabledColor : null,
|
||||||
),
|
),
|
||||||
title: Text(
|
title: Text(
|
||||||
label,
|
label,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
color: disabled ? theme.disabledColor : null, // Change color when disabled
|
color: disabled ? theme.disabledColor : null,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
trailing: Icon(
|
trailing: Icon(
|
||||||
Symbols.arrow_right,
|
Symbols.arrow_right,
|
||||||
size: 20,
|
size: 20,
|
||||||
color: disabled ? theme.disabledColor : null, // Change color when disabled
|
color: disabled ? theme.disabledColor : null,
|
||||||
),
|
),
|
||||||
onTap: disabled ? null : onTap, // Disable onTap when disabled
|
onTap: disabled ? null : onTap,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -98,7 +98,7 @@
|
|||||||
"accountOpeningDeposit": "Account Opening Request - Deposit",
|
"accountOpeningDeposit": "Account Opening Request - Deposit",
|
||||||
"accountOpeningLoan": "Account Opening Request - Loan",
|
"accountOpeningLoan": "Account Opening Request - Loan",
|
||||||
"quickLinks": "Quick Links",
|
"quickLinks": "Quick Links",
|
||||||
"branchLocator": "Branch Locator",
|
"branchLocator": "Branch and ATM Locator",
|
||||||
"emailLaunchError": "Could not launch email client for",
|
"emailLaunchError": "Could not launch email client for",
|
||||||
"dialerLaunchError": "Could not launch dialer for",
|
"dialerLaunchError": "Could not launch dialer for",
|
||||||
"writeToUs": "Write to us",
|
"writeToUs": "Write to us",
|
||||||
@@ -248,7 +248,7 @@
|
|||||||
"validateBeneficiaryproceeding": "Please validate beneficiary before proceeding",
|
"validateBeneficiaryproceeding": "Please validate beneficiary before proceeding",
|
||||||
"findnearbybranched": "Find nearby branches",
|
"findnearbybranched": "Find nearby branches",
|
||||||
"searchbranch": "Search Branch",
|
"searchbranch": "Search Branch",
|
||||||
"searchbranchby": "Search by Branch Name / Code / IFSC",
|
"searchbranchby": "Name / Code / IFSC / Address",
|
||||||
"branchsearchsoon": "Branch search coming soon...",
|
"branchsearchsoon": "Branch search coming soon...",
|
||||||
"loginFailed": "Login failed",
|
"loginFailed": "Login failed",
|
||||||
"invalidCredentials": "Invalid credentials",
|
"invalidCredentials": "Invalid credentials",
|
||||||
@@ -315,5 +315,21 @@
|
|||||||
"themeModeDark": "Dark",
|
"themeModeDark": "Dark",
|
||||||
"details": "Details",
|
"details": "Details",
|
||||||
"remarks": "Remarks (Optional)",
|
"remarks": "Remarks (Optional)",
|
||||||
"kccbMobile": "KCCB Mobile"
|
"kccbMobile": "KCCB Mobile",
|
||||||
|
"faq": "Frequently Asked Questions(FAQs)",
|
||||||
|
"branches": "Branches",
|
||||||
|
"atms": "ATMs",
|
||||||
|
"dailylimit": "Daily Transaction Limit",
|
||||||
|
"currentDailyLimit": "Current Daily Limit",
|
||||||
|
"noLimitSet": "No Limit Set",
|
||||||
|
"addLimit": "Add Limit",
|
||||||
|
"editLimit": "Edit Limit",
|
||||||
|
"removeLimit": "Remove Limit",
|
||||||
|
"limitAmount": "Limit Amount",
|
||||||
|
"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?"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -98,7 +98,7 @@
|
|||||||
"accountOpeningDeposit": "खाता खोलने का अनुरोध - जमा",
|
"accountOpeningDeposit": "खाता खोलने का अनुरोध - जमा",
|
||||||
"accountOpeningLoan": "खाता खोलने का अनुरोध - ऋण",
|
"accountOpeningLoan": "खाता खोलने का अनुरोध - ऋण",
|
||||||
"quickLinks": "त्वरित लिंक",
|
"quickLinks": "त्वरित लिंक",
|
||||||
"branchLocator": "शाखा लोकेटर",
|
"branchLocator": "शाखा और एटीएम लोकेटर",
|
||||||
"emailLaunchError": "ईमेल क्लाइंट खोलने में विफल: ",
|
"emailLaunchError": "ईमेल क्लाइंट खोलने में विफल: ",
|
||||||
"dialerLaunchError": "डायलर खोलने में विफल: ",
|
"dialerLaunchError": "डायलर खोलने में विफल: ",
|
||||||
"writeToUs": "हमें लिखें",
|
"writeToUs": "हमें लिखें",
|
||||||
@@ -249,7 +249,7 @@
|
|||||||
"validateBeneficiaryproceeding": "कृपया आगे बढ़ने से पहले लाभार्थी को पट्टे पर मान्य करें",
|
"validateBeneficiaryproceeding": "कृपया आगे बढ़ने से पहले लाभार्थी को पट्टे पर मान्य करें",
|
||||||
"findnearbybranched": "आस-पास की शाखाएँ खोजें",
|
"findnearbybranched": "आस-पास की शाखाएँ खोजें",
|
||||||
"searchbranch": "शाखा खोजें",
|
"searchbranch": "शाखा खोजें",
|
||||||
"searchbranchby": "शाखा खोजें नाम / बैंक कोड / आईएफएससी द्वारा",
|
"searchbranchby": "शाखा नाम / बैंक कोड / आईएफएससी / पता",
|
||||||
"branchsearchsoon": "शाखा खोज सुविधा जल्द ही आ रही है...",
|
"branchsearchsoon": "शाखा खोज सुविधा जल्द ही आ रही है...",
|
||||||
"loginFailed": "लॉगिन विफल",
|
"loginFailed": "लॉगिन विफल",
|
||||||
"invalidCredentials": "अवैध प्रत्यय पत्र",
|
"invalidCredentials": "अवैध प्रत्यय पत्र",
|
||||||
@@ -316,5 +316,21 @@
|
|||||||
"themeModeDark": "डार्क",
|
"themeModeDark": "डार्क",
|
||||||
"details": "विवरण",
|
"details": "विवरण",
|
||||||
"remarks": "विचार (अनिवार्य नहीं)",
|
"remarks": "विचार (अनिवार्य नहीं)",
|
||||||
"kccbMobile": "केसीसीबी मोबाइल"
|
"kccbMobile": "केसीसीबी मोबाइल",
|
||||||
|
"faq": "अक्सर पूछे जाने वाले प्रश्न",
|
||||||
|
"branches": "शाखाओं",
|
||||||
|
"atms": "एटीएम",
|
||||||
|
"dailylimit": "दैनिक लेनदेन सीमा",
|
||||||
|
"currentDailyLimit": "वर्तमान दैनिक सीमा",
|
||||||
|
"noLimitSet": "कोई सीमा निर्धारित नहीं",
|
||||||
|
"addLimit": "सीमा जोड़ें",
|
||||||
|
"editLimit": "सीमा संपादित करें",
|
||||||
|
"removeLimit": "सीमा हटाएँ",
|
||||||
|
"limitAmount": "सीमा राशि",
|
||||||
|
"save": "जमा करें",
|
||||||
|
"deregister": "अपंजीकृत",
|
||||||
|
"deregistercheck": "क्या आप वाकई पंजीकरण रद्द करना चाहते हैं??",
|
||||||
|
"biometricsNotAvailable": "इस डिवाइस पर बायोमेट्रिक्स उपलब्ध नहीं है",
|
||||||
|
"disableFingerprintLogin": "फ़िंगरप्रिंट लॉगिन अक्षम करें",
|
||||||
|
"disableFingerprintMessage": "क्या आप फ़िंगरप्रिंट लॉगिन अक्षम करना चाहते हैं?"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,4 @@
|
|||||||
// 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/features/security/security_error_screen.dart';
|
import 'package:kmobile/features/security/security_error_screen.dart';
|
||||||
@@ -10,22 +9,19 @@ import 'app.dart';
|
|||||||
void main() async {
|
void main() async {
|
||||||
WidgetsFlutterBinding.ensureInitialized();
|
WidgetsFlutterBinding.ensureInitialized();
|
||||||
|
|
||||||
// Set preferred orientations
|
|
||||||
await SystemChrome.setPreferredOrientations([
|
await SystemChrome.setPreferredOrientations([
|
||||||
DeviceOrientation.portraitUp,
|
DeviceOrientation.portraitUp,
|
||||||
DeviceOrientation.portraitDown,
|
DeviceOrientation.portraitDown,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
// Check for device compromise
|
// Check for device compromise
|
||||||
final compromisedMessage = await SecurityService.deviceCompromisedMessage;
|
// final compromisedMessage = await SecurityService.deviceCompromisedMessage;
|
||||||
if (compromisedMessage != null) {
|
// if (compromisedMessage != null) {
|
||||||
runApp(MaterialApp(
|
// runApp(MaterialApp(
|
||||||
home: SecurityErrorScreen(message: compromisedMessage),
|
// home: SecurityErrorScreen(message: compromisedMessage),
|
||||||
));
|
// ));
|
||||||
return;
|
// return;
|
||||||
}
|
// }
|
||||||
|
|
||||||
// Initialize dependencies
|
|
||||||
await setupDependencies();
|
await setupDependencies();
|
||||||
runApp(const KMobile());
|
runApp(const KMobile());
|
||||||
}
|
}
|
||||||
@@ -22,7 +22,7 @@ Widget getBankLogo(String? bankName, BuildContext context) {
|
|||||||
height: 40,
|
height: 40,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
if (bankName != null && bankName.toLowerCase().contains('icici bank ltd')) {
|
if (bankName != null && bankName.toLowerCase().contains('icici')) {
|
||||||
return Image.asset(
|
return Image.asset(
|
||||||
'assets/images/icici_logo.png',
|
'assets/images/icici_logo.png',
|
||||||
width: 40,
|
width: 40,
|
||||||
@@ -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"))
|
||||||
|
|||||||
24
pubspec.lock
24
pubspec.lock
@@ -541,6 +541,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 +805,14 @@ 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"
|
||||||
sky_engine:
|
sky_engine:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description: flutter
|
description: flutter
|
||||||
|
|||||||
@@ -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,8 @@ 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
|
||||||
# jailbreak_root_detection: "^1.1.6"
|
# jailbreak_root_detection: "^1.1.6"
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
Reference in New Issue
Block a user