Compare commits
23 Commits
b19bc2e222
...
f024f200a4
| Author | SHA1 | Date | |
|---|---|---|---|
| f024f200a4 | |||
| 54a7b8f1b0 | |||
| 60270a5fa9 | |||
| 9d8c8dc8bd | |||
| 537a4faa62 | |||
| 149d4dbc83 | |||
| 3fa40f133a | |||
| 07d5ea8fbe | |||
| 72a2c56392 | |||
| aef82237ac | |||
| 974f42bf95 | |||
| 4a8c69bb1e | |||
| 86aaaa1f6d | |||
| 6796793aac | |||
| fbf6df7181 | |||
| c7111d518a | |||
| 5d307607fd | |||
| 992092052a | |||
| 64fedabd89 | |||
| 4fc6f54fcd | |||
| 8c7e94759a | |||
| 8aa5b170ca | |||
| 04a1ce26ec |
@@ -34,6 +34,7 @@ android {
|
|||||||
ndkVersion "27.0.12077973"
|
ndkVersion "27.0.12077973"
|
||||||
|
|
||||||
compileOptions {
|
compileOptions {
|
||||||
|
coreLibraryDesugaringEnabled true
|
||||||
sourceCompatibility JavaVersion.VERSION_1_8
|
sourceCompatibility JavaVersion.VERSION_1_8
|
||||||
targetCompatibility JavaVersion.VERSION_1_8
|
targetCompatibility JavaVersion.VERSION_1_8
|
||||||
}
|
}
|
||||||
@@ -80,4 +81,6 @@ flutter {
|
|||||||
source '../..'
|
source '../..'
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {}
|
dependencies {
|
||||||
|
coreLibraryDesugaring 'com.android.tools:desugar_jdk_libs:2.1.5'
|
||||||
|
}
|
||||||
|
|||||||
BIN
android/app/src/main/res/drawable/notification_icon.png
Normal file
BIN
android/app/src/main/res/drawable/notification_icon.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 90 KiB |
BIN
assets/images/profile.png
Normal file
BIN
assets/images/profile.png
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 3.3 KiB |
BIN
assets/images/profile.svg
Normal file
BIN
assets/images/profile.svg
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 4.2 KiB |
@@ -105,7 +105,7 @@ class BranchService {
|
|||||||
if (response.statusCode == 200) {
|
if (response.statusCode == 200) {
|
||||||
return Branch.listFromJson(response.data);
|
return Branch.listFromJson(response.data);
|
||||||
} else {
|
} else {
|
||||||
throw Exception("Failed to fetch beneficiaries");
|
throw Exception("Failed to fetch");
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
return [];
|
return [];
|
||||||
|
|||||||
138
lib/api/services/cheque_service.dart
Normal file
138
lib/api/services/cheque_service.dart
Normal file
@@ -0,0 +1,138 @@
|
|||||||
|
import 'dart:convert';
|
||||||
|
|
||||||
|
import 'package:dio/dio.dart';
|
||||||
|
|
||||||
|
class Cheque {
|
||||||
|
final String? type;
|
||||||
|
final String? InstrType;
|
||||||
|
final String? Date;
|
||||||
|
final String? branchCode;
|
||||||
|
final String? fromCheque;
|
||||||
|
final String? toCheque;
|
||||||
|
final String? Chequescount;
|
||||||
|
final String? ChequeNumber;
|
||||||
|
final String? transactionCode;
|
||||||
|
final int? amount;
|
||||||
|
final String? status;
|
||||||
|
final String? stopIssueDate;
|
||||||
|
final String? StopExpiryDate;
|
||||||
|
|
||||||
|
Cheque({
|
||||||
|
this.type,
|
||||||
|
this.InstrType,
|
||||||
|
this.Date,
|
||||||
|
this.branchCode,
|
||||||
|
this.fromCheque,
|
||||||
|
this.toCheque,
|
||||||
|
this.Chequescount,
|
||||||
|
this.ChequeNumber,
|
||||||
|
this.transactionCode,
|
||||||
|
this.amount,
|
||||||
|
this.status,
|
||||||
|
this.stopIssueDate,
|
||||||
|
this.StopExpiryDate,
|
||||||
|
});
|
||||||
|
|
||||||
|
factory Cheque.fromJson(Map<String, dynamic> json) {
|
||||||
|
return Cheque(
|
||||||
|
type: json['type'] ?? '',
|
||||||
|
InstrType: json['InstrType'] ?? '',
|
||||||
|
Date: json['Date'] ?? '',
|
||||||
|
branchCode: json['branchCode'] ?? '',
|
||||||
|
fromCheque: json['fromCheque'] ?? '',
|
||||||
|
toCheque: json['toCheque'] ?? '',
|
||||||
|
Chequescount: json['Chequescount'] ?? '',
|
||||||
|
ChequeNumber: json['ChequeNumber'] ?? '',
|
||||||
|
transactionCode: json['transactionCode'] ?? '',
|
||||||
|
amount: json['amount'],
|
||||||
|
status: json['status'] ?? '',
|
||||||
|
stopIssueDate: json['stopIssueDate'] ?? '',
|
||||||
|
StopExpiryDate: json['StopExpiryDate'] ?? '',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
static List<Cheque> listFromJson(List<dynamic> jsonList) {
|
||||||
|
final chequeList =
|
||||||
|
jsonList.map((cheque) => Cheque.fromJson(cheque)).toList();
|
||||||
|
return chequeList;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ChequeService {
|
||||||
|
final Dio _dio;
|
||||||
|
ChequeService(this._dio);
|
||||||
|
|
||||||
|
Future<List<Cheque>> ChequeEnquiry({
|
||||||
|
required String accountNumber,
|
||||||
|
required String instrType,
|
||||||
|
}) async {
|
||||||
|
try {
|
||||||
|
final response = await _dio.get(
|
||||||
|
"/api/cheque/enquiry",
|
||||||
|
queryParameters: {
|
||||||
|
'accountNumber': accountNumber,
|
||||||
|
'instrumentType': instrType,
|
||||||
|
},
|
||||||
|
options: Options(
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
},
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (response.statusCode == 200) {
|
||||||
|
if (response.data is Map<String, dynamic> &&
|
||||||
|
response.data.containsKey('records')) {
|
||||||
|
final records = response.data['records'];
|
||||||
|
if (records is List) {
|
||||||
|
return Cheque.listFromJson(records);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
throw Exception(
|
||||||
|
"Unexpected API response format: 'records' list not found or malformed");
|
||||||
|
} else {
|
||||||
|
throw Exception("Failed to fetch");
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
print('Error in ChequeEnquiry: $e');
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future stopCheque({
|
||||||
|
required String accountno,
|
||||||
|
required String stopFromChequeNo,
|
||||||
|
required String instrType,
|
||||||
|
String? stopToChequeNo,
|
||||||
|
String? stopIssueDate,
|
||||||
|
String? stopExpiryDate,
|
||||||
|
String? stopAmount,
|
||||||
|
String? stopComment,
|
||||||
|
String? chequeIssueDate,
|
||||||
|
required String tpin,
|
||||||
|
}) async {
|
||||||
|
final response = await _dio.post(
|
||||||
|
'/api/cheque/stop',
|
||||||
|
options: Options(
|
||||||
|
validateStatus: (int? status) => true,
|
||||||
|
receiveDataWhenStatusError: true,
|
||||||
|
),
|
||||||
|
data: {
|
||||||
|
'accountNumber': accountno,
|
||||||
|
'stopFromChequeNo': stopFromChequeNo,
|
||||||
|
'instrumentType': instrType,
|
||||||
|
'stopToChequeNo': stopToChequeNo,
|
||||||
|
'stopIssueDate': stopIssueDate,
|
||||||
|
'stopExpiryDate': stopExpiryDate,
|
||||||
|
'stopAmount': stopAmount,
|
||||||
|
'stopComment': stopComment,
|
||||||
|
'chqIssueDate': chequeIssueDate,
|
||||||
|
'tpin': tpin,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
if (response.statusCode != 200) {
|
||||||
|
throw Exception(jsonEncode(response.data));
|
||||||
|
}
|
||||||
|
return response.toString();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -54,4 +54,34 @@ class LimitService {
|
|||||||
throw Exception('Unexpected error: ${e.toString()}');
|
throw Exception('Unexpected error: ${e.toString()}');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Future getOtpTLimit({
|
||||||
|
required String mobileNumber,
|
||||||
|
}) async {
|
||||||
|
final response = await _dio.post(
|
||||||
|
'/api/otp/send',
|
||||||
|
data: {'mobileNumber': mobileNumber, 'type': "TLIMIT"},
|
||||||
|
);
|
||||||
|
if (response.statusCode != 200) {
|
||||||
|
throw Exception("Invalid Mobile Number/Type");
|
||||||
|
}
|
||||||
|
print(response.toString());
|
||||||
|
return response.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future validateOtp({
|
||||||
|
required String otp,
|
||||||
|
required String mobileNumber,
|
||||||
|
}) async {
|
||||||
|
final response = await _dio.post(
|
||||||
|
'/api/otp/verify?mobileNumber=$mobileNumber',
|
||||||
|
data: {
|
||||||
|
'otp': otp,
|
||||||
|
},
|
||||||
|
);
|
||||||
|
if (response.statusCode != 200) {
|
||||||
|
throw Exception("Wrong OTP");
|
||||||
|
}
|
||||||
|
return response.toString();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,10 +12,8 @@ import 'package:kmobile/features/auth/controllers/theme_state.dart';
|
|||||||
import 'config/routes.dart';
|
import 'config/routes.dart';
|
||||||
import 'di/injection.dart';
|
import 'di/injection.dart';
|
||||||
import 'features/auth/controllers/auth_cubit.dart';
|
import 'features/auth/controllers/auth_cubit.dart';
|
||||||
import 'features/card/screens/card_management_screen.dart';
|
|
||||||
import 'features/accounts/screens/account_statement_screen.dart';
|
import 'features/accounts/screens/account_statement_screen.dart';
|
||||||
import 'package:kmobile/features/auth/controllers/auth_state.dart';
|
import 'package:kmobile/features/auth/controllers/auth_state.dart';
|
||||||
|
|
||||||
import 'features/auth/screens/login_screen.dart';
|
import 'features/auth/screens/login_screen.dart';
|
||||||
import 'features/service/screens/service_screen.dart';
|
import 'features/service/screens/service_screen.dart';
|
||||||
import 'features/dashboard/screens/dashboard_screen.dart';
|
import 'features/dashboard/screens/dashboard_screen.dart';
|
||||||
@@ -330,7 +328,6 @@ class _NavigationScaffoldState extends State<NavigationScaffold> {
|
|||||||
return const Center(child: CircularProgressIndicator());
|
return const Center(child: CircularProgressIndicator());
|
||||||
},
|
},
|
||||||
),
|
),
|
||||||
const CardManagementScreen(),
|
|
||||||
const ServiceScreen(),
|
const ServiceScreen(),
|
||||||
];
|
];
|
||||||
|
|
||||||
@@ -392,10 +389,6 @@ class _NavigationScaffoldState extends State<NavigationScaffold> {
|
|||||||
icon: const Icon(Icons.swap_vert_sharp),
|
icon: const Icon(Icons.swap_vert_sharp),
|
||||||
label: AppLocalizations.of(context).transactions,
|
label: AppLocalizations.of(context).transactions,
|
||||||
),
|
),
|
||||||
BottomNavigationBarItem(
|
|
||||||
icon: const Icon(Icons.credit_card),
|
|
||||||
label: AppLocalizations.of(context).card,
|
|
||||||
),
|
|
||||||
BottomNavigationBarItem(
|
BottomNavigationBarItem(
|
||||||
icon: const Icon(Icons.miscellaneous_services),
|
icon: const Icon(Icons.miscellaneous_services),
|
||||||
label: AppLocalizations.of(context).services,
|
label: AppLocalizations.of(context).services,
|
||||||
|
|||||||
@@ -15,7 +15,17 @@ class AppThemes {
|
|||||||
colorScheme: colorScheme,
|
colorScheme: colorScheme,
|
||||||
useMaterial3: true,
|
useMaterial3: true,
|
||||||
textTheme: GoogleFonts.rubikTextTheme(),
|
textTheme: GoogleFonts.rubikTextTheme(),
|
||||||
);
|
).copyWith(
|
||||||
|
appBarTheme: AppBarTheme(
|
||||||
|
backgroundColor: const Color(0xFF01A04C),
|
||||||
|
titleTextStyle: TextStyle(
|
||||||
|
color: colorScheme.onPrimary,
|
||||||
|
fontWeight: FontWeight.w700,
|
||||||
|
fontSize: 20,
|
||||||
|
),
|
||||||
|
iconTheme: IconThemeData(color: colorScheme.onPrimary),
|
||||||
|
actionsIconTheme: IconThemeData(color: colorScheme.onPrimary),
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
static ThemeData getDarkTheme(ThemeType type) {
|
static ThemeData getDarkTheme(ThemeType type) {
|
||||||
@@ -32,7 +42,17 @@ class AppThemes {
|
|||||||
textTheme: GoogleFonts.rubikTextTheme(
|
textTheme: GoogleFonts.rubikTextTheme(
|
||||||
ThemeData(brightness: Brightness.dark).textTheme,
|
ThemeData(brightness: Brightness.dark).textTheme,
|
||||||
),
|
),
|
||||||
);
|
).copyWith(
|
||||||
|
appBarTheme: AppBarTheme(
|
||||||
|
backgroundColor: const Color(0xFF01A04C),
|
||||||
|
titleTextStyle: TextStyle(
|
||||||
|
color: colorScheme.onPrimary,
|
||||||
|
fontWeight: FontWeight.w700,
|
||||||
|
fontSize: 20,
|
||||||
|
),
|
||||||
|
iconTheme: IconThemeData(color: colorScheme.onPrimary),
|
||||||
|
actionsIconTheme: IconThemeData(color: colorScheme.onPrimary),
|
||||||
|
));
|
||||||
}
|
}
|
||||||
|
|
||||||
static Color _getSeedColor(ThemeType type) {
|
static Color _getSeedColor(ThemeType type) {
|
||||||
|
|||||||
@@ -1,5 +1,6 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:kmobile/api/services/branch_service.dart';
|
import 'package:kmobile/api/services/branch_service.dart';
|
||||||
|
import 'package:kmobile/api/services/cheque_service.dart';
|
||||||
import 'package:kmobile/api/services/limit_service.dart';
|
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';
|
||||||
@@ -56,6 +57,7 @@ Future<void> setupDependencies() async {
|
|||||||
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.registerSingleton<BranchService>(BranchService(getIt<Dio>()));
|
getIt.registerSingleton<BranchService>(BranchService(getIt<Dio>()));
|
||||||
|
getIt.registerSingleton<ChequeService>(ChequeService(getIt<Dio>()));
|
||||||
getIt.registerLazySingleton<ChangePasswordService>(
|
getIt.registerLazySingleton<ChangePasswordService>(
|
||||||
() => ChangePasswordService(getIt<Dio>()),
|
() => ChangePasswordService(getIt<Dio>()),
|
||||||
);
|
);
|
||||||
@@ -74,9 +76,9 @@ Dio _createDioClient() {
|
|||||||
final dio = Dio(
|
final dio = Dio(
|
||||||
BaseOptions(
|
BaseOptions(
|
||||||
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', //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', //prod small
|
//'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: {
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ class _AccountInfoScreen extends State<AccountInfoScreen> {
|
|||||||
return AppLocalizations.of(context).recurringDeposit;
|
return AppLocalizations.of(context).recurringDeposit;
|
||||||
case 'ca':
|
case 'ca':
|
||||||
return "Current Account";
|
return "Current Account";
|
||||||
case 'cc':
|
case 'cc':
|
||||||
return "Cash Credit Account";
|
return "Cash Credit Account";
|
||||||
case 'od':
|
case 'od':
|
||||||
return "Overdraft Account";
|
return "Overdraft Account";
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -2,6 +2,8 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:kmobile/data/models/user.dart';
|
import 'package:kmobile/data/models/user.dart';
|
||||||
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
||||||
|
|
||||||
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
class AllAccountsScreen extends StatefulWidget {
|
class AllAccountsScreen extends StatefulWidget {
|
||||||
final List<User> users;
|
final List<User> users;
|
||||||
const AllAccountsScreen({super.key, required this.users});
|
const AllAccountsScreen({super.key, required this.users});
|
||||||
@@ -19,19 +21,23 @@ class _AllAccountsScreenState extends State<AllAccountsScreen> {
|
|||||||
if (accountType == null || accountType.isEmpty) return 'N/A';
|
if (accountType == null || accountType.isEmpty) return 'N/A';
|
||||||
switch (accountType.toLowerCase()) {
|
switch (accountType.toLowerCase()) {
|
||||||
case 'sa':
|
case 'sa':
|
||||||
return "Savings Account"; // Using hardcoded strings for simplicity
|
return AppLocalizations.of(context).savingsAccount;
|
||||||
case 'sb':
|
case 'sb':
|
||||||
return "Savings Account";
|
return AppLocalizations.of(context).savingsAccount;
|
||||||
case 'ln':
|
case 'ln':
|
||||||
return "Loan Account";
|
return AppLocalizations.of(context).loanAccount;
|
||||||
case 'td':
|
case 'td':
|
||||||
return "Term Deposit";
|
return AppLocalizations.of(context).termDeposit;
|
||||||
case 'rd':
|
case 'rd':
|
||||||
return "Recurring Deposit";
|
return AppLocalizations.of(context).recurringDeposit;
|
||||||
case 'ca':
|
case 'ca':
|
||||||
return "Current Account";
|
return "Current Account";
|
||||||
|
case 'cc':
|
||||||
|
return "Cash Credit Account";
|
||||||
|
case 'od':
|
||||||
|
return "Overdraft Account";
|
||||||
default:
|
default:
|
||||||
return "Unknown Account";
|
return AppLocalizations.of(context).unknownAccount;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -39,18 +45,25 @@ class _AllAccountsScreenState extends State<AllAccountsScreen> {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: const Text('All Accounts'),
|
title: Text(AppLocalizations.of(context).viewall),
|
||||||
),
|
),
|
||||||
body: ListView.builder(
|
body: Column(
|
||||||
itemCount: widget.users.length,
|
children: [
|
||||||
itemBuilder: (context, index) {
|
const SizedBox(height: 16.0), // Added space below the app bar
|
||||||
final user = widget.users[index];
|
Expanded(
|
||||||
return Padding(
|
child: ListView.builder(
|
||||||
padding:
|
itemCount: widget.users.length,
|
||||||
const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
|
itemBuilder: (context, index) {
|
||||||
child: _buildAccountCard(user),
|
final user = widget.users[index];
|
||||||
);
|
return Padding(
|
||||||
},
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: 16.0, vertical: 8.0),
|
||||||
|
child: _buildAccountCard(user),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
), // Closing Expanded
|
||||||
|
], // Closing Column
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -147,7 +147,9 @@ class _ManageBeneficiariesScreen extends State<ManageBeneficiariesScreen> {
|
|||||||
child: TextField(
|
child: TextField(
|
||||||
controller: _searchController,
|
controller: _searchController,
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
hintText: "Search by name or account number",
|
hintText:
|
||||||
|
AppLocalizations.of(context).searchByNameOrAccountHint,
|
||||||
|
|
||||||
prefixIcon: const Icon(Icons.search),
|
prefixIcon: const Icon(Icons.search),
|
||||||
border: OutlineInputBorder(
|
border: OutlineInputBorder(
|
||||||
borderRadius: BorderRadius.circular(12),
|
borderRadius: BorderRadius.circular(12),
|
||||||
|
|||||||
@@ -56,8 +56,6 @@ class _BlockCardScreen extends State<BlockCardScreen> {
|
|||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text(
|
||||||
AppLocalizations.of(context).blockCard,
|
AppLocalizations.of(context).blockCard,
|
||||||
style:
|
|
||||||
const TextStyle(color: Colors.black, fontWeight: FontWeight.w500),
|
|
||||||
),
|
),
|
||||||
centerTitle: false,
|
centerTitle: false,
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -46,8 +46,6 @@ class _CardPinChangeDetailsScreen extends State<CardPinChangeDetailsScreen> {
|
|||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text(
|
||||||
AppLocalizations.of(context).cardDetails,
|
AppLocalizations.of(context).cardDetails,
|
||||||
style:
|
|
||||||
const TextStyle(color: Colors.black, fontWeight: FontWeight.w500),
|
|
||||||
),
|
),
|
||||||
centerTitle: false,
|
centerTitle: false,
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -46,8 +46,6 @@ class _CardPinSetScreen extends State<CardPinSetScreen> {
|
|||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text(
|
||||||
AppLocalizations.of(context).cardPin,
|
AppLocalizations.of(context).cardPin,
|
||||||
style:
|
|
||||||
const TextStyle(color: Colors.black, fontWeight: FontWeight.w500),
|
|
||||||
),
|
),
|
||||||
centerTitle: false,
|
centerTitle: false,
|
||||||
),
|
),
|
||||||
|
|||||||
361
lib/features/cheque/screens/cheque_enquiry_screen.dart
Normal file
361
lib/features/cheque/screens/cheque_enquiry_screen.dart
Normal file
@@ -0,0 +1,361 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:kmobile/api/services/cheque_service.dart';
|
||||||
|
import 'package:kmobile/data/models/user.dart';
|
||||||
|
import 'package:kmobile/di/injection.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
|
||||||
|
class ChequeEnquiryScreen extends StatefulWidget {
|
||||||
|
final List<User> users;
|
||||||
|
final int selectedIndex;
|
||||||
|
const ChequeEnquiryScreen({
|
||||||
|
super.key,
|
||||||
|
required this.users,
|
||||||
|
required this.selectedIndex,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<ChequeEnquiryScreen> createState() => _ChequeEnquiryScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _ChequeEnquiryScreenState extends State<ChequeEnquiryScreen> {
|
||||||
|
User? _selectedAccount;
|
||||||
|
final TextEditingController _searchController = TextEditingController();
|
||||||
|
var service = getIt<ChequeService>();
|
||||||
|
bool _isLoading = true;
|
||||||
|
List<Cheque> _allCheques = [];
|
||||||
|
Map<String, List<Cheque>> _groupedCheques = {};
|
||||||
|
List<User> _filteredUsers = [];
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_filteredUsers = widget.users
|
||||||
|
.where((user) => ['SA', 'SB', 'CA', 'CC'].contains(user.accountType))
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
if (widget.users.isNotEmpty && widget.selectedIndex < widget.users.length) {
|
||||||
|
if (_filteredUsers.isNotEmpty) {
|
||||||
|
if (_filteredUsers.contains(widget.users[widget.selectedIndex])) {
|
||||||
|
_selectedAccount = widget.users[widget.selectedIndex];
|
||||||
|
} else {
|
||||||
|
_selectedAccount = _filteredUsers.first;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
_selectedAccount = widget.users[widget.selectedIndex];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (_filteredUsers.isNotEmpty) {
|
||||||
|
_selectedAccount = _filteredUsers.first;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_loadCheques();
|
||||||
|
_searchController.addListener(() {
|
||||||
|
_filterCheques(_searchController.text);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _loadCheques() async {
|
||||||
|
if (_selectedAccount == null) {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
_groupedCheques = {};
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setState(() {
|
||||||
|
_isLoading = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
String instrType;
|
||||||
|
switch (_selectedAccount!.accountType) {
|
||||||
|
case 'SA':
|
||||||
|
case 'SB':
|
||||||
|
instrType = '10';
|
||||||
|
break;
|
||||||
|
case 'CA':
|
||||||
|
instrType = '11';
|
||||||
|
break;
|
||||||
|
case 'CC':
|
||||||
|
instrType = '13';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
instrType = '10';
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
final data = await service.ChequeEnquiry(
|
||||||
|
accountNumber: _selectedAccount!.accountNo!, instrType: instrType);
|
||||||
|
_allCheques = data;
|
||||||
|
_groupedCheques.clear();
|
||||||
|
for (var cheque in _allCheques) {
|
||||||
|
if (cheque.type != null) {
|
||||||
|
if (!_groupedCheques.containsKey(cheque.type)) {
|
||||||
|
_groupedCheques[cheque.type!] = [];
|
||||||
|
}
|
||||||
|
_groupedCheques[cheque.type!]!.add(cheque);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
_groupedCheques = {};
|
||||||
|
});
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text('Failed to fetch cheque status: ${e.toString()}'),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void _filterCheques(String query) {
|
||||||
|
_groupedCheques.clear();
|
||||||
|
List<Cheque> filteredCheques;
|
||||||
|
if (query.isEmpty) {
|
||||||
|
filteredCheques = _allCheques;
|
||||||
|
} else {
|
||||||
|
filteredCheques = _allCheques.where((cheque) {
|
||||||
|
final lowerQuery = query.toLowerCase();
|
||||||
|
return (cheque.ChequeNumber?.toLowerCase().contains(lowerQuery) ??
|
||||||
|
false) ||
|
||||||
|
(cheque.status?.toLowerCase().contains(lowerQuery) ?? false) ||
|
||||||
|
(cheque.fromCheque?.toLowerCase().contains(lowerQuery) ?? false) ||
|
||||||
|
(cheque.toCheque?.toLowerCase().contains(lowerQuery) ?? false);
|
||||||
|
}).toList();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var cheque in filteredCheques) {
|
||||||
|
if (cheque.type != null) {
|
||||||
|
if (!_groupedCheques.containsKey(cheque.type)) {
|
||||||
|
_groupedCheques[cheque.type!] = [];
|
||||||
|
}
|
||||||
|
_groupedCheques[cheque.type!]!.add(cheque);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setState(() {});
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(AppLocalizations.of(context).chequeEnquiryTitle),
|
||||||
|
centerTitle: false,
|
||||||
|
),
|
||||||
|
body: Stack(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Card(
|
||||||
|
elevation: 4,
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
AppLocalizations.of(context).accountNumber,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontWeight: FontWeight.bold, fontSize: 18),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 16),
|
||||||
|
if (_selectedAccount != null)
|
||||||
|
Expanded(
|
||||||
|
child: DropdownButton<User>(
|
||||||
|
value: _selectedAccount,
|
||||||
|
onChanged: (User? newUser) {
|
||||||
|
if (newUser != null) {
|
||||||
|
setState(() {
|
||||||
|
_selectedAccount = newUser;
|
||||||
|
_loadCheques();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
items: _filteredUsers.map((user) {
|
||||||
|
return DropdownMenuItem<User>(
|
||||||
|
value: user,
|
||||||
|
child: Text(user.accountNo.toString()),
|
||||||
|
);
|
||||||
|
}).toList(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
else
|
||||||
|
const Text('No accounts found'),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
Card(
|
||||||
|
elevation: 4,
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(8.0),
|
||||||
|
child: TextField(
|
||||||
|
controller: _searchController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context)
|
||||||
|
.searchByChequeDetailsHint,
|
||||||
|
prefixIcon: const Icon(Icons.search),
|
||||||
|
border: InputBorder
|
||||||
|
.none, // Remove border to make it look like it's inside the card
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
Expanded(
|
||||||
|
child: _isLoading
|
||||||
|
? const Center(child: CircularProgressIndicator())
|
||||||
|
: _groupedCheques.isEmpty
|
||||||
|
? Center(
|
||||||
|
child: Text(AppLocalizations.of(context)
|
||||||
|
.noChequeStatusFound))
|
||||||
|
: ListView(
|
||||||
|
children: _groupedCheques.entries.map((entry) {
|
||||||
|
return Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
...entry.value.map((cheque) =>
|
||||||
|
ChequeStatusTile(cheque: cheque)),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}).toList(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
IgnorePointer(
|
||||||
|
child: Center(
|
||||||
|
child: Opacity(
|
||||||
|
opacity: 0.07, // Reduced opacity
|
||||||
|
child: ClipOval(
|
||||||
|
child: Image.asset(
|
||||||
|
'assets/images/logo.png',
|
||||||
|
width: 200, // Adjust size as needed
|
||||||
|
height: 200, // Adjust size as needed
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class ChequeStatusTile extends StatelessWidget {
|
||||||
|
final Cheque cheque;
|
||||||
|
|
||||||
|
const ChequeStatusTile({
|
||||||
|
super.key,
|
||||||
|
required this.cheque,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
switch (cheque.type) {
|
||||||
|
case 'CI':
|
||||||
|
return _buildCiTile(context);
|
||||||
|
case 'PR':
|
||||||
|
return _buildPrTile(context);
|
||||||
|
case 'ST':
|
||||||
|
return _buildStTile(context);
|
||||||
|
default:
|
||||||
|
return const SizedBox.shrink();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildCiTile(BuildContext context) {
|
||||||
|
return Card(
|
||||||
|
margin: const EdgeInsets.symmetric(
|
||||||
|
vertical: 8.0,
|
||||||
|
),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(AppLocalizations.of(context).chequebookIssuedLabel,
|
||||||
|
style: Theme.of(context).textTheme.titleLarge),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
_buildInfoRow('Branch Code:', cheque.branchCode),
|
||||||
|
_buildInfoRow('From Cheque:', cheque.fromCheque),
|
||||||
|
_buildInfoRow('To Cheque:', cheque.toCheque),
|
||||||
|
_buildInfoRow('Date:', cheque.Date),
|
||||||
|
_buildInfoRow('Cheques Count:', cheque.Chequescount),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildPrTile(BuildContext context) {
|
||||||
|
return Card(
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(AppLocalizations.of(context).presentedChequeLabel,
|
||||||
|
style: Theme.of(context).textTheme.titleLarge),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
_buildInfoRow('Branch Code:', cheque.branchCode),
|
||||||
|
_buildInfoRow('Cheque Number:', cheque.ChequeNumber),
|
||||||
|
_buildInfoRow('Date:', cheque.Date),
|
||||||
|
_buildInfoRow('Transaction Code:', cheque.transactionCode),
|
||||||
|
_buildInfoRow('Amount:', '₹ ${cheque.amount.toString()}'),
|
||||||
|
_buildInfoRow('Status:', cheque.status),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildStTile(BuildContext context) {
|
||||||
|
return Card(
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(AppLocalizations.of(context).stopChequeLabel,
|
||||||
|
style: Theme.of(context).textTheme.titleLarge),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
_buildInfoRow('Branch Code:', cheque.branchCode),
|
||||||
|
_buildInfoRow('From Cheque:', cheque.fromCheque),
|
||||||
|
_buildInfoRow('To Cheque:', cheque.toCheque),
|
||||||
|
_buildInfoRow('Stop Issue Date:', cheque.stopIssueDate),
|
||||||
|
_buildInfoRow('Stop Expiry Date:', cheque.StopExpiryDate),
|
||||||
|
_buildInfoRow('Cheques Count:', cheque.Chequescount),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildInfoRow(String label, String? value) {
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 4.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
Text(label, style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
Text(value ?? ''),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,76 +1,82 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:kmobile/features/enquiry/screens/enquiry_screen.dart';
|
import 'package:kmobile/data/models/user.dart';
|
||||||
|
import 'package:kmobile/features/cheque/screens/cheque_enquiry_screen.dart';
|
||||||
|
import 'package:kmobile/features/cheque/screens/stop_cheque_screen.dart';
|
||||||
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
||||||
import '../../../l10n/app_localizations.dart';
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
class ChequeManagementScreen extends StatefulWidget {
|
class ChequeManagementScreen extends StatefulWidget {
|
||||||
const ChequeManagementScreen({super.key});
|
final List<User> users;
|
||||||
|
final int selectedIndex;
|
||||||
|
const ChequeManagementScreen({
|
||||||
|
super.key,
|
||||||
|
required this.users,
|
||||||
|
required this.selectedIndex,
|
||||||
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
State<ChequeManagementScreen> createState() => _ChequeManagementScreen();
|
State<ChequeManagementScreen> createState() => _ChequeManagementScreen();
|
||||||
}
|
}
|
||||||
|
|
||||||
class _ChequeManagementScreen extends State<ChequeManagementScreen> {
|
class _ChequeManagementScreen extends State<ChequeManagementScreen> {
|
||||||
|
List<User> get users => widget.users;
|
||||||
|
int get selectedAccountIndex => widget.selectedIndex;
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text(
|
||||||
AppLocalizations.of(context).chequeManagement,
|
AppLocalizations.of(context).chequeManagement,
|
||||||
style: TextStyle(
|
|
||||||
color: Theme.of(context).textTheme.titleLarge?.color,
|
|
||||||
fontWeight: FontWeight.w500),
|
|
||||||
),
|
),
|
||||||
centerTitle: false,
|
centerTitle: false,
|
||||||
),
|
),
|
||||||
body: Stack(
|
body: Stack(
|
||||||
children: [
|
children: [
|
||||||
ListView(
|
Padding(
|
||||||
children: [
|
padding: const EdgeInsets.symmetric(vertical: 16.0),
|
||||||
const SizedBox(height: 15),
|
child: Column(
|
||||||
ChequeManagementTile(
|
crossAxisAlignment: CrossAxisAlignment.stretch,
|
||||||
icon: Symbols.add,
|
children: [
|
||||||
label: AppLocalizations.of(context).requestChequeBook,
|
Expanded(
|
||||||
onTap: () {},
|
child: ChequeManagementCardTile(
|
||||||
),
|
icon: Symbols.payments,
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
label: AppLocalizations.of(context).chequeEnquiryTitle,
|
||||||
ChequeManagementTile(
|
subtitle:
|
||||||
icon: Symbols.data_alert,
|
AppLocalizations.of(context).chequeEnquirySubtitle,
|
||||||
label: AppLocalizations.of(context).enquiry,
|
onTap: () {
|
||||||
onTap: () {
|
Navigator.push(
|
||||||
Navigator.push(
|
context,
|
||||||
context,
|
MaterialPageRoute(
|
||||||
MaterialPageRoute(
|
builder: (context) => ChequeEnquiryScreen(
|
||||||
builder: (context) => const EnquiryScreen()),
|
users: users,
|
||||||
);
|
selectedIndex: selectedAccountIndex,
|
||||||
},
|
),
|
||||||
),
|
),
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
);
|
||||||
ChequeManagementTile(
|
},
|
||||||
icon: Symbols.approval_delegation,
|
),
|
||||||
label: AppLocalizations.of(context).chequeDeposit,
|
),
|
||||||
onTap: () {},
|
Expanded(
|
||||||
),
|
child: ChequeManagementCardTile(
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
icon: Symbols.block_sharp,
|
||||||
ChequeManagementTile(
|
label: AppLocalizations.of(context).stopCheque,
|
||||||
icon: Symbols.front_hand,
|
subtitle: AppLocalizations.of(context).stopChequeSubtitle,
|
||||||
label: AppLocalizations.of(context).stopCheque,
|
onTap: () {
|
||||||
onTap: () {},
|
Navigator.push(
|
||||||
),
|
context,
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
MaterialPageRoute(
|
||||||
ChequeManagementTile(
|
builder: (context) => StopChequeScreen(
|
||||||
icon: Symbols.cancel_presentation,
|
users: users,
|
||||||
label: AppLocalizations.of(context).revokeStop,
|
selectedIndex: selectedAccountIndex,
|
||||||
onTap: () {},
|
),
|
||||||
),
|
),
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
);
|
||||||
ChequeManagementTile(
|
},
|
||||||
icon: Symbols.payments,
|
),
|
||||||
label: AppLocalizations.of(context).positivePay,
|
),
|
||||||
onTap: () {},
|
],
|
||||||
),
|
),
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
|
||||||
],
|
|
||||||
),
|
),
|
||||||
IgnorePointer(
|
IgnorePointer(
|
||||||
child: Center(
|
child: Center(
|
||||||
@@ -92,25 +98,79 @@ class _ChequeManagementScreen extends State<ChequeManagementScreen> {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
class ChequeManagementTile extends StatelessWidget {
|
class ChequeManagementCardTile extends StatelessWidget {
|
||||||
final IconData icon;
|
final IconData icon;
|
||||||
final String label;
|
final String label;
|
||||||
|
final String? subtitle;
|
||||||
final VoidCallback onTap;
|
final VoidCallback onTap;
|
||||||
|
final bool disable;
|
||||||
|
|
||||||
const ChequeManagementTile({
|
const ChequeManagementCardTile({
|
||||||
super.key,
|
super.key,
|
||||||
required this.icon,
|
required this.icon,
|
||||||
required this.label,
|
required this.label,
|
||||||
|
this.subtitle,
|
||||||
required this.onTap,
|
required this.onTap,
|
||||||
|
this.disable = false,
|
||||||
});
|
});
|
||||||
|
|
||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return ListTile(
|
final theme = Theme.of(context);
|
||||||
leading: Icon(icon),
|
return Card(
|
||||||
title: Text(label),
|
margin: const EdgeInsets.symmetric(horizontal: 16.0, vertical: 8.0),
|
||||||
trailing: const Icon(Symbols.arrow_right, size: 20),
|
shape: RoundedRectangleBorder(
|
||||||
onTap: onTap,
|
borderRadius: BorderRadius.circular(12.0),
|
||||||
|
),
|
||||||
|
elevation: 4, // Add some elevation for better visual separation
|
||||||
|
child: InkWell(
|
||||||
|
onTap:
|
||||||
|
disable ? null : onTap, // Disable InkWell if the tile is disabled
|
||||||
|
borderRadius: BorderRadius.circular(12.0),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 24.0, horizontal: 16.0),
|
||||||
|
child: Center(
|
||||||
|
child: SingleChildScrollView(
|
||||||
|
child: Column(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
Icon(
|
||||||
|
icon,
|
||||||
|
size: 48, // Make icon larger
|
||||||
|
color: disable
|
||||||
|
? theme.disabledColor
|
||||||
|
: theme.colorScheme.primary,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 12),
|
||||||
|
Text(
|
||||||
|
label,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: theme.textTheme.titleLarge?.copyWith(
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: disable
|
||||||
|
? theme.disabledColor
|
||||||
|
: theme.colorScheme.onSurface,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
if (subtitle != null)
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 8.0),
|
||||||
|
child: Text(
|
||||||
|
subtitle!,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: theme.textTheme.bodyMedium?.copyWith(
|
||||||
|
color: disable
|
||||||
|
? theme.disabledColor
|
||||||
|
: theme.colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
349
lib/features/cheque/screens/stop_cheque_screen.dart
Normal file
349
lib/features/cheque/screens/stop_cheque_screen.dart
Normal file
@@ -0,0 +1,349 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:kmobile/features/cheque/screens/stop_multiple_cheques_screen.dart';
|
||||||
|
import 'package:kmobile/features/cheque/screens/stop_single_cheque_screen.dart';
|
||||||
|
import 'package:kmobile/api/services/cheque_service.dart';
|
||||||
|
import 'package:kmobile/data/models/user.dart';
|
||||||
|
import 'package:kmobile/di/injection.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
|
||||||
|
class StopChequeScreen extends StatefulWidget {
|
||||||
|
final List<User> users;
|
||||||
|
final int selectedIndex;
|
||||||
|
const StopChequeScreen({
|
||||||
|
super.key,
|
||||||
|
required this.users,
|
||||||
|
required this.selectedIndex,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<StopChequeScreen> createState() => _StopChequeScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _StopChequeScreenState extends State<StopChequeScreen> {
|
||||||
|
User? _selectedAccount;
|
||||||
|
var service = getIt<ChequeService>();
|
||||||
|
bool _isLoading = true;
|
||||||
|
Cheque? _ciCheque;
|
||||||
|
List<User> _filteredUsers = [];
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
_filteredUsers = widget.users
|
||||||
|
.where((user) => ['SA', 'SB', 'CA', 'CC'].contains(user.accountType))
|
||||||
|
.toList();
|
||||||
|
|
||||||
|
if (widget.users.isNotEmpty && widget.selectedIndex < widget.users.length) {
|
||||||
|
if (_filteredUsers.isNotEmpty) {
|
||||||
|
if (_filteredUsers.contains(widget.users[widget.selectedIndex])) {
|
||||||
|
_selectedAccount = widget.users[widget.selectedIndex];
|
||||||
|
} else {
|
||||||
|
_selectedAccount = _filteredUsers.first;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
_selectedAccount = widget.users[widget.selectedIndex];
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (_filteredUsers.isNotEmpty) {
|
||||||
|
_selectedAccount = _filteredUsers.first;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_loadCheques();
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _loadCheques() async {
|
||||||
|
if (_selectedAccount == null) {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
_ciCheque = null;
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
setState(() {
|
||||||
|
_isLoading = true;
|
||||||
|
});
|
||||||
|
|
||||||
|
String instrType;
|
||||||
|
switch (_selectedAccount!.accountType) {
|
||||||
|
case 'SA':
|
||||||
|
case 'SB':
|
||||||
|
instrType = '10';
|
||||||
|
break;
|
||||||
|
case 'CA':
|
||||||
|
instrType = '11';
|
||||||
|
break;
|
||||||
|
case 'CC':
|
||||||
|
instrType = '13';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
instrType = '10';
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
final data = await service.ChequeEnquiry(
|
||||||
|
accountNumber: _selectedAccount!.accountNo!, instrType: instrType);
|
||||||
|
final ciCheques = data.where((cheque) => cheque.type == 'CI').toList();
|
||||||
|
setState(() {
|
||||||
|
_ciCheque = ciCheques.isNotEmpty ? ciCheques.first : null;
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
} catch (e) {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
_ciCheque = null;
|
||||||
|
});
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text('Failed to fetch cheque status: ${e.toString()}'),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
String _getAccountTypeDisplayName(String accountType) {
|
||||||
|
switch (accountType.toLowerCase()) {
|
||||||
|
case 'sa':
|
||||||
|
return AppLocalizations.of(context).savingsAccount;
|
||||||
|
case 'sb':
|
||||||
|
return AppLocalizations.of(context).savingsAccount;
|
||||||
|
case 'ca':
|
||||||
|
return "Current Account";
|
||||||
|
case 'cc':
|
||||||
|
return "Cash Credit Account";
|
||||||
|
default:
|
||||||
|
return accountType;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(AppLocalizations.of(context).stopChequeTitle),
|
||||||
|
centerTitle: false,
|
||||||
|
),
|
||||||
|
body: Stack(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Column(
|
||||||
|
children: [
|
||||||
|
Card(
|
||||||
|
elevation: 4,
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
AppLocalizations.of(context).accountNumber,
|
||||||
|
style: const TextStyle(
|
||||||
|
fontWeight: FontWeight.bold, fontSize: 18),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 16),
|
||||||
|
if (_selectedAccount != null)
|
||||||
|
Expanded(
|
||||||
|
child: DropdownButton<User>(
|
||||||
|
value: _selectedAccount,
|
||||||
|
onChanged: (User? newUser) {
|
||||||
|
if (newUser != null) {
|
||||||
|
setState(() {
|
||||||
|
_selectedAccount = newUser;
|
||||||
|
_loadCheques();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
items: _filteredUsers.map((user) {
|
||||||
|
return DropdownMenuItem<User>(
|
||||||
|
value: user,
|
||||||
|
child: Text(user.accountNo.toString()),
|
||||||
|
);
|
||||||
|
}).toList(),
|
||||||
|
),
|
||||||
|
)
|
||||||
|
else
|
||||||
|
Text(AppLocalizations.of(context).noAccountsFound),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
Row(
|
||||||
|
children: [
|
||||||
|
Expanded(
|
||||||
|
child: Card(
|
||||||
|
color: Theme.of(context).colorScheme.primaryContainer,
|
||||||
|
elevation: 4,
|
||||||
|
child: InkWell(
|
||||||
|
onTap: () {
|
||||||
|
if (_selectedAccount != null && _ciCheque != null) {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => StopSingleChequeScreen(
|
||||||
|
selectedAccount: _selectedAccount!,
|
||||||
|
date: _ciCheque!.Date!,
|
||||||
|
instrType: _ciCheque!.InstrType!,
|
||||||
|
fromCheque: _ciCheque!.fromCheque!,
|
||||||
|
toCheque: _ciCheque!.toCheque!,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text(AppLocalizations.of(context)
|
||||||
|
.noChequebookToStop),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Center(
|
||||||
|
child: Text(
|
||||||
|
AppLocalizations.of(context)
|
||||||
|
.stopSingleChequeTitle,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 16,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onPrimaryContainer,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 10),
|
||||||
|
Expanded(
|
||||||
|
child: Card(
|
||||||
|
color: Theme.of(context).colorScheme.primaryContainer,
|
||||||
|
elevation: 4,
|
||||||
|
child: InkWell(
|
||||||
|
onTap: () {
|
||||||
|
if (_selectedAccount != null) {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) =>
|
||||||
|
StopMultipleChequesScreen(
|
||||||
|
selectedAccount: _selectedAccount!,
|
||||||
|
date: _ciCheque!.Date!,
|
||||||
|
instrType: _ciCheque!.InstrType!,
|
||||||
|
fromCheque: _ciCheque!.fromCheque!,
|
||||||
|
toCheque: _ciCheque!.toCheque!,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text(AppLocalizations.of(context)
|
||||||
|
.pleaseSelectAccountFirst),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Center(
|
||||||
|
child: Text(
|
||||||
|
AppLocalizations.of(context)
|
||||||
|
.stopMultipleChequesButton,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 16,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onSecondaryContainer,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
const SizedBox(height: 20),
|
||||||
|
Expanded(
|
||||||
|
child: _isLoading
|
||||||
|
? const Center(child: CircularProgressIndicator())
|
||||||
|
: _ciCheque == null
|
||||||
|
? Center(
|
||||||
|
child: Text(AppLocalizations.of(context)
|
||||||
|
.noChequeIssuedStatus))
|
||||||
|
: _buildCiTile(context, _ciCheque!),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
IgnorePointer(
|
||||||
|
child: Center(
|
||||||
|
child: Opacity(
|
||||||
|
opacity: 0.07, // Reduced opacity
|
||||||
|
child: ClipOval(
|
||||||
|
child: Image.asset(
|
||||||
|
'assets/images/logo.png',
|
||||||
|
width: 200, // Adjust size as needed
|
||||||
|
height: 200, // Adjust size as needed
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildCiTile(BuildContext context, Cheque cheque) {
|
||||||
|
return Card(
|
||||||
|
margin: const EdgeInsets.symmetric(
|
||||||
|
vertical: 8.0,
|
||||||
|
),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(AppLocalizations.of(context).chequebookDetailsTitle,
|
||||||
|
style: Theme.of(context).textTheme.titleLarge),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
_buildInfoRow('Account Number:', _selectedAccount!.accountNo!),
|
||||||
|
_buildInfoRow('Customer Name:', _selectedAccount!.name!),
|
||||||
|
_buildInfoRow('CIF Number:', _selectedAccount!.cifNumber!),
|
||||||
|
_buildInfoRow('Account Type:',
|
||||||
|
_getAccountTypeDisplayName(_selectedAccount!.accountType!)),
|
||||||
|
_buildInfoRow('Branch Code:', cheque.branchCode),
|
||||||
|
_buildInfoRow('Starting Cheque Number:', cheque.fromCheque),
|
||||||
|
_buildInfoRow('Ending Cheque Number:', cheque.toCheque),
|
||||||
|
_buildInfoRow('Issue Date:', cheque.Date),
|
||||||
|
_buildInfoRow('Number of Cheques:', cheque.Chequescount),
|
||||||
|
_buildInfoRow('Instrument Type:', cheque.InstrType),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildInfoRow(String label, String? value) {
|
||||||
|
return Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 4.0),
|
||||||
|
child: Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
|
children: [
|
||||||
|
Text(label, style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
Text(value ?? ''),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
287
lib/features/cheque/screens/stop_multiple_cheques_screen.dart
Normal file
287
lib/features/cheque/screens/stop_multiple_cheques_screen.dart
Normal file
@@ -0,0 +1,287 @@
|
|||||||
|
import 'dart:convert';
|
||||||
|
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:kmobile/api/services/cheque_service.dart';
|
||||||
|
import 'package:kmobile/data/models/user.dart';
|
||||||
|
import 'package:kmobile/di/injection.dart';
|
||||||
|
import 'package:kmobile/features/fund_transfer/screens/transaction_pin_screen.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
|
||||||
|
class StopMultipleChequesScreen extends StatefulWidget {
|
||||||
|
final User selectedAccount;
|
||||||
|
final String date;
|
||||||
|
final String instrType;
|
||||||
|
final String fromCheque;
|
||||||
|
final String toCheque;
|
||||||
|
|
||||||
|
const StopMultipleChequesScreen(
|
||||||
|
{super.key,
|
||||||
|
required this.selectedAccount,
|
||||||
|
required this.date,
|
||||||
|
required this.instrType,
|
||||||
|
required this.fromCheque,
|
||||||
|
required this.toCheque});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<StopMultipleChequesScreen> createState() =>
|
||||||
|
_StopMultipleChequesScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _StopMultipleChequesScreenState extends State<StopMultipleChequesScreen> {
|
||||||
|
final _formKey = GlobalKey<FormState>();
|
||||||
|
final _stopFromChequeNoController = TextEditingController();
|
||||||
|
final _stopToChequeNoController = TextEditingController();
|
||||||
|
final _stopIssueDateController = TextEditingController();
|
||||||
|
final _stopExpiryDateController = TextEditingController();
|
||||||
|
final _stopAmountController = TextEditingController();
|
||||||
|
final _stopCommentController = TextEditingController();
|
||||||
|
final _chequeService = getIt<ChequeService>();
|
||||||
|
|
||||||
|
String _formatDate(String dateString) {
|
||||||
|
if (dateString.length != 8) {
|
||||||
|
return dateString; // Return as is if not in expected ddmmyyyy format
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
final day = dateString.substring(0, 2);
|
||||||
|
final month = dateString.substring(2, 4);
|
||||||
|
final year = dateString.substring(4, 8);
|
||||||
|
return '$day/$month/$year';
|
||||||
|
} catch (e) {
|
||||||
|
return dateString; // Return original string on error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _showResponseDialog(String title, String message) async {
|
||||||
|
return showDialog<void>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false, // user must tap button!
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: Text(title),
|
||||||
|
content: SingleChildScrollView(
|
||||||
|
child: ListBody(
|
||||||
|
children: <Widget>[
|
||||||
|
Text(message),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
actions: <Widget>[
|
||||||
|
TextButton(
|
||||||
|
child: const Text('Close'),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(AppLocalizations.of(context).stopMultipleChequesTitle),
|
||||||
|
),
|
||||||
|
body: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Form(
|
||||||
|
key: _formKey,
|
||||||
|
child: ListView(
|
||||||
|
children: [
|
||||||
|
Card(
|
||||||
|
elevation: 0,
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||||
|
child: ListTile(
|
||||||
|
leading: Image.asset(
|
||||||
|
'assets/images/logo.png',
|
||||||
|
width: 40,
|
||||||
|
height: 40,
|
||||||
|
),
|
||||||
|
title: Text(widget.selectedAccount.accountNo!),
|
||||||
|
subtitle:
|
||||||
|
Text(AppLocalizations.of(context).accountNumberTitle),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopFromChequeNoController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).fromChequeNumberHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
validator: (value) {
|
||||||
|
if (value == null || value.isEmpty) {
|
||||||
|
return AppLocalizations.of(context)
|
||||||
|
.pleaseEnterChequeNumberError;
|
||||||
|
}
|
||||||
|
final chequeNumber = int.tryParse(value);
|
||||||
|
final fromCheque = int.tryParse(widget.fromCheque);
|
||||||
|
final toCheque = int.tryParse(widget.toCheque);
|
||||||
|
if (chequeNumber == null ||
|
||||||
|
fromCheque == null ||
|
||||||
|
toCheque == null) {
|
||||||
|
return AppLocalizations.of(context)
|
||||||
|
.invalidChequeNumberFormatError;
|
||||||
|
}
|
||||||
|
if (chequeNumber < fromCheque || chequeNumber > toCheque) {
|
||||||
|
return AppLocalizations.of(context).chequeNumberRangeError(
|
||||||
|
widget.fromCheque, widget.toCheque);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopToChequeNoController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).toChequeNumberHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
validator: (value) {
|
||||||
|
if (value == null || value.isEmpty) {
|
||||||
|
return AppLocalizations.of(context)
|
||||||
|
.pleaseEnterChequeNumberError;
|
||||||
|
}
|
||||||
|
final chequeNumber = int.tryParse(value);
|
||||||
|
final fromCheque = int.tryParse(widget.fromCheque);
|
||||||
|
final toCheque = int.tryParse(widget.toCheque);
|
||||||
|
if (chequeNumber == null ||
|
||||||
|
fromCheque == null ||
|
||||||
|
toCheque == null) {
|
||||||
|
return AppLocalizations.of(context)
|
||||||
|
.invalidChequeNumberFormatError;
|
||||||
|
}
|
||||||
|
if (chequeNumber < fromCheque || chequeNumber > toCheque) {
|
||||||
|
return AppLocalizations.of(context).chequeNumberRangeError(
|
||||||
|
widget.fromCheque, widget.toCheque);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
initialValue: widget.instrType,
|
||||||
|
readOnly: true,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).instrumentTypeLabel,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopIssueDateController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopIssueDateHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.datetime,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopExpiryDateController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopExpiryDateHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.datetime,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopAmountController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopAmountHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopCommentController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopCommentHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
initialValue: _formatDate(widget.date),
|
||||||
|
readOnly: true,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText:
|
||||||
|
AppLocalizations.of(context).chequebookIssueDateHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 32),
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: () {
|
||||||
|
if (_formKey.currentState!.validate()) {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => TransactionPinScreen(
|
||||||
|
onPinCompleted: (ctx, pin) async {
|
||||||
|
Navigator.pop(context);
|
||||||
|
try {
|
||||||
|
final response = await _chequeService.stopCheque(
|
||||||
|
accountno: widget.selectedAccount.accountNo!,
|
||||||
|
stopFromChequeNo:
|
||||||
|
_stopFromChequeNoController.text,
|
||||||
|
instrType: widget.instrType,
|
||||||
|
stopToChequeNo: _stopToChequeNoController.text,
|
||||||
|
stopIssueDate: _stopIssueDateController.text,
|
||||||
|
stopExpiryDate: _stopExpiryDateController.text,
|
||||||
|
stopAmount: _stopAmountController.text,
|
||||||
|
stopComment: _stopCommentController.text,
|
||||||
|
chequeIssueDate: widget.date,
|
||||||
|
tpin: pin,
|
||||||
|
);
|
||||||
|
if (!mounted) return;
|
||||||
|
final decodedResponse = jsonDecode(response);
|
||||||
|
final status = decodedResponse['status'];
|
||||||
|
final message = decodedResponse['message'];
|
||||||
|
if (status == 'SUCCESS') {
|
||||||
|
_showResponseDialog('Success', message);
|
||||||
|
} else {
|
||||||
|
_showResponseDialog('Error', message);
|
||||||
|
}
|
||||||
|
} on Exception catch (e) {
|
||||||
|
print('inside catch block');
|
||||||
|
print(e.toString());
|
||||||
|
|
||||||
|
try {
|
||||||
|
final errorBodyString =
|
||||||
|
e.toString().split('Exception: ')[1];
|
||||||
|
final errorBody = jsonDecode(errorBodyString);
|
||||||
|
if (errorBody.containsKey('error') &&
|
||||||
|
errorBody['error'] == 'INCORRECT_TPIN') {
|
||||||
|
_showResponseDialog('Invalid TPIN',
|
||||||
|
'The TPIN you entered is incorrect. Please try again.');
|
||||||
|
} else {
|
||||||
|
_showResponseDialog(
|
||||||
|
'Error', 'Internal Server Error');
|
||||||
|
}
|
||||||
|
} catch (_) {
|
||||||
|
_showResponseDialog(
|
||||||
|
'Error', 'Internal Server Error');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Text(AppLocalizations.of(context).stopChequeButton),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
256
lib/features/cheque/screens/stop_single_cheque_screen.dart
Normal file
256
lib/features/cheque/screens/stop_single_cheque_screen.dart
Normal file
@@ -0,0 +1,256 @@
|
|||||||
|
import 'dart:convert';
|
||||||
|
import 'package:kmobile/data/models/user.dart';
|
||||||
|
import 'package:kmobile/di/injection.dart';
|
||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:kmobile/api/services/cheque_service.dart';
|
||||||
|
import 'package:kmobile/features/fund_transfer/screens/transaction_pin_screen.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
|
|
||||||
|
class StopSingleChequeScreen extends StatefulWidget {
|
||||||
|
final User selectedAccount;
|
||||||
|
final String date;
|
||||||
|
final String instrType;
|
||||||
|
final String fromCheque;
|
||||||
|
final String toCheque;
|
||||||
|
|
||||||
|
const StopSingleChequeScreen(
|
||||||
|
{super.key,
|
||||||
|
required this.selectedAccount,
|
||||||
|
required this.date,
|
||||||
|
required this.instrType,
|
||||||
|
required this.fromCheque,
|
||||||
|
required this.toCheque});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<StopSingleChequeScreen> createState() => _StopSingleChequeScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _StopSingleChequeScreenState extends State<StopSingleChequeScreen> {
|
||||||
|
final _formKey = GlobalKey<FormState>();
|
||||||
|
final _stopFromChequeNoController = TextEditingController();
|
||||||
|
final _stopIssueDateController = TextEditingController();
|
||||||
|
final _stopExpiryDateController = TextEditingController();
|
||||||
|
final _stopAmountController = TextEditingController();
|
||||||
|
final _stopCommentController = TextEditingController();
|
||||||
|
final _chequeService = getIt<ChequeService>();
|
||||||
|
|
||||||
|
String _formatDate(String dateString) {
|
||||||
|
if (dateString.length != 8) {
|
||||||
|
return dateString; // Return as is if not in expected ddmmyyyy format
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
final day = dateString.substring(0, 2);
|
||||||
|
final month = dateString.substring(2, 4);
|
||||||
|
final year = dateString.substring(4, 8);
|
||||||
|
return '$day/$month/$year';
|
||||||
|
} catch (e) {
|
||||||
|
return dateString; // Return original string on error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Future<void> _showResponseDialog(String title, String message) async {
|
||||||
|
return showDialog<void>(
|
||||||
|
context: context,
|
||||||
|
barrierDismissible: false, // user must tap button!
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: Text(title),
|
||||||
|
content: SingleChildScrollView(
|
||||||
|
child: ListBody(
|
||||||
|
children: <Widget>[
|
||||||
|
Text(message),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
actions: <Widget>[
|
||||||
|
TextButton(
|
||||||
|
child: Text(AppLocalizations.of(context).closeButton),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(
|
||||||
|
title: Text(AppLocalizations.of(context).stopSingleChequeTitle),
|
||||||
|
),
|
||||||
|
body: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: Form(
|
||||||
|
key: _formKey,
|
||||||
|
child: ListView(
|
||||||
|
children: [
|
||||||
|
Card(
|
||||||
|
elevation: 0,
|
||||||
|
margin: const EdgeInsets.symmetric(vertical: 8.0),
|
||||||
|
child: ListTile(
|
||||||
|
leading: Image.asset(
|
||||||
|
'assets/images/logo.png',
|
||||||
|
width: 40,
|
||||||
|
height: 40,
|
||||||
|
),
|
||||||
|
title: Text(widget.selectedAccount.accountNo!),
|
||||||
|
subtitle:
|
||||||
|
Text(AppLocalizations.of(context).accountNumberLabel),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopFromChequeNoController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).chequeNumberLabel,
|
||||||
|
border: OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
validator: (value) {
|
||||||
|
if (value == null || value.isEmpty) {
|
||||||
|
return AppLocalizations.of(context)
|
||||||
|
.pleaseEnterChequeNumberError;
|
||||||
|
}
|
||||||
|
final chequeNumber = int.tryParse(value);
|
||||||
|
final fromCheque = int.tryParse(widget.fromCheque);
|
||||||
|
final toCheque = int.tryParse(widget.toCheque);
|
||||||
|
if (chequeNumber == null ||
|
||||||
|
fromCheque == null ||
|
||||||
|
toCheque == null) {
|
||||||
|
return AppLocalizations.of(context)
|
||||||
|
.invalidChequeNumberFormatError;
|
||||||
|
}
|
||||||
|
if (chequeNumber < fromCheque || chequeNumber > toCheque) {
|
||||||
|
return AppLocalizations.of(context).chequeNumberRangeError(
|
||||||
|
widget.fromCheque, widget.toCheque);
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
},
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
initialValue: widget.instrType,
|
||||||
|
readOnly: true,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).instrumentTypeLabel,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopIssueDateController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopIssueDateLabel,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.datetime,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopExpiryDateController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopExpiryDateLabel,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.datetime,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopAmountController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopAmountHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
controller: _stopCommentController,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).stopCommentHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
TextFormField(
|
||||||
|
initialValue: _formatDate(widget.date),
|
||||||
|
readOnly: true,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText:
|
||||||
|
AppLocalizations.of(context).chequebookIssueDateHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 32),
|
||||||
|
ElevatedButton(
|
||||||
|
onPressed: () {
|
||||||
|
if (_formKey.currentState!.validate()) {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => TransactionPinScreen(
|
||||||
|
onPinCompleted: (ctx, pin) async {
|
||||||
|
Navigator.pop(context);
|
||||||
|
try {
|
||||||
|
final response = await _chequeService.stopCheque(
|
||||||
|
accountno: widget.selectedAccount.accountNo!,
|
||||||
|
stopFromChequeNo:
|
||||||
|
_stopFromChequeNoController.text,
|
||||||
|
instrType: widget.instrType,
|
||||||
|
stopToChequeNo:
|
||||||
|
_stopFromChequeNoController.text,
|
||||||
|
stopIssueDate: _stopIssueDateController.text,
|
||||||
|
stopExpiryDate: _stopExpiryDateController.text,
|
||||||
|
stopAmount: _stopAmountController.text,
|
||||||
|
stopComment: _stopCommentController.text,
|
||||||
|
chequeIssueDate: widget.date,
|
||||||
|
tpin: pin,
|
||||||
|
);
|
||||||
|
if (!mounted) return;
|
||||||
|
final decodedResponse = jsonDecode(response);
|
||||||
|
final status = decodedResponse['status'];
|
||||||
|
final message = decodedResponse['message'];
|
||||||
|
if (status == 'SUCCESS') {
|
||||||
|
_showResponseDialog('Success', message);
|
||||||
|
} else {
|
||||||
|
_showResponseDialog('Error', message);
|
||||||
|
}
|
||||||
|
} on Exception catch (e) {
|
||||||
|
print('inside catch block');
|
||||||
|
print(e.toString());
|
||||||
|
|
||||||
|
try {
|
||||||
|
final errorBodyString =
|
||||||
|
e.toString().split('Exception: ')[1];
|
||||||
|
final errorBody = jsonDecode(errorBodyString);
|
||||||
|
if (errorBody.containsKey('error') &&
|
||||||
|
errorBody['error'] == 'INCORRECT_TPIN') {
|
||||||
|
_showResponseDialog('Invalid TPIN',
|
||||||
|
'The TPIN you entered is incorrect. Please try again.');
|
||||||
|
} else {
|
||||||
|
_showResponseDialog(
|
||||||
|
'Error', 'Internal Server Error');
|
||||||
|
}
|
||||||
|
} catch (_) {
|
||||||
|
_showResponseDialog(
|
||||||
|
'Error', 'Internal Server Error');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
child: Text(AppLocalizations.of(context).stopChequeButton),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
|
import 'package:flutter/cupertino.dart';
|
||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_svg/svg.dart';
|
|
||||||
import 'package:kmobile/data/models/user.dart';
|
import 'package:kmobile/data/models/user.dart';
|
||||||
|
import 'package:material_symbols_icons/material_symbols_icons.dart';
|
||||||
import '../../../l10n/app_localizations.dart';
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
class CustomerInfoScreen extends StatefulWidget {
|
class CustomerInfoScreen extends StatefulWidget {
|
||||||
@@ -13,6 +14,7 @@ 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;
|
||||||
|
int _selectedCard = 0; // 0 for Personal Info, 1 for KYC
|
||||||
|
|
||||||
String _maskPrimaryId(String? primaryId) {
|
String _maskPrimaryId(String? primaryId) {
|
||||||
if (primaryId == null || primaryId.length <= 4) {
|
if (primaryId == null || primaryId.length <= 4) {
|
||||||
@@ -26,100 +28,207 @@ class _CustomerInfoScreenState extends State<CustomerInfoScreen> {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final theme = Theme.of(context);
|
final theme = Theme.of(context);
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text(
|
||||||
AppLocalizations.of(context)
|
AppLocalizations.of(context)
|
||||||
.customerInfo
|
.customerInfo
|
||||||
.replaceFirst(RegExp('\n'), ''),
|
.replaceFirst(RegExp('\n'), ''),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
),
|
body: SafeArea(
|
||||||
body: Stack(
|
child: Stack(
|
||||||
children: [
|
children: [
|
||||||
SingleChildScrollView(
|
SingleChildScrollView(
|
||||||
physics: const AlwaysScrollableScrollPhysics(),
|
physics: const AlwaysScrollableScrollPhysics(),
|
||||||
child: Padding(
|
child: Padding(
|
||||||
padding: const EdgeInsets.all(16.0),
|
padding: const EdgeInsets.all(16.0),
|
||||||
child: SafeArea(
|
|
||||||
child: Center(
|
|
||||||
child: Column(
|
child: Column(
|
||||||
children: [
|
children: [
|
||||||
const SizedBox(height: 30),
|
Card(
|
||||||
CircleAvatar(
|
elevation: 0,
|
||||||
radius: 50,
|
shape: RoundedRectangleBorder(
|
||||||
child: SvgPicture.asset(
|
borderRadius: BorderRadius.circular(12),
|
||||||
'assets/images/avatar_male.svg',
|
side: BorderSide(
|
||||||
width: 150,
|
color: theme.colorScheme.outline.withOpacity(0.2),
|
||||||
height: 150,
|
width: 1,
|
||||||
fit: BoxFit.cover,
|
),
|
||||||
),
|
),
|
||||||
),
|
child: Padding(
|
||||||
Padding(
|
padding: const EdgeInsets.all(16.0),
|
||||||
padding: const EdgeInsets.only(top: 10.0),
|
child: Row(
|
||||||
child: Text(
|
children: [
|
||||||
user.name ?? '',
|
const SizedBox(
|
||||||
style: TextStyle(
|
width: 56,
|
||||||
fontSize: 20,
|
height: 56,
|
||||||
color: theme.colorScheme.onSurface,
|
child: CircleAvatar(
|
||||||
fontWeight: FontWeight.w500,
|
radius: 50,
|
||||||
|
child: Icon(
|
||||||
|
Symbols.person,
|
||||||
|
size: 56,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(width: 12),
|
||||||
|
// Name + mobile
|
||||||
|
Expanded(
|
||||||
|
child: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
// If you want to show the user's name instead, replace below.
|
||||||
|
user.name ?? '',
|
||||||
|
style:
|
||||||
|
theme.textTheme.titleLarge?.copyWith(
|
||||||
|
fontWeight: FontWeight.w600,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 4),
|
||||||
|
Text(
|
||||||
|
user.cifNumber ?? '',
|
||||||
|
style:
|
||||||
|
theme.textTheme.bodyMedium?.copyWith(
|
||||||
|
color: theme.colorScheme.onSurface
|
||||||
|
.withOpacity(0.7),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
Text(
|
const SizedBox(height: 16),
|
||||||
'${AppLocalizations.of(context).cif}: ${user.cifNumber ?? 'N/A'}',
|
// Toggle Buttons for Personal Info and KYC
|
||||||
style: TextStyle(
|
SizedBox(
|
||||||
fontSize: 16,
|
width: double.infinity,
|
||||||
color: theme.colorScheme.onSurfaceVariant),
|
child: CupertinoSlidingSegmentedControl<int>(
|
||||||
|
groupValue: _selectedCard,
|
||||||
|
thumbColor: Theme.of(context)
|
||||||
|
.colorScheme
|
||||||
|
.onPrimary, // Set selected switch color to theme primary color
|
||||||
|
onValueChanged: (int? newValue) {
|
||||||
|
if (newValue != null) {
|
||||||
|
setState(() {
|
||||||
|
_selectedCard = newValue;
|
||||||
|
});
|
||||||
|
}
|
||||||
|
},
|
||||||
|
children: {
|
||||||
|
0: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: 20, vertical: 10),
|
||||||
|
child: Text(
|
||||||
|
AppLocalizations.of(context).personaldetails),
|
||||||
|
),
|
||||||
|
1: Padding(
|
||||||
|
padding: const EdgeInsets.symmetric(
|
||||||
|
horizontal: 20, vertical: 10),
|
||||||
|
child:
|
||||||
|
Text(AppLocalizations.of(context).kycdetails),
|
||||||
|
),
|
||||||
|
},
|
||||||
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 30),
|
const SizedBox(height: 16),
|
||||||
InfoField(
|
// Card that shows content based on the toggle
|
||||||
label: AppLocalizations.of(context).activeAccounts,
|
Card(
|
||||||
value: user.activeAccounts?.toString() ?? 'N/A',
|
elevation: 0,
|
||||||
|
shape: RoundedRectangleBorder(
|
||||||
|
borderRadius: BorderRadius.circular(12),
|
||||||
|
side: BorderSide(
|
||||||
|
color: theme.colorScheme.outline.withOpacity(0.2),
|
||||||
|
width: 1,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
child: Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: AnimatedSwitcher(
|
||||||
|
duration: const Duration(milliseconds: 300),
|
||||||
|
child: _selectedCard == 0
|
||||||
|
? _buildPersonalInfo(theme)
|
||||||
|
: _buildKycDetails(theme),
|
||||||
|
),
|
||||||
|
),
|
||||||
),
|
),
|
||||||
InfoField(
|
|
||||||
label: AppLocalizations.of(context).mobileNumber,
|
|
||||||
value: user.mobileNo ?? 'N/A',
|
|
||||||
),
|
|
||||||
InfoField(
|
|
||||||
label: AppLocalizations.of(context).dateOfBirth,
|
|
||||||
value: (user.dateOfBirth != null &&
|
|
||||||
user.dateOfBirth!.length == 8)
|
|
||||||
? '${user.dateOfBirth!.substring(0, 2)}-${user.dateOfBirth!.substring(2, 4)}-${user.dateOfBirth!.substring(4, 8)}'
|
|
||||||
: 'N/A',
|
|
||||||
), // Replace with DOB if available
|
|
||||||
InfoField(
|
|
||||||
label: AppLocalizations.of(context).branchCode,
|
|
||||||
value: user.branchId ?? 'N/A',
|
|
||||||
),
|
|
||||||
InfoField(
|
|
||||||
label: AppLocalizations.of(context).address,
|
|
||||||
value: user.address ?? 'N/A',
|
|
||||||
), // Replace with Aadhar if available
|
|
||||||
InfoField(
|
|
||||||
label: AppLocalizations.of(context).primaryId,
|
|
||||||
value: _maskPrimaryId(user.primaryId),
|
|
||||||
), // Replace with PAN if available
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
IgnorePointer(
|
||||||
),
|
child: Center(
|
||||||
IgnorePointer(
|
child: Opacity(
|
||||||
child: Center(
|
opacity: 0.07, // Reduced opacity
|
||||||
child: Opacity(
|
child: ClipOval(
|
||||||
opacity: 0.07, // Reduced opacity
|
child: Image.asset(
|
||||||
child: ClipOval(
|
'assets/images/logo.png',
|
||||||
child: Image.asset(
|
width: 200, // Adjust size as needed
|
||||||
'assets/images/logo.png',
|
height: 200, // Adjust size as needed
|
||||||
width: 200, // Adjust size as needed
|
),
|
||||||
height: 200, // Adjust size as needed
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
],
|
||||||
),
|
),
|
||||||
],
|
));
|
||||||
),
|
}
|
||||||
|
|
||||||
|
Widget _buildPersonalInfo(ThemeData theme) {
|
||||||
|
return Column(
|
||||||
|
key: const ValueKey('personal_info'),
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
AppLocalizations.of(context).personaldetails,
|
||||||
|
style: theme.textTheme.titleMedium?.copyWith(
|
||||||
|
fontWeight: FontWeight.w600,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
InfoField(
|
||||||
|
label: AppLocalizations.of(context).activeAccounts,
|
||||||
|
value: user.activeAccounts?.toString() ?? 'N/A',
|
||||||
|
),
|
||||||
|
InfoField(
|
||||||
|
label: AppLocalizations.of(context).mobileNumber,
|
||||||
|
value: user.mobileNo ?? 'N/A',
|
||||||
|
),
|
||||||
|
InfoField(
|
||||||
|
label: AppLocalizations.of(context).dateOfBirth,
|
||||||
|
value: (user.dateOfBirth != null && user.dateOfBirth!.length == 8)
|
||||||
|
? '${user.dateOfBirth!.substring(0, 2)}-${user.dateOfBirth!.substring(2, 4)}-${user.dateOfBirth!.substring(4, 8)}'
|
||||||
|
: 'N/A',
|
||||||
|
),
|
||||||
|
InfoField(
|
||||||
|
label: AppLocalizations.of(context).branchCode,
|
||||||
|
value: user.branchId ?? 'N/A',
|
||||||
|
),
|
||||||
|
InfoField(
|
||||||
|
label: AppLocalizations.of(context).address,
|
||||||
|
value: user.address ?? 'N/A',
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
Widget _buildKycDetails(ThemeData theme) {
|
||||||
|
return Column(
|
||||||
|
key: const ValueKey('kyc_details'),
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
AppLocalizations.of(context).kycdetails,
|
||||||
|
style: theme.textTheme.titleMedium?.copyWith(
|
||||||
|
fontWeight: FontWeight.w600,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 16),
|
||||||
|
InfoField(
|
||||||
|
label: AppLocalizations.of(context).primaryId,
|
||||||
|
value: _maskPrimaryId(user.primaryId),
|
||||||
|
),
|
||||||
|
],
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -141,16 +250,16 @@ class InfoField extends StatelessWidget {
|
|||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
label,
|
label,
|
||||||
style: TextStyle(
|
style: theme.textTheme.bodySmall?.copyWith(
|
||||||
fontSize: 15,
|
color: theme.colorScheme.onSurface.withOpacity(0.6),
|
||||||
fontWeight: FontWeight.w500,
|
|
||||||
color: theme.colorScheme.onSurfaceVariant,
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 3),
|
const SizedBox(height: 4),
|
||||||
Text(
|
Text(
|
||||||
value,
|
value.isEmpty ? 'N/A' : value,
|
||||||
style: TextStyle(fontSize: 16, color: theme.colorScheme.onSurface),
|
style: theme.textTheme.titleMedium?.copyWith(
|
||||||
|
fontWeight: FontWeight.w600,
|
||||||
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:flutter_bloc/flutter_bloc.dart';
|
import 'package:flutter_bloc/flutter_bloc.dart';
|
||||||
import 'package:flutter_svg/svg.dart';
|
|
||||||
import 'package:kmobile/data/models/user.dart';
|
import 'package:kmobile/data/models/user.dart';
|
||||||
import 'package:kmobile/di/injection.dart';
|
import 'package:kmobile/di/injection.dart';
|
||||||
import 'package:kmobile/features/accounts/screens/account_info_screen.dart';
|
import 'package:kmobile/features/accounts/screens/account_info_screen.dart';
|
||||||
@@ -8,8 +8,8 @@ import 'package:kmobile/features/accounts/screens/account_statement_screen.dart'
|
|||||||
import 'package:kmobile/features/accounts/screens/all_accounts_screen.dart';
|
import 'package:kmobile/features/accounts/screens/all_accounts_screen.dart';
|
||||||
import 'package:kmobile/features/auth/controllers/auth_cubit.dart';
|
import 'package:kmobile/features/auth/controllers/auth_cubit.dart';
|
||||||
import 'package:kmobile/features/auth/controllers/auth_state.dart';
|
import 'package:kmobile/features/auth/controllers/auth_state.dart';
|
||||||
import 'package:kmobile/features/customer_info/screens/customer_info_screen.dart';
|
|
||||||
import 'package:kmobile/features/cheque/screens/cheque_management_screen.dart';
|
import 'package:kmobile/features/cheque/screens/cheque_management_screen.dart';
|
||||||
|
import 'package:kmobile/features/customer_info/screens/customer_info_screen.dart';
|
||||||
import 'package:kmobile/features/beneficiaries/screens/manage_beneficiaries_screen.dart';
|
import 'package:kmobile/features/beneficiaries/screens/manage_beneficiaries_screen.dart';
|
||||||
import 'package:kmobile/features/enquiry/screens/enquiry_screen.dart';
|
import 'package:kmobile/features/enquiry/screens/enquiry_screen.dart';
|
||||||
import 'package:kmobile/features/fund_transfer/screens/fund_transfer_screen.dart';
|
import 'package:kmobile/features/fund_transfer/screens/fund_transfer_screen.dart';
|
||||||
@@ -65,68 +65,24 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildViewAllTab(List<User> users) {
|
|
||||||
return GestureDetector(
|
|
||||||
onTap: () {
|
|
||||||
Navigator.push(
|
|
||||||
context,
|
|
||||||
MaterialPageRoute(
|
|
||||||
builder: (context) => AllAccountsScreen(users: users),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
child: Container(
|
|
||||||
width: 40, // Small width for the tab
|
|
||||||
height: 160,
|
|
||||||
decoration: BoxDecoration(
|
|
||||||
color: Theme.of(context).colorScheme.surfaceVariant,
|
|
||||||
),
|
|
||||||
child: const Column(
|
|
||||||
mainAxisAlignment: MainAxisAlignment.center,
|
|
||||||
children: [
|
|
||||||
Text(
|
|
||||||
"View",
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 12,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Text(
|
|
||||||
"All",
|
|
||||||
style: TextStyle(
|
|
||||||
fontWeight: FontWeight.bold,
|
|
||||||
fontSize: 12,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
Icon(
|
|
||||||
Icons.arrow_forward_ios,
|
|
||||||
size: 16,
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildAccountCard(User user, bool isSelected) {
|
Widget _buildAccountCard(User user, bool isSelected) {
|
||||||
final theme = Theme.of(context);
|
final theme = Theme.of(context);
|
||||||
final bool isCardVisible = _visibilityMap[user.accountNo] ?? false;
|
final bool isCardVisible = _visibilityMap[user.accountNo] ?? false;
|
||||||
// Animated scale for the selected card
|
// Animated scale for the selected card
|
||||||
final scale = isSelected ? 1.0 : 0.85;
|
final scale = isSelected ? 1.02 : 0.9;
|
||||||
return AnimatedScale(
|
return Padding(
|
||||||
duration: const Duration(milliseconds: 200),
|
padding: const EdgeInsets.symmetric(horizontal: 3.0),
|
||||||
scale: scale,
|
child: AnimatedScale(
|
||||||
child: Transform.translate(
|
duration: const Duration(milliseconds: 200),
|
||||||
offset: isSelected ? const Offset(10.0, 0.0) : Offset.zero,
|
scale: scale,
|
||||||
child: Container(
|
child: Container(
|
||||||
margin: const EdgeInsets.symmetric(horizontal: 2),
|
|
||||||
padding: const EdgeInsets.symmetric(
|
padding: const EdgeInsets.symmetric(
|
||||||
horizontal: 18,
|
horizontal: 18,
|
||||||
vertical: 10,
|
vertical: 10,
|
||||||
),
|
),
|
||||||
decoration: BoxDecoration(
|
decoration: BoxDecoration(
|
||||||
color: const Color(0xFF01A04C),
|
color: const Color(0xFF01A04C),
|
||||||
borderRadius: BorderRadius.circular(16),
|
borderRadius: BorderRadius.circular(20),
|
||||||
),
|
),
|
||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
@@ -388,6 +344,19 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
@override
|
@override
|
||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
final theme = Theme.of(context);
|
final theme = Theme.of(context);
|
||||||
|
final authState = context.read<AuthCubit>().state;
|
||||||
|
String mobileNumberToPass = '';
|
||||||
|
String customerNo = '';
|
||||||
|
String customerName = '';
|
||||||
|
if (authState is Authenticated) {
|
||||||
|
if (selectedAccountIndex >= 0 &&
|
||||||
|
selectedAccountIndex < authState.users.length) {
|
||||||
|
mobileNumberToPass =
|
||||||
|
authState.users[selectedAccountIndex].mobileNo ?? '';
|
||||||
|
customerNo = authState.users[selectedAccountIndex].cifNumber ?? '';
|
||||||
|
customerName = authState.users[selectedAccountIndex].name ?? '';
|
||||||
|
}
|
||||||
|
}
|
||||||
return BlocListener<AuthCubit, AuthState>(
|
return BlocListener<AuthCubit, AuthState>(
|
||||||
listener: (context, state) async {
|
listener: (context, state) async {
|
||||||
if (state is Authenticated && !_biometricPromptShown) {
|
if (state is Authenticated && !_biometricPromptShown) {
|
||||||
@@ -402,59 +371,67 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
child: Scaffold(
|
child: Scaffold(
|
||||||
backgroundColor: theme.scaffoldBackgroundColor,
|
backgroundColor: theme.scaffoldBackgroundColor,
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
backgroundColor: theme.scaffoldBackgroundColor,
|
|
||||||
leading: Padding(
|
leading: Padding(
|
||||||
padding: const EdgeInsets.only(left: 10.0),
|
padding: const EdgeInsets.only(left: 10.0),
|
||||||
child: InkWell(
|
child: Material(
|
||||||
borderRadius: BorderRadius.circular(20),
|
elevation: 4.0,
|
||||||
onTap: () {
|
clipBehavior: Clip.antiAlias,
|
||||||
final authState = context.read<AuthCubit>().state;
|
shape: RoundedRectangleBorder(
|
||||||
String mobileNumberToPass = '';
|
side: const BorderSide(color: Colors.white, width: 1.5),
|
||||||
String customerNo = '';
|
borderRadius: BorderRadius.circular(12.0),
|
||||||
String customerName = '';
|
),
|
||||||
if (authState is Authenticated) {
|
child: Container(
|
||||||
if (selectedAccountIndex >= 0 &&
|
width: 40,
|
||||||
selectedAccountIndex < authState.users.length) {
|
height: 40,
|
||||||
mobileNumberToPass =
|
decoration: const BoxDecoration(
|
||||||
authState.users[selectedAccountIndex].mobileNo ?? '';
|
color: Colors.white,
|
||||||
customerNo =
|
),
|
||||||
authState.users[selectedAccountIndex].cifNumber ?? '';
|
child: Image.asset(
|
||||||
customerName =
|
'assets/images/logo.png',
|
||||||
authState.users[selectedAccountIndex].name ?? '';
|
fit: BoxFit.fill,
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Navigator.push(
|
|
||||||
context,
|
|
||||||
MaterialPageRoute(
|
|
||||||
builder: (context) => ProfileScreen(
|
|
||||||
mobileNumber: mobileNumberToPass,
|
|
||||||
customerNo: customerNo,
|
|
||||||
customerName: customerName),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
child: CircleAvatar(
|
|
||||||
backgroundColor: Colors.grey[200],
|
|
||||||
radius: 20,
|
|
||||||
child: SvgPicture.asset(
|
|
||||||
'assets/images/avatar_male.svg',
|
|
||||||
width: 40,
|
|
||||||
height: 40,
|
|
||||||
fit: BoxFit.cover,
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
title: Text(
|
title: Text(
|
||||||
AppLocalizations.of(context).kccbMobile,
|
AppLocalizations.of(context).kccBankFull.replaceAll('-', '\u2011'),
|
||||||
textAlign: TextAlign.left,
|
textAlign: TextAlign.center,
|
||||||
|
softWrap: true, // Explicitly allow wrapping
|
||||||
|
maxLines: 2, // Allow text to wrap to a maximum of 2 lines
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
color: theme.colorScheme.primary,
|
color: theme.colorScheme.onPrimary,
|
||||||
fontWeight: FontWeight.w700,
|
fontWeight: FontWeight.w700,
|
||||||
|
fontSize: 20,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
centerTitle: true,
|
// Removed centerTitle: true to give more space for text wrapping
|
||||||
|
|
||||||
|
actions: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(right: 10.0),
|
||||||
|
child: InkWell(
|
||||||
|
borderRadius: BorderRadius.circular(20),
|
||||||
|
onTap: () {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => ProfileScreen(
|
||||||
|
mobileNumber: mobileNumberToPass,
|
||||||
|
customerNo: customerNo,
|
||||||
|
customerName: customerName),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
child: const CircleAvatar(
|
||||||
|
radius: 21,
|
||||||
|
child: Icon(
|
||||||
|
Symbols.person,
|
||||||
|
size: 30,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
),
|
),
|
||||||
body: BlocBuilder<AuthCubit, AuthState>(
|
body: BlocBuilder<AuthCubit, AuthState>(
|
||||||
builder: (context, state) {
|
builder: (context, state) {
|
||||||
@@ -467,16 +444,16 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
final accountType = currAccount.accountType?.toLowerCase();
|
final accountType = currAccount.accountType?.toLowerCase();
|
||||||
final isPaymentDisabled = accountType != 'sa' &&
|
final isPaymentDisabled = accountType != 'sa' &&
|
||||||
accountType != 'sb' &&
|
accountType != 'sb' &&
|
||||||
accountType != 'ca';
|
accountType != 'ca' &&
|
||||||
|
accountType != 'cc';
|
||||||
// first‐time load
|
// first‐time load
|
||||||
if (!_txInitialized) {
|
if (!_txInitialized) {
|
||||||
_txInitialized = true;
|
_txInitialized = true;
|
||||||
WidgetsBinding.instance.addPostFrameCallback((_) {
|
WidgetsBinding.instance.addPostFrameCallback((_) {});
|
||||||
});
|
|
||||||
}
|
}
|
||||||
_pageController ??= PageController(
|
_pageController ??= PageController(
|
||||||
initialPage: selectedAccountIndex,
|
initialPage: selectedAccountIndex,
|
||||||
viewportFraction: 0.80,
|
viewportFraction: 0.75,
|
||||||
);
|
);
|
||||||
final firstName = getProcessedFirstName(currAccount.name);
|
final firstName = getProcessedFirstName(currAccount.name);
|
||||||
|
|
||||||
@@ -486,8 +463,9 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
child: Column(
|
child: Column(
|
||||||
crossAxisAlignment: CrossAxisAlignment.start,
|
crossAxisAlignment: CrossAxisAlignment.start,
|
||||||
children: [
|
children: [
|
||||||
|
const SizedBox(height: 16), // Added spacing
|
||||||
Padding(
|
Padding(
|
||||||
padding: const EdgeInsets.only(left: 8.0),
|
padding: const EdgeInsets.only(left: 4.0),
|
||||||
child: Text(
|
child: Text(
|
||||||
"${AppLocalizations.of(context).hi} $firstName!",
|
"${AppLocalizations.of(context).hi} $firstName!",
|
||||||
style: GoogleFonts.baumans().copyWith(
|
style: GoogleFonts.baumans().copyWith(
|
||||||
@@ -500,59 +478,60 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
const SizedBox(height: 16),
|
const SizedBox(height: 16),
|
||||||
|
|
||||||
// Account Info Cards
|
// Account Info Cards
|
||||||
ClipRect(
|
SizedBox(
|
||||||
child: SizedBox(
|
height: 160,
|
||||||
// This SizedBox defines the height for the Stack
|
child: PageView.builder(
|
||||||
height: 160,
|
clipBehavior: Clip.none,
|
||||||
child: Stack(
|
controller: _pageController,
|
||||||
children: [
|
itemCount:
|
||||||
// PageView part, painted underneath
|
users.length, // Keep this to show adjacent cards
|
||||||
Padding(
|
|
||||||
padding: const EdgeInsets.only(
|
|
||||||
left: 48.0), // Space for tab (40) + gap (8)
|
|
||||||
child: SizedBox(
|
|
||||||
// Keep SizedBox for PageView height
|
|
||||||
height: 160,
|
|
||||||
child: PageView.builder(
|
|
||||||
controller: _pageController,
|
|
||||||
itemCount: users.length,
|
|
||||||
clipBehavior: Clip
|
|
||||||
.none, // Keep this to show adjacent cards
|
|
||||||
padEnds: false,
|
|
||||||
onPageChanged: (int newIndex) async {
|
|
||||||
if (newIndex == selectedAccountIndex)
|
|
||||||
return;
|
|
||||||
|
|
||||||
// Hide the balance of the old card when scrolling away
|
onPageChanged: (int newIndex) async {
|
||||||
final oldAccountNo =
|
if (newIndex == selectedAccountIndex) return;
|
||||||
users[selectedAccountIndex].accountNo;
|
|
||||||
if (oldAccountNo != null) {
|
|
||||||
_visibilityMap[oldAccountNo] = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
setState(() {
|
// Hide the balance of the old card when scrolling away
|
||||||
selectedAccountIndex = newIndex;
|
final oldAccountNo =
|
||||||
});
|
users[selectedAccountIndex].accountNo;
|
||||||
},
|
if (oldAccountNo != null) {
|
||||||
itemBuilder: (context, index) {
|
_visibilityMap[oldAccountNo] = false;
|
||||||
final user = users[index];
|
}
|
||||||
final isSelected =
|
|
||||||
index == selectedAccountIndex;
|
setState(() {
|
||||||
return _buildAccountCard(
|
selectedAccountIndex = newIndex;
|
||||||
user, isSelected);
|
});
|
||||||
},
|
},
|
||||||
),
|
itemBuilder: (context, index) {
|
||||||
),
|
final user = users[index];
|
||||||
),
|
final isSelected = index == selectedAccountIndex;
|
||||||
// View All tab part, painted on top
|
return _buildAccountCard(user, isSelected);
|
||||||
Align(
|
},
|
||||||
alignment: Alignment.centerLeft,
|
|
||||||
child: _buildViewAllTab(users),
|
|
||||||
),
|
|
||||||
],
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
const SizedBox(height: 8),
|
||||||
|
Row(
|
||||||
|
mainAxisAlignment: MainAxisAlignment.end,
|
||||||
|
children: [
|
||||||
|
GestureDetector(
|
||||||
|
onTap: () {
|
||||||
|
Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) =>
|
||||||
|
AllAccountsScreen(users: users),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
child: Text(
|
||||||
|
AppLocalizations.of(context).viewall,
|
||||||
|
style: TextStyle(
|
||||||
|
fontSize: 14,
|
||||||
|
fontWeight: FontWeight.bold,
|
||||||
|
color: theme.colorScheme.primary,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
const SizedBox(height: 18),
|
const SizedBox(height: 18),
|
||||||
Text(
|
Text(
|
||||||
AppLocalizations.of(context).quickLinks,
|
AppLocalizations.of(context).quickLinks,
|
||||||
@@ -639,8 +618,8 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
selectedIndex: selectedAccountIndex,
|
selectedIndex: selectedAccountIndex,
|
||||||
)));
|
)));
|
||||||
}),
|
}),
|
||||||
_buildQuickLink(Icons.location_pin, "Branch Locator",
|
_buildQuickLink(Icons.location_pin,
|
||||||
() {
|
AppLocalizations.of(context).branchlocator, () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
@@ -666,18 +645,19 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
const EnquiryScreen()));
|
const EnquiryScreen()));
|
||||||
}),
|
}),
|
||||||
_buildQuickLink(
|
_buildQuickLink(
|
||||||
Symbols.request_quote,
|
Symbols.checkbook,
|
||||||
AppLocalizations.of(context).chequeManagement,
|
AppLocalizations.of(context).chequeManagement,
|
||||||
() {
|
() {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
builder: (context) =>
|
builder: (context) => ChequeManagementScreen(
|
||||||
const ChequeManagementScreen(),
|
users: users,
|
||||||
|
selectedIndex: selectedAccountIndex),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
disable: true,
|
disable: isPaymentDisabled,
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
@@ -695,32 +675,6 @@ class _DashboardScreenState extends State<DashboardScreen>
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
List<Widget> _buildTransactionShimmer() {
|
|
||||||
final theme = Theme.of(context);
|
|
||||||
return List.generate(3, (i) {
|
|
||||||
return ListTile(
|
|
||||||
leading: Shimmer.fromColors(
|
|
||||||
baseColor: Colors.grey[300]!,
|
|
||||||
highlightColor: Colors.grey[100]!,
|
|
||||||
child: CircleAvatar(
|
|
||||||
radius: 12, backgroundColor: theme.scaffoldBackgroundColor),
|
|
||||||
),
|
|
||||||
title: Shimmer.fromColors(
|
|
||||||
baseColor: Colors.grey[300]!,
|
|
||||||
highlightColor: Colors.grey[100]!,
|
|
||||||
child: Container(
|
|
||||||
height: 10, width: 100, color: theme.scaffoldBackgroundColor),
|
|
||||||
),
|
|
||||||
subtitle: Shimmer.fromColors(
|
|
||||||
baseColor: Colors.grey[300]!,
|
|
||||||
highlightColor: Colors.grey[100]!,
|
|
||||||
child: Container(
|
|
||||||
height: 8, width: 60, color: theme.scaffoldBackgroundColor),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
Widget _buildQuickLink(
|
Widget _buildQuickLink(
|
||||||
IconData icon,
|
IconData icon,
|
||||||
String label,
|
String label,
|
||||||
|
|||||||
@@ -129,7 +129,7 @@ class _EnquiryScreen extends State<EnquiryScreen> {
|
|||||||
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
mainAxisAlignment: MainAxisAlignment.spaceBetween,
|
||||||
children: [
|
children: [
|
||||||
Text(
|
Text(
|
||||||
"Complaint Form",
|
AppLocalizations.of(context).complaintFormTitle,
|
||||||
style: TextStyle(
|
style: TextStyle(
|
||||||
fontSize: 15,
|
fontSize: 15,
|
||||||
color: Theme.of(context).colorScheme.primary,
|
color: Theme.of(context).colorScheme.primary,
|
||||||
|
|||||||
@@ -496,7 +496,7 @@ class _FundTransferAmountScreenState extends State<FundTransferAmountScreen> {
|
|||||||
Text(AppLocalizations.of(context).fetchingDailyLimit),
|
Text(AppLocalizations.of(context).fetchingDailyLimit),
|
||||||
if (!_isLoadingLimit && _limit != null)
|
if (!_isLoadingLimit && _limit != null)
|
||||||
Text(
|
Text(
|
||||||
'Remaining Daily Limit: ${_formatCurrency.format(_limit!.dailyLimit - _limit!.usedLimit)}',
|
'${AppLocalizations.of(context).remainingDailyLimit} ${_formatCurrency.format(_limit!.dailyLimit - _limit!.usedLimit)}',
|
||||||
style: Theme.of(context).textTheme.bodySmall,
|
style: Theme.of(context).textTheme.bodySmall,
|
||||||
),
|
),
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
|
|||||||
@@ -198,7 +198,8 @@ class _FundTransferBeneficiaryScreenState
|
|||||||
child: TextField(
|
child: TextField(
|
||||||
controller: _searchController,
|
controller: _searchController,
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
hintText: "Search by name or account number",
|
hintText:
|
||||||
|
AppLocalizations.of(context).searchByNameOrAccountHint,
|
||||||
prefixIcon: const Icon(Icons.search),
|
prefixIcon: const Icon(Icons.search),
|
||||||
border: OutlineInputBorder(
|
border: OutlineInputBorder(
|
||||||
borderRadius: BorderRadius.circular(12),
|
borderRadius: BorderRadius.circular(12),
|
||||||
|
|||||||
@@ -42,7 +42,9 @@ class FundTransferScreen extends StatelessWidget {
|
|||||||
Expanded(
|
Expanded(
|
||||||
child: FundTransferManagementTile(
|
child: FundTransferManagementTile(
|
||||||
icon: Symbols.person,
|
icon: Symbols.person,
|
||||||
label: "Self Pay",
|
label: AppLocalizations.of(context).selfPay,
|
||||||
|
subtitle:
|
||||||
|
AppLocalizations.of(context).ftselfpaysubtitle,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
@@ -64,6 +66,7 @@ class FundTransferScreen extends StatelessWidget {
|
|||||||
child: FundTransferManagementTile(
|
child: FundTransferManagementTile(
|
||||||
icon: Symbols.input_circle,
|
icon: Symbols.input_circle,
|
||||||
label: AppLocalizations.of(context).ownBank,
|
label: AppLocalizations.of(context).ownBank,
|
||||||
|
subtitle: AppLocalizations.of(context).ftownsubtitle,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
@@ -84,6 +87,8 @@ class FundTransferScreen extends StatelessWidget {
|
|||||||
child: FundTransferManagementTile(
|
child: FundTransferManagementTile(
|
||||||
icon: Symbols.output_circle,
|
icon: Symbols.output_circle,
|
||||||
label: AppLocalizations.of(context).outsideBank,
|
label: AppLocalizations.of(context).outsideBank,
|
||||||
|
subtitle:
|
||||||
|
AppLocalizations.of(context).ftoutsidesubtitle,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
@@ -127,6 +132,7 @@ class FundTransferScreen extends StatelessWidget {
|
|||||||
class FundTransferManagementTile extends StatelessWidget {
|
class FundTransferManagementTile extends StatelessWidget {
|
||||||
final IconData icon;
|
final IconData icon;
|
||||||
final String label;
|
final String label;
|
||||||
|
final String? subtitle;
|
||||||
final VoidCallback onTap;
|
final VoidCallback onTap;
|
||||||
final bool disable;
|
final bool disable;
|
||||||
|
|
||||||
@@ -134,6 +140,7 @@ class FundTransferManagementTile extends StatelessWidget {
|
|||||||
super.key,
|
super.key,
|
||||||
required this.icon,
|
required this.icon,
|
||||||
required this.label,
|
required this.label,
|
||||||
|
this.subtitle,
|
||||||
required this.onTap,
|
required this.onTap,
|
||||||
this.disable = false,
|
this.disable = false,
|
||||||
});
|
});
|
||||||
@@ -174,6 +181,19 @@ class FundTransferManagementTile extends StatelessWidget {
|
|||||||
: theme.colorScheme.onSurface,
|
: theme.colorScheme.onSurface,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
if (subtitle != null)
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 8.0),
|
||||||
|
child: Text(
|
||||||
|
subtitle!,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: theme.textTheme.bodyMedium?.copyWith(
|
||||||
|
color: disable
|
||||||
|
? theme.disabledColor
|
||||||
|
: theme.colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:kmobile/data/models/user.dart';
|
import 'package:kmobile/data/models/user.dart';
|
||||||
import 'package:kmobile/features/fund_transfer/screens/fund_transfer_self_amount_screen.dart';
|
import 'package:kmobile/features/fund_transfer/screens/fund_transfer_self_amount_screen.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
import 'package:kmobile/widgets/bank_logos.dart';
|
import 'package:kmobile/widgets/bank_logos.dart';
|
||||||
|
|
||||||
class FundTransferSelfAccountsScreen extends StatelessWidget {
|
class FundTransferSelfAccountsScreen extends StatelessWidget {
|
||||||
@@ -43,7 +44,7 @@ class FundTransferSelfAccountsScreen extends StatelessWidget {
|
|||||||
|
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: const Text("Select Account"),
|
title: Text(AppLocalizations.of(context).selectAccount),
|
||||||
),
|
),
|
||||||
body: Stack(
|
body: Stack(
|
||||||
children: [
|
children: [
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import 'package:kmobile/data/models/user.dart';
|
|||||||
import 'package:kmobile/di/injection.dart';
|
import 'package:kmobile/di/injection.dart';
|
||||||
import 'package:kmobile/features/fund_transfer/screens/payment_animation.dart';
|
import 'package:kmobile/features/fund_transfer/screens/payment_animation.dart';
|
||||||
import 'package:kmobile/features/fund_transfer/screens/transaction_pin_screen.dart';
|
import 'package:kmobile/features/fund_transfer/screens/transaction_pin_screen.dart';
|
||||||
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
import 'package:kmobile/widgets/bank_logos.dart';
|
import 'package:kmobile/widgets/bank_logos.dart';
|
||||||
|
|
||||||
class FundTransferSelfAmountScreen extends StatefulWidget {
|
class FundTransferSelfAmountScreen extends StatefulWidget {
|
||||||
@@ -134,7 +135,7 @@ class _FundTransferSelfAmountScreenState
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: const Text("Fund Transfer"),
|
title: Text(AppLocalizations.of(context).fundTransferTitle),
|
||||||
),
|
),
|
||||||
body: SafeArea(
|
body: SafeArea(
|
||||||
child: Stack(
|
child: Stack(
|
||||||
@@ -148,7 +149,7 @@ class _FundTransferSelfAmountScreenState
|
|||||||
children: [
|
children: [
|
||||||
// Debit Account (User)
|
// Debit Account (User)
|
||||||
Text(
|
Text(
|
||||||
"Debit From",
|
AppLocalizations.of(context).debitFromLabel,
|
||||||
style: Theme.of(context).textTheme.titleSmall,
|
style: Theme.of(context).textTheme.titleSmall,
|
||||||
),
|
),
|
||||||
Card(
|
Card(
|
||||||
@@ -168,7 +169,7 @@ class _FundTransferSelfAmountScreenState
|
|||||||
|
|
||||||
// Credit Account (Self)
|
// Credit Account (Self)
|
||||||
Text(
|
Text(
|
||||||
"Credited To",
|
AppLocalizations.of(context).creditedTo,
|
||||||
style: Theme.of(context).textTheme.titleSmall,
|
style: Theme.of(context).textTheme.titleSmall,
|
||||||
),
|
),
|
||||||
Card(
|
Card(
|
||||||
@@ -186,9 +187,10 @@ class _FundTransferSelfAmountScreenState
|
|||||||
// Remarks
|
// Remarks
|
||||||
TextFormField(
|
TextFormField(
|
||||||
controller: _remarksController,
|
controller: _remarksController,
|
||||||
decoration: const InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: "Remarks (Optional)",
|
labelText:
|
||||||
border: OutlineInputBorder(),
|
AppLocalizations.of(context).remarksOptionalHint,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 24),
|
const SizedBox(height: 24),
|
||||||
@@ -197,18 +199,18 @@ class _FundTransferSelfAmountScreenState
|
|||||||
TextFormField(
|
TextFormField(
|
||||||
controller: _amountController,
|
controller: _amountController,
|
||||||
keyboardType: TextInputType.number,
|
keyboardType: TextInputType.number,
|
||||||
decoration: const InputDecoration(
|
decoration: InputDecoration(
|
||||||
labelText: "Amount",
|
labelText: AppLocalizations.of(context).amountLabel,
|
||||||
border: OutlineInputBorder(),
|
border: const OutlineInputBorder(),
|
||||||
prefixIcon: Icon(Icons.currency_rupee),
|
prefixIcon: const Icon(Icons.currency_rupee),
|
||||||
),
|
),
|
||||||
validator: (value) {
|
validator: (value) {
|
||||||
if (value == null || value.isEmpty) {
|
if (value == null || value.isEmpty) {
|
||||||
return "Amount is required";
|
return AppLocalizations.of(context).amountRequired;
|
||||||
}
|
}
|
||||||
if (double.tryParse(value) == null ||
|
if (double.tryParse(value) == null ||
|
||||||
double.parse(value) <= 0) {
|
double.parse(value) <= 0) {
|
||||||
return "Please enter a valid amount";
|
return AppLocalizations.of(context).validAmountError;
|
||||||
}
|
}
|
||||||
return null;
|
return null;
|
||||||
},
|
},
|
||||||
@@ -216,10 +218,12 @@ class _FundTransferSelfAmountScreenState
|
|||||||
const SizedBox(height: 8),
|
const SizedBox(height: 8),
|
||||||
|
|
||||||
// Daily Limit Display
|
// Daily Limit Display
|
||||||
if (_isLoadingLimit) const Text('Fetching daily limit...'),
|
if (_isLoadingLimit)
|
||||||
|
Text(AppLocalizations.of(context)
|
||||||
|
.fetchingDailyLimitLoader),
|
||||||
if (!_isLoadingLimit && _limit != null)
|
if (!_isLoadingLimit && _limit != null)
|
||||||
Text(
|
Text(
|
||||||
'Remaining Daily Limit: ${_formatCurrency.format(_limit!.dailyLimit - _limit!.usedLimit)}',
|
'${AppLocalizations.of(context).remainingDailyLimit} ${_formatCurrency.format(_limit!.dailyLimit - _limit!.usedLimit)}',
|
||||||
style: Theme.of(context).textTheme.bodySmall,
|
style: Theme.of(context).textTheme.bodySmall,
|
||||||
),
|
),
|
||||||
const Spacer(),
|
const Spacer(),
|
||||||
@@ -232,7 +236,7 @@ class _FundTransferSelfAmountScreenState
|
|||||||
style: ElevatedButton.styleFrom(
|
style: ElevatedButton.styleFrom(
|
||||||
padding: const EdgeInsets.symmetric(vertical: 16),
|
padding: const EdgeInsets.symmetric(vertical: 16),
|
||||||
),
|
),
|
||||||
child: const Text("Proceed"),
|
child: Text(AppLocalizations.of(context).proceedButton),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 10),
|
const SizedBox(height: 10),
|
||||||
|
|||||||
120
lib/features/profile/change_limit_otp_screen.dart
Normal file
120
lib/features/profile/change_limit_otp_screen.dart
Normal file
@@ -0,0 +1,120 @@
|
|||||||
|
import 'package:flutter/material.dart';
|
||||||
|
import 'package:kmobile/api/services/limit_service.dart';
|
||||||
|
import '../../../di/injection.dart';
|
||||||
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
|
class ChangeLimitOTPScreen extends StatefulWidget {
|
||||||
|
final String newLimit;
|
||||||
|
final String mobileNumber;
|
||||||
|
|
||||||
|
// ignore: use_key_in_widget_constructors
|
||||||
|
const ChangeLimitOTPScreen({
|
||||||
|
required this.newLimit,
|
||||||
|
required this.mobileNumber,
|
||||||
|
});
|
||||||
|
|
||||||
|
@override
|
||||||
|
State<ChangeLimitOTPScreen> createState() => _ChangeLimitOTPScreenState();
|
||||||
|
}
|
||||||
|
|
||||||
|
class _ChangeLimitOTPScreenState extends State<ChangeLimitOTPScreen> {
|
||||||
|
bool _isLoading = true;
|
||||||
|
final TextEditingController otpController = TextEditingController();
|
||||||
|
|
||||||
|
@override
|
||||||
|
void initState() {
|
||||||
|
super.initState();
|
||||||
|
// Simulate OTP sending delay
|
||||||
|
Future.delayed(const Duration(seconds: 2), () {
|
||||||
|
setState(() {
|
||||||
|
_isLoading = false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
final limitService = getIt<LimitService>();
|
||||||
|
Future<void> _validateOTP() async {
|
||||||
|
try {
|
||||||
|
await limitService.validateOtp(
|
||||||
|
otp: otpController.text,
|
||||||
|
mobileNumber: widget.mobileNumber,
|
||||||
|
);
|
||||||
|
|
||||||
|
// If OTP is valid, then change the limit
|
||||||
|
limitService.editLimit(
|
||||||
|
double.parse(widget.newLimit),
|
||||||
|
);
|
||||||
|
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(const SnackBar(
|
||||||
|
content: Text("Limit has been Changed"),
|
||||||
|
));
|
||||||
|
|
||||||
|
// Navigate back to profile or login
|
||||||
|
Navigator.of(context).popUntil((route) => route.isFirst);
|
||||||
|
} catch (e) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(content: Text(AppLocalizations.of(context).invalidOtp)),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@override
|
||||||
|
Widget build(BuildContext context) {
|
||||||
|
return Scaffold(
|
||||||
|
appBar: AppBar(title: Text(AppLocalizations.of(context).otpVerification)),
|
||||||
|
body: Stack(
|
||||||
|
children: [
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.all(16.0),
|
||||||
|
child: _isLoading
|
||||||
|
? const Center(child: CircularProgressIndicator())
|
||||||
|
: Column(
|
||||||
|
crossAxisAlignment: CrossAxisAlignment.center,
|
||||||
|
children: [
|
||||||
|
Text(
|
||||||
|
AppLocalizations.of(context).otpSent,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: const TextStyle(fontSize: 16),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
TextFormField(
|
||||||
|
controller: otpController,
|
||||||
|
keyboardType: TextInputType.number,
|
||||||
|
decoration: InputDecoration(
|
||||||
|
labelText: AppLocalizations.of(context).enterOTP,
|
||||||
|
border: const OutlineInputBorder(),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
const SizedBox(height: 24),
|
||||||
|
SizedBox(
|
||||||
|
width: double.infinity,
|
||||||
|
child: ElevatedButton(
|
||||||
|
onPressed: _validateOTP,
|
||||||
|
style: ElevatedButton.styleFrom(
|
||||||
|
padding: const EdgeInsets.symmetric(vertical: 16),
|
||||||
|
),
|
||||||
|
child: Text(AppLocalizations.of(context).validateOTP),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
),
|
||||||
|
IgnorePointer(
|
||||||
|
child: Center(
|
||||||
|
child: Opacity(
|
||||||
|
opacity: 0.07, // Reduced opacity
|
||||||
|
child: ClipOval(
|
||||||
|
child: Image.asset(
|
||||||
|
'assets/images/logo.png',
|
||||||
|
width: 200, // Adjust size as needed
|
||||||
|
height: 200, // Adjust size as needed
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
|
],
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -2,11 +2,13 @@ import 'package:flutter/material.dart';
|
|||||||
import 'package:flutter/services.dart';
|
import 'package:flutter/services.dart';
|
||||||
import 'package:kmobile/api/services/limit_service.dart';
|
import 'package:kmobile/api/services/limit_service.dart';
|
||||||
import 'package:kmobile/di/injection.dart';
|
import 'package:kmobile/di/injection.dart';
|
||||||
|
import 'package:kmobile/features/profile/change_limit_otp_screen.dart';
|
||||||
import 'package:kmobile/l10n/app_localizations.dart';
|
import 'package:kmobile/l10n/app_localizations.dart';
|
||||||
import 'package:intl/intl.dart';
|
import 'package:intl/intl.dart';
|
||||||
|
|
||||||
class DailyLimitScreen extends StatefulWidget {
|
class DailyLimitScreen extends StatefulWidget {
|
||||||
const DailyLimitScreen({super.key});
|
final String mobileNumber;
|
||||||
|
const DailyLimitScreen({super.key, required this.mobileNumber});
|
||||||
@override
|
@override
|
||||||
State<DailyLimitScreen> createState() => _DailyLimitScreenState();
|
State<DailyLimitScreen> createState() => _DailyLimitScreenState();
|
||||||
}
|
}
|
||||||
@@ -74,22 +76,40 @@ class _DailyLimitScreenState extends State<DailyLimitScreen> {
|
|||||||
child: Text(localizations.cancel),
|
child: Text(localizations.cancel),
|
||||||
),
|
),
|
||||||
ElevatedButton(
|
ElevatedButton(
|
||||||
onPressed: () {
|
onPressed: () async {
|
||||||
final value = double.tryParse(_limitController.text);
|
final value = double.tryParse(_limitController.text);
|
||||||
if (value == null || value <= 0) return;
|
if (value == null || value <= 0) return;
|
||||||
|
|
||||||
if (value > 200000) {
|
if (value > 200000) {
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
SnackBar(
|
SnackBar(
|
||||||
content: const Text(
|
content:
|
||||||
"Limit To be Set must be less than 200000"),
|
Text(localizations.limitToBeSetMustBeLessThan200000),
|
||||||
behavior: SnackBarBehavior.floating,
|
behavior: SnackBarBehavior.floating,
|
||||||
backgroundColor: theme.colorScheme.error,
|
backgroundColor: theme.colorScheme.error,
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
service.editLimit(value);
|
try {
|
||||||
Navigator.of(dialogContext).pop(value);
|
await service.getOtpTLimit(
|
||||||
|
mobileNumber: widget.mobileNumber);
|
||||||
|
Navigator.of(context).push(
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => ChangeLimitOTPScreen(
|
||||||
|
newLimit: value.toString(),
|
||||||
|
mobileNumber: widget.mobileNumber,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
} catch (e) {
|
||||||
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
|
SnackBar(
|
||||||
|
content: Text("Error: $e"),
|
||||||
|
behavior: SnackBarBehavior.floating,
|
||||||
|
backgroundColor: theme.colorScheme.error,
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
child: Text(localizations.save),
|
child: Text(localizations.save),
|
||||||
@@ -164,7 +184,7 @@ class _DailyLimitScreenState extends State<DailyLimitScreen> {
|
|||||||
if (_currentLimit != null) ...[
|
if (_currentLimit != null) ...[
|
||||||
const SizedBox(height: 24),
|
const SizedBox(height: 24),
|
||||||
Text(
|
Text(
|
||||||
"Remaining Limit Today", // This should be localized
|
localizations.remainingLimitToday, // This should be localized
|
||||||
style: theme.textTheme.titleMedium,
|
style: theme.textTheme.titleMedium,
|
||||||
),
|
),
|
||||||
const SizedBox(height: 4),
|
const SizedBox(height: 4),
|
||||||
@@ -204,14 +224,6 @@ class _DailyLimitScreenState extends State<DailyLimitScreen> {
|
|||||||
),
|
),
|
||||||
),
|
),
|
||||||
const SizedBox(height: 16),
|
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,
|
|
||||||
// ),
|
|
||||||
// ),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -23,42 +23,50 @@ class PreferenceScreen extends StatelessWidget {
|
|||||||
return Stack(
|
return Stack(
|
||||||
children: [
|
children: [
|
||||||
ListView(
|
ListView(
|
||||||
|
padding:
|
||||||
|
const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
|
||||||
children: [
|
children: [
|
||||||
//Set Prefered Username
|
|
||||||
// ListTile(
|
|
||||||
// leading: const Icon(Icons.person),
|
|
||||||
// title: const Text("Set Prefered Username"),
|
|
||||||
// onTap: () {
|
|
||||||
// }),
|
|
||||||
// Language Selection
|
// Language Selection
|
||||||
ListTile(
|
Card(
|
||||||
leading: const Icon(Icons.language),
|
margin: const EdgeInsets.only(bottom: 10),
|
||||||
title: Text(loc.language),
|
child: ListTile(
|
||||||
onTap: () {
|
leading: const Icon(Icons.language),
|
||||||
showDialog(
|
title: Text(loc.language),
|
||||||
context: context,
|
trailing: const Icon(Icons.chevron_right),
|
||||||
builder: (_) => const LanguageDialog(),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
//Theme Mode Switch (Light/Dark)
|
|
||||||
ListTile(
|
|
||||||
leading: const Icon(Icons.brightness_6),
|
|
||||||
title: Text(AppLocalizations.of(context).themeMode),
|
|
||||||
onTap: () {
|
|
||||||
showThemeModeDialog(context);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
//Color_Theme_Selection
|
|
||||||
ListTile(
|
|
||||||
leading: const Icon(Icons.color_lens),
|
|
||||||
title: Text(AppLocalizations.of(context).themeColor),
|
|
||||||
onTap: () {
|
onTap: () {
|
||||||
showDialog(
|
showDialog(
|
||||||
context: context,
|
context: context,
|
||||||
builder: (_) => const ColorThemeDialog(),
|
builder: (_) => const LanguageDialog(),
|
||||||
);
|
);
|
||||||
}),
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
//Theme Mode Switch (Light/Dark)
|
||||||
|
Card(
|
||||||
|
margin: const EdgeInsets.only(bottom: 10),
|
||||||
|
child: ListTile(
|
||||||
|
leading: const Icon(Icons.brightness_6),
|
||||||
|
title: Text(AppLocalizations.of(context).themeMode),
|
||||||
|
trailing: const Icon(Icons.chevron_right),
|
||||||
|
onTap: () {
|
||||||
|
showThemeModeDialog(context);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
),
|
||||||
|
//Color_Theme_Selection
|
||||||
|
Card(
|
||||||
|
margin: const EdgeInsets.only(bottom: 10),
|
||||||
|
child: ListTile(
|
||||||
|
leading: const Icon(Icons.color_lens),
|
||||||
|
title: Text(AppLocalizations.of(context).themeColor),
|
||||||
|
trailing: const Icon(Icons.chevron_right),
|
||||||
|
onTap: () {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (_) => const ColorThemeDialog(),
|
||||||
|
);
|
||||||
|
}),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
IgnorePointer(
|
IgnorePointer(
|
||||||
|
|||||||
@@ -7,8 +7,8 @@ import 'package:kmobile/features/profile/logout_dialog.dart';
|
|||||||
import 'package:kmobile/features/profile/security_settings_screen.dart';
|
import 'package:kmobile/features/profile/security_settings_screen.dart';
|
||||||
import 'package:kmobile/security/secure_storage.dart';
|
import 'package:kmobile/security/secure_storage.dart';
|
||||||
import 'package:local_auth/local_auth.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 'package:material_symbols_icons/material_symbols_icons.dart';
|
||||||
import '../../di/injection.dart';
|
import '../../di/injection.dart';
|
||||||
import '../../l10n/app_localizations.dart';
|
import '../../l10n/app_localizations.dart';
|
||||||
import 'package:kmobile/features/profile/preferences/preference_screen.dart';
|
import 'package:kmobile/features/profile/preferences/preference_screen.dart';
|
||||||
@@ -36,8 +36,7 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Future<String> _getAppVersion() async {
|
Future<String> _getAppVersion() async {
|
||||||
final PackageInfo info = await PackageInfo.fromPlatform();
|
return 'Version 1.0.1 (1))';
|
||||||
return 'Version ${info.version} (${info.buildNumber})';
|
|
||||||
}
|
}
|
||||||
|
|
||||||
Future<void> _loadBiometricStatus() async {
|
Future<void> _loadBiometricStatus() async {
|
||||||
@@ -188,12 +187,11 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
Container(
|
Container(
|
||||||
width: 56,
|
width: 56,
|
||||||
height: 56,
|
height: 56,
|
||||||
decoration: BoxDecoration(
|
child: const CircleAvatar(
|
||||||
shape: BoxShape.circle,
|
radius: 50,
|
||||||
color: theme.colorScheme.surface,
|
child: Icon(
|
||||||
image: const DecorationImage(
|
Symbols.person,
|
||||||
image: AssetImage('assets/images/logo.png'),
|
size: 56,
|
||||||
fit: BoxFit.cover,
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -221,17 +219,6 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
// Edit/Profile button (optional)
|
|
||||||
TextButton.icon(
|
|
||||||
onPressed: () {
|
|
||||||
// TODO: Navigate to edit profile if required
|
|
||||||
},
|
|
||||||
icon: const Icon(Icons.edit, size: 18),
|
|
||||||
label: const Text("Edit"),
|
|
||||||
style: TextButton.styleFrom(
|
|
||||||
foregroundColor: theme.colorScheme.onSurface,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
@@ -285,7 +272,8 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
MaterialPageRoute(
|
MaterialPageRoute(
|
||||||
builder: (context) => const DailyLimitScreen(),
|
builder: (context) =>
|
||||||
|
DailyLimitScreen(mobileNumber: widget.mobileNumber),
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
@@ -417,25 +405,6 @@ class _ProfileScreenState extends State<ProfileScreen> {
|
|||||||
const SizedBox(height: 24),
|
const SizedBox(height: 24),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|
||||||
// ===== Watermark (kept subtle, no theme change) =====
|
|
||||||
IgnorePointer(
|
|
||||||
child: Positioned.fill(
|
|
||||||
child: Center(
|
|
||||||
child: Opacity(
|
|
||||||
opacity: 0.06,
|
|
||||||
child: ClipOval(
|
|
||||||
child: Image.asset(
|
|
||||||
'assets/images/logo.png',
|
|
||||||
width: 200,
|
|
||||||
height: 200,
|
|
||||||
filterQuality: FilterQuality.medium,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
),
|
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
|
|||||||
@@ -24,99 +24,107 @@ class SecuritySettingsScreen extends StatelessWidget {
|
|||||||
body: Stack(
|
body: Stack(
|
||||||
children: [
|
children: [
|
||||||
ListView(
|
ListView(
|
||||||
|
padding: const EdgeInsets.symmetric(horizontal: 16, vertical: 12),
|
||||||
children: [
|
children: [
|
||||||
ListTile(
|
Card(
|
||||||
leading: const Icon(Icons.lock_outline),
|
margin: const EdgeInsets.only(bottom: 10),
|
||||||
title: Text(loc.changeLoginPassword),
|
child: ListTile(
|
||||||
trailing: const Icon(Icons.chevron_right),
|
leading: const Icon(Icons.lock_outline),
|
||||||
onTap: () {
|
title: Text(loc.changeLoginPassword),
|
||||||
Navigator.push(
|
trailing: const Icon(Icons.chevron_right),
|
||||||
context,
|
onTap: () {
|
||||||
MaterialPageRoute(
|
Navigator.push(
|
||||||
builder: (context) => ChangePasswordScreen(
|
context,
|
||||||
mobileNumber: mobileNumber,
|
MaterialPageRoute(
|
||||||
),
|
builder: (context) => ChangePasswordScreen(
|
||||||
),
|
mobileNumber: mobileNumber,
|
||||||
);
|
),
|
||||||
},
|
|
||||||
),
|
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
|
||||||
ListTile(
|
|
||||||
leading: const Icon(Icons.pin),
|
|
||||||
title: Text(loc.changeMpin),
|
|
||||||
trailing: const Icon(Icons.chevron_right),
|
|
||||||
onTap: () async {
|
|
||||||
final result = await Navigator.push(
|
|
||||||
context,
|
|
||||||
MaterialPageRoute(
|
|
||||||
builder: (context) => const ChangeMpinScreen(),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
|
|
||||||
if (result == true && context.mounted) {
|
|
||||||
ScaffoldMessenger.of(context).showSnackBar(
|
|
||||||
SnackBar(
|
|
||||||
content: Text(loc.mpinChangedSuccessfully),
|
|
||||||
backgroundColor:
|
|
||||||
Theme.of(context).colorScheme.secondary,
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
},
|
||||||
},
|
),
|
||||||
),
|
),
|
||||||
Divider(height: 1, color: Theme.of(context).dividerColor),
|
Card(
|
||||||
ListTile(
|
margin: const EdgeInsets.only(bottom: 10),
|
||||||
leading: const Icon(Icons.password),
|
child: ListTile(
|
||||||
title: const Text('Change TPIN'),
|
leading: const Icon(Icons.pin),
|
||||||
trailing: const Icon(Icons.chevron_right),
|
title: Text(loc.changeMpin),
|
||||||
onTap: () async {
|
trailing: const Icon(Icons.chevron_right),
|
||||||
final authService = getIt<AuthService>();
|
onTap: () async {
|
||||||
final isTpinSet = await authService.checkTpin();
|
final result = await Navigator.push(
|
||||||
|
context,
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) => const ChangeMpinScreen(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
|
||||||
if (!isTpinSet) {
|
if (result == true && context.mounted) {
|
||||||
if (context.mounted) {
|
ScaffoldMessenger.of(context).showSnackBar(
|
||||||
showDialog(
|
SnackBar(
|
||||||
context: context,
|
content: Text(loc.mpinChangedSuccessfully),
|
||||||
builder: (BuildContext context) {
|
backgroundColor:
|
||||||
return AlertDialog(
|
Theme.of(context).colorScheme.secondary,
|
||||||
title: const Text('TPIN Not Set'),
|
|
||||||
content: const Text(
|
|
||||||
'You have not set a TPIN yet. Please set a TPIN to proceed.'),
|
|
||||||
actions: <Widget>[
|
|
||||||
TextButton(
|
|
||||||
child: const Text('Back'),
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
},
|
|
||||||
),
|
|
||||||
TextButton(
|
|
||||||
child: const Text('Proceed'),
|
|
||||||
onPressed: () {
|
|
||||||
Navigator.of(context).pop();
|
|
||||||
Navigator.of(context).push(
|
|
||||||
MaterialPageRoute(
|
|
||||||
builder: (context) =>
|
|
||||||
const TpinSetScreen(),
|
|
||||||
),
|
|
||||||
);
|
|
||||||
},
|
|
||||||
),
|
|
||||||
],
|
|
||||||
);
|
|
||||||
},
|
|
||||||
);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
if (context.mounted) {
|
|
||||||
Navigator.of(context).push(
|
|
||||||
MaterialPageRoute(
|
|
||||||
builder: (context) =>
|
|
||||||
ChangeTpinScreen(mobileNumber: mobileNumber),
|
|
||||||
),
|
),
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
},
|
||||||
},
|
),
|
||||||
|
),
|
||||||
|
Card(
|
||||||
|
margin: const EdgeInsets.only(bottom: 10),
|
||||||
|
child: ListTile(
|
||||||
|
leading: const Icon(Icons.password),
|
||||||
|
title: const Text('Change TPIN'),
|
||||||
|
trailing: const Icon(Icons.chevron_right),
|
||||||
|
onTap: () async {
|
||||||
|
final authService = getIt<AuthService>();
|
||||||
|
final isTpinSet = await authService.checkTpin();
|
||||||
|
|
||||||
|
if (!isTpinSet) {
|
||||||
|
if (context.mounted) {
|
||||||
|
showDialog(
|
||||||
|
context: context,
|
||||||
|
builder: (BuildContext context) {
|
||||||
|
return AlertDialog(
|
||||||
|
title: const Text('TPIN Not Set'),
|
||||||
|
content: const Text(
|
||||||
|
'You have not set a TPIN yet. Please set a TPIN to proceed.'),
|
||||||
|
actions: <Widget>[
|
||||||
|
TextButton(
|
||||||
|
child: const Text('Back'),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
},
|
||||||
|
),
|
||||||
|
TextButton(
|
||||||
|
child: const Text('Proceed'),
|
||||||
|
onPressed: () {
|
||||||
|
Navigator.of(context).pop();
|
||||||
|
Navigator.of(context).push(
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) =>
|
||||||
|
const TpinSetScreen(),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
},
|
||||||
|
),
|
||||||
|
],
|
||||||
|
);
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
if (context.mounted) {
|
||||||
|
Navigator.of(context).push(
|
||||||
|
MaterialPageRoute(
|
||||||
|
builder: (context) =>
|
||||||
|
ChangeTpinScreen(mobileNumber: mobileNumber),
|
||||||
|
),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
),
|
||||||
),
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -32,6 +32,7 @@ class _QuickPayScreen extends State<QuickPayScreen> {
|
|||||||
child: QuickPayManagementTile(
|
child: QuickPayManagementTile(
|
||||||
icon: Symbols.input_circle,
|
icon: Symbols.input_circle,
|
||||||
label: AppLocalizations.of(context).ownBank,
|
label: AppLocalizations.of(context).ownBank,
|
||||||
|
subtitle: AppLocalizations.of(context).quickownsubtitle,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
@@ -49,6 +50,7 @@ class _QuickPayScreen extends State<QuickPayScreen> {
|
|||||||
child: QuickPayManagementTile(
|
child: QuickPayManagementTile(
|
||||||
icon: Symbols.output_circle,
|
icon: Symbols.output_circle,
|
||||||
label: AppLocalizations.of(context).outsideBank,
|
label: AppLocalizations.of(context).outsideBank,
|
||||||
|
subtitle: AppLocalizations.of(context).quickoutsidesubtitle,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
@@ -87,6 +89,7 @@ class _QuickPayScreen extends State<QuickPayScreen> {
|
|||||||
class QuickPayManagementTile extends StatelessWidget {
|
class QuickPayManagementTile extends StatelessWidget {
|
||||||
final IconData icon;
|
final IconData icon;
|
||||||
final String label;
|
final String label;
|
||||||
|
final String? subtitle;
|
||||||
final VoidCallback onTap;
|
final VoidCallback onTap;
|
||||||
final bool disable;
|
final bool disable;
|
||||||
|
|
||||||
@@ -94,6 +97,7 @@ class QuickPayManagementTile extends StatelessWidget {
|
|||||||
super.key,
|
super.key,
|
||||||
required this.icon,
|
required this.icon,
|
||||||
required this.label,
|
required this.label,
|
||||||
|
this.subtitle,
|
||||||
required this.onTap,
|
required this.onTap,
|
||||||
this.disable = false,
|
this.disable = false,
|
||||||
});
|
});
|
||||||
@@ -133,6 +137,19 @@ class QuickPayManagementTile extends StatelessWidget {
|
|||||||
: theme.colorScheme.onSurface,
|
: theme.colorScheme.onSurface,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
if (subtitle != null) // Conditionally display subtitle
|
||||||
|
Padding(
|
||||||
|
padding: const EdgeInsets.only(top: 8.0),
|
||||||
|
child: Text(
|
||||||
|
subtitle!,
|
||||||
|
textAlign: TextAlign.center,
|
||||||
|
style: theme.textTheme.bodyMedium?.copyWith(
|
||||||
|
color: disable
|
||||||
|
? theme.disabledColor
|
||||||
|
: theme.colorScheme.onSurfaceVariant,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -143,9 +143,6 @@ class _QuickPayWithinBankScreen extends State<QuickPayWithinBankScreen> {
|
|||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text(
|
||||||
AppLocalizations.of(context).quickPayOwnBank,
|
AppLocalizations.of(context).quickPayOwnBank,
|
||||||
style: TextStyle(
|
|
||||||
color: Theme.of(context).colorScheme.onSurface,
|
|
||||||
fontWeight: FontWeight.w500),
|
|
||||||
),
|
),
|
||||||
centerTitle: false,
|
centerTitle: false,
|
||||||
),
|
),
|
||||||
|
|||||||
@@ -1,7 +1,9 @@
|
|||||||
import 'package:flutter/material.dart';
|
import 'package:flutter/material.dart';
|
||||||
import 'package:kmobile/api/services/branch_service.dart'; // Added: Import BranchService for Atm model and API calls
|
import 'package:kmobile/api/services/branch_service.dart'; // Added: Import BranchService for Atm model and API calls
|
||||||
import 'package:kmobile/di/injection.dart'; // Added: Import for dependency injection (getIt)
|
import 'package:kmobile/di/injection.dart'; // Added: Import for dependency injection (getIt)
|
||||||
import 'package:shimmer/shimmer.dart'; // Added: Import for shimmer loading effect
|
import 'package:shimmer/shimmer.dart';
|
||||||
|
|
||||||
|
import '../../../l10n/app_localizations.dart'; // Added: Import for shimmer loading effect
|
||||||
|
|
||||||
// Removed: The local 'Location' class is no longer needed as we use the 'Atm' model from branch_service.dart
|
// Removed: The local 'Location' class is no longer needed as we use the 'Atm' model from branch_service.dart
|
||||||
|
|
||||||
@@ -60,7 +62,8 @@ class _ATMLocatorScreenState extends State<ATMLocatorScreen> {
|
|||||||
Widget build(BuildContext context) {
|
Widget build(BuildContext context) {
|
||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: const Text("ATM Locator"), // Title for the app bar
|
title: Text(
|
||||||
|
AppLocalizations.of(context).atmlocator), // Title for the app bar
|
||||||
),
|
),
|
||||||
body: Stack(
|
body: Stack(
|
||||||
children: [
|
children: [
|
||||||
@@ -73,7 +76,8 @@ class _ATMLocatorScreenState extends State<ATMLocatorScreen> {
|
|||||||
onChanged:
|
onChanged:
|
||||||
_filterAtms, // Updated: Call _filterAtms on text change
|
_filterAtms, // Updated: Call _filterAtms on text change
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
hintText: "Name/Address", // Hint text for the search bar
|
hintText: AppLocalizations.of(context)
|
||||||
|
.nameAddress, // Hint text for the search bar
|
||||||
prefixIcon: const Icon(Icons.search), // Search icon
|
prefixIcon: const Icon(Icons.search), // Search icon
|
||||||
border: OutlineInputBorder(
|
border: OutlineInputBorder(
|
||||||
borderRadius: BorderRadius.circular(12),
|
borderRadius: BorderRadius.circular(12),
|
||||||
@@ -87,9 +91,9 @@ class _ATMLocatorScreenState extends State<ATMLocatorScreen> {
|
|||||||
child: _isLoading
|
child: _isLoading
|
||||||
? _buildShimmerList() // Display shimmer while loading
|
? _buildShimmerList() // Display shimmer while loading
|
||||||
: _filteredAtms.isEmpty
|
: _filteredAtms.isEmpty
|
||||||
? const Center(
|
? Center(
|
||||||
child: Text(
|
child: Text(AppLocalizations.of(context)
|
||||||
"No matching ATMs found")) // Message if no ATMs found
|
.noMatchingAtmsFound)) // Message if no ATMs found
|
||||||
: ListView.builder(
|
: ListView.builder(
|
||||||
itemCount: _filteredAtms
|
itemCount: _filteredAtms
|
||||||
.length, // Number of items in the filtered list
|
.length, // Number of items in the filtered list
|
||||||
@@ -128,7 +132,7 @@ class _ATMLocatorScreenState extends State<ATMLocatorScreen> {
|
|||||||
return Card(
|
return Card(
|
||||||
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
||||||
child: ListTile(
|
child: ListTile(
|
||||||
leading: const Icon(Icons.currency_rupee), // Icon for ATM
|
leading: const Icon(Icons.credit_card), // Icon for ATM
|
||||||
title: Text(atm.name, // Display the ATM's name
|
title: Text(atm.name, // Display the ATM's name
|
||||||
style: const TextStyle(fontWeight: FontWeight.bold)),
|
style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
onTap: () {
|
onTap: () {
|
||||||
|
|||||||
@@ -6,6 +6,8 @@ import 'package:kmobile/di/injection.dart';
|
|||||||
import 'package:shimmer/shimmer.dart';
|
import 'package:shimmer/shimmer.dart';
|
||||||
import 'package:kmobile/features/service/screens/branch_details_screen.dart';
|
import 'package:kmobile/features/service/screens/branch_details_screen.dart';
|
||||||
|
|
||||||
|
import '../../../l10n/app_localizations.dart';
|
||||||
|
|
||||||
class BranchLocatorScreen extends StatefulWidget {
|
class BranchLocatorScreen extends StatefulWidget {
|
||||||
const BranchLocatorScreen({super.key});
|
const BranchLocatorScreen({super.key});
|
||||||
|
|
||||||
@@ -54,10 +56,7 @@ class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
|||||||
return Scaffold(
|
return Scaffold(
|
||||||
appBar: AppBar(
|
appBar: AppBar(
|
||||||
title: Text(
|
title: Text(
|
||||||
"Branch Locator",
|
AppLocalizations.of(context).branchlocator,
|
||||||
style: Theme.of(context).textTheme.titleLarge?.copyWith(
|
|
||||||
color: Theme.of(context).colorScheme.onSurface,
|
|
||||||
),
|
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
body: Stack(
|
body: Stack(
|
||||||
@@ -70,7 +69,7 @@ class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
|||||||
controller: _searchController,
|
controller: _searchController,
|
||||||
onChanged: _filterBranches, // Updated
|
onChanged: _filterBranches, // Updated
|
||||||
decoration: InputDecoration(
|
decoration: InputDecoration(
|
||||||
hintText: "Branch Name",
|
hintText: AppLocalizations.of(context).searchbranch,
|
||||||
prefixIcon: const Icon(Icons.search),
|
prefixIcon: const Icon(Icons.search),
|
||||||
border: OutlineInputBorder(
|
border: OutlineInputBorder(
|
||||||
borderRadius: BorderRadius.circular(12),
|
borderRadius: BorderRadius.circular(12),
|
||||||
@@ -84,9 +83,9 @@ class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
|||||||
child: _isLoading
|
child: _isLoading
|
||||||
? _buildShimmerList() // Changed to shimmer
|
? _buildShimmerList() // Changed to shimmer
|
||||||
: _filteredBranches.isEmpty
|
: _filteredBranches.isEmpty
|
||||||
? const Center(
|
? Center(
|
||||||
child: Text(
|
child: Text(AppLocalizations.of(context)
|
||||||
"No matching branches found")) // Updated tex
|
.noMatchingBranchesFound)) // Updated tex
|
||||||
: ListView.builder(
|
: ListView.builder(
|
||||||
itemCount: _filteredBranches.length,
|
itemCount: _filteredBranches.length,
|
||||||
itemBuilder: (context, index) {
|
itemBuilder: (context, index) {
|
||||||
@@ -137,7 +136,7 @@ class _BranchLocatorScreenState extends State<BranchLocatorScreen> {
|
|||||||
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
margin: const EdgeInsets.symmetric(horizontal: 12, vertical: 6),
|
||||||
child: ListTile(
|
child: ListTile(
|
||||||
leading: const CircleAvatar(
|
leading: const CircleAvatar(
|
||||||
child: Icon(Icons.location_city),
|
child: Icon(Icons.account_balance),
|
||||||
),
|
),
|
||||||
title: Text(branch.branch_name,
|
title: Text(branch.branch_name,
|
||||||
style: const TextStyle(fontWeight: FontWeight.bold)),
|
style: const TextStyle(fontWeight: FontWeight.bold)),
|
||||||
|
|||||||
@@ -61,7 +61,7 @@ class _ServiceScreen extends State<ServiceScreen> {
|
|||||||
Expanded(
|
Expanded(
|
||||||
child: ServiceManagementTile(
|
child: ServiceManagementTile(
|
||||||
icon: Symbols.location_pin,
|
icon: Symbols.location_pin,
|
||||||
label: "ATM Locator",
|
label: AppLocalizations.of(context).atmlocator,
|
||||||
onTap: () {
|
onTap: () {
|
||||||
Navigator.push(
|
Navigator.push(
|
||||||
context,
|
context,
|
||||||
|
|||||||
@@ -225,7 +225,7 @@
|
|||||||
"pinMismatch": "PINs do not match",
|
"pinMismatch": "PINs do not match",
|
||||||
"securitySettings": "Security Settings",
|
"securitySettings": "Security Settings",
|
||||||
"kconnect": "Kconnect",
|
"kconnect": "Kconnect",
|
||||||
"kccBankFull": "Kangra Central Co-operative Bank",
|
"kccBankFull": "The Kangra Central Co-operative Bank Ltd.",
|
||||||
"themeColor": "Theme Color",
|
"themeColor": "Theme Color",
|
||||||
"selectThemeColor": "Select Theme Color",
|
"selectThemeColor": "Select Theme Color",
|
||||||
"violet": "Violet",
|
"violet": "Violet",
|
||||||
@@ -323,7 +323,7 @@
|
|||||||
"details": "Details",
|
"details": "Details",
|
||||||
"remarks": "Remarks (Optional)",
|
"remarks": "Remarks (Optional)",
|
||||||
"kccbMobile": "KCCB Mobile",
|
"kccbMobile": "KCCB Mobile",
|
||||||
"faq": "Frequently Asked Questions(FAQs)",
|
"faq": "Frequently Asked Questions",
|
||||||
"branches": "Branches",
|
"branches": "Branches",
|
||||||
"atms": "ATMs",
|
"atms": "ATMs",
|
||||||
"dailylimit": "Daily Transaction Limit",
|
"dailylimit": "Daily Transaction Limit",
|
||||||
@@ -406,5 +406,154 @@
|
|||||||
"rbiCode2": "RBI Code 2",
|
"rbiCode2": "RBI Code 2",
|
||||||
"latitude": "Latitude",
|
"latitude": "Latitude",
|
||||||
"address": "Customer Address",
|
"address": "Customer Address",
|
||||||
"transactions": "Transactions"
|
"transactions": "Transactions",
|
||||||
|
"quickownsubtitle": "Seamlessly send money to your loved ones within Kangra Bank. Fast, secure, and always at your fingertips",
|
||||||
|
"quickoutsidesubtitle": "Transfer funds to any bank across India with ease. Your transactions are secure and processed quickly",
|
||||||
|
"ftselfpaysubtitle": "Move money between your own accounts with ease. Your funds, your way",
|
||||||
|
"ftownsubtitle": "Send money to your saved beneficiaries within Kangra Bank",
|
||||||
|
"ftoutsidesubtitle": "Transfer funds to your saved beneficiaries in any other bank across India",
|
||||||
|
"personaldetails": "Personal Details",
|
||||||
|
"kycdetails": "KYC Details",
|
||||||
|
"viewall": "View All",
|
||||||
|
"branchlocator": "Branch Locator",
|
||||||
|
"atmlocator": "ATM Locator",
|
||||||
|
"limitSetError": "Limit to be set must be less than {maxAmount}",
|
||||||
|
"genericError": "Error: {errorMessage}",
|
||||||
|
"limitUpdatedSuccess": "Limit Updated",
|
||||||
|
"remainingLimitToday": "Remaining Limit Today",
|
||||||
|
"branchNameHint": "Branch Name",
|
||||||
|
"noMatchingBranches": "No matching branches found",
|
||||||
|
"selfPay": "Self Pay",
|
||||||
|
"savingsAccountType": "Savings",
|
||||||
|
"amountExceedsDailyLimit": "Amount exceeds remaining daily limit of ",
|
||||||
|
"incorrectTpinError": "Please Enter the correct TPIN",
|
||||||
|
"insufficientFundsError": "Your account does not have sufficient balance",
|
||||||
|
"somethingWentWrongError": "Something Went Wrong",
|
||||||
|
"currencyINR": "INR",
|
||||||
|
"remainingDailyLimit": "Remaining Daily Limit:",
|
||||||
|
"searchByNameOrAccount": "Search by name or account number",
|
||||||
|
"beneficiaryCooldownMessage": "Beneficiary will be enabled after the cooldown period.",
|
||||||
|
"notApplicable": "N/A",
|
||||||
|
"savingsAccountLabel": "Savings Account",
|
||||||
|
"loanAccountLabel": "Loan Account",
|
||||||
|
"termDepositLabel": "Term Deposit",
|
||||||
|
"recurringDepositLabel": "Recurring Deposit",
|
||||||
|
"currentAccountLabel": "Current Account",
|
||||||
|
"unknownAccountLabel": "Unknown Account",
|
||||||
|
"selectAccountTitle": "Select Account",
|
||||||
|
"noOtherAccounts": "No other accounts found",
|
||||||
|
"kccbBankName": "Kangra Central Co-operative Bank",
|
||||||
|
"fundTransferTitle": "Fund Transfer",
|
||||||
|
"debitFromLabel": "Debit From",
|
||||||
|
"creditToLabel": "Credited To",
|
||||||
|
"remarksOptionalHint": "Remarks (Optional)",
|
||||||
|
"amountLabel": "Amount",
|
||||||
|
"amountRequiredError": "Amount is required",
|
||||||
|
"validAmountError": "Please enter a valid amount",
|
||||||
|
"fetchingDailyLimitLoader": "Fetching daily limit...",
|
||||||
|
"proceedButton": "Proceed",
|
||||||
|
"enterKey": "Enter",
|
||||||
|
"backKey": "back",
|
||||||
|
"doneKey": "done",
|
||||||
|
"transactionDate": "On {date}",
|
||||||
|
"paymentResultPng": "/payment_result.png",
|
||||||
|
"rubikFont": "Rubik",
|
||||||
|
"transactionDateLabel": "Date: {date}",
|
||||||
|
"utrLabel": "UTR: {utr}",
|
||||||
|
"searchByNameOrAccountHint": "Search by name or account number",
|
||||||
|
"savingsAccountDropdown": "Savings",
|
||||||
|
"beneficiaryExistsError": "Beneficiary already exists",
|
||||||
|
"somethingWentWrongShort": "Something went Wrong",
|
||||||
|
"currentAccountDropdown": "Current",
|
||||||
|
"failedToDeleteBeneficiaryError": "Failed to delete beneficiary: {error}",
|
||||||
|
"notAvailable": "N/A",
|
||||||
|
"bankNameLabel": "Bank Name",
|
||||||
|
"accountNumberLabel": "Account Number",
|
||||||
|
"accountTypeLabel": "Account Type",
|
||||||
|
"ifscCodeLabel": "IFSC Code",
|
||||||
|
"branchNameLabel": "Branch Name",
|
||||||
|
"enquiryEmailSubject": "Enquiry",
|
||||||
|
"couldNotOpenEmailApp": "Could not open email app for {email}",
|
||||||
|
"couldNotOpenDialer": "Could not open dialer for {phone}",
|
||||||
|
"couldNotLaunchUrl": "Could not launch {url}",
|
||||||
|
"complaintFormUrl": "https://kccbhp.bank.in/complaint-form/",
|
||||||
|
"complaintFormTitle": "Complaint Form",
|
||||||
|
"chairmanEmail": "chairman@kccb.in",
|
||||||
|
"chairmanPhone": "01892-222677",
|
||||||
|
"mdEmail": "md@kccb.in",
|
||||||
|
"mdPhone": "01892-224969",
|
||||||
|
"gmwEmail": "gmw@kccb.in",
|
||||||
|
"gmwPhone": "01892-223280",
|
||||||
|
"gmnEmail": "gmn@kccb.in",
|
||||||
|
"gmnPhone": "01892-224607",
|
||||||
|
"atmNameAddressHint": "Name/Address",
|
||||||
|
"noMatchingAtms": "No matching ATMs found",
|
||||||
|
"faq1Question": "How do I log in to the mobile banking app?",
|
||||||
|
"faq1Answer": "You can log in using your customer number and password. Biometric login (fingerprint) and MPIN is also available for supported evices.",
|
||||||
|
"faq2Question": "Is my banking information secure on this app?",
|
||||||
|
"faq2Answer": "Yes. We use industry-standard encryption and multi-factor authentication to ensure your data is safe.",
|
||||||
|
"faq3Question": "How can I check my account balance?",
|
||||||
|
"faq3Answer": "Once logged in, your account balance will be displayed on the home screen. You can also view detailed balances under the “Accounts” section.",
|
||||||
|
"faq4Question": "Can I transfer money to other bank accounts?",
|
||||||
|
"faq4Answer": "Yes. You can use NEFT, RTGS or IMPS to transfer funds to any bank account in India.",
|
||||||
|
"faq5Question": "How do I view my transaction history?",
|
||||||
|
"faq5Answer": "Click on the “Account Statement” icon under the Home Screen to view recent and past transactions.",
|
||||||
|
"chequeEnquiryTitle": "Cheque Enquiry",
|
||||||
|
"chequeEnquirySubtitle": "You can view the status of your issued cheque book, presented cheques and details of stopped cheques including relevant dates, cheque numbers and other essential information",
|
||||||
|
"stopChequeSubtitle": "Initiate stop for one or more cheques from your issued checkbook. This essential service helps prevent unauthorized transactions and protects against fraud.",
|
||||||
|
"chequeEnquiryFailedError": "Failed to fetch cheque status: {error}",
|
||||||
|
"accountNumberTitle": "Account Number",
|
||||||
|
"noAccountsFound": "No accounts found",
|
||||||
|
"searchByChequeDetailsHint": "Search by Cheque Details",
|
||||||
|
"noChequeStatusFound": "No cheque status found.",
|
||||||
|
"chequebookIssuedLabel": "Chequebook Issued (CI)",
|
||||||
|
"branchCodeLabel": "Branch Code:",
|
||||||
|
"fromChequeLabel": "From Cheque:",
|
||||||
|
"toChequeLabel": "To Cheque:",
|
||||||
|
"dateLabel": "Date:",
|
||||||
|
"chequesCountLabel": "Cheques Count:",
|
||||||
|
"presentedChequeLabel": "Presented Cheque (PR)",
|
||||||
|
"chequeNumberLabel": "Cheque Number:",
|
||||||
|
"transactionCodeLabel": "Transaction Code:",
|
||||||
|
"amountLabelWithColon": "Amount:",
|
||||||
|
"statusLabel": "Status:",
|
||||||
|
"stopChequeLabel": "Stop Cheque (ST)",
|
||||||
|
"stopIssueDateLabel": "Stop Issue Date:",
|
||||||
|
"stopExpiryDateLabel": "Stop Expiry Date:",
|
||||||
|
"emptyString": "",
|
||||||
|
"cashCreditAccountLabel": "Cash Credit Account",
|
||||||
|
"stopChequeTitle": "Stop Cheque",
|
||||||
|
"noChequebookToStop": "No cheque book found to stop cheques from.",
|
||||||
|
"stopSingleChequeButton": "Stop Single Cheque",
|
||||||
|
"pleaseSelectAccountFirst": "Please select an account first.",
|
||||||
|
"stopMultipleChequesButton": "Stop Multiple Cheques",
|
||||||
|
"noChequeIssuedStatus": "No Cheque Issued status found.",
|
||||||
|
"chequebookDetailsTitle": "Chequebook Details",
|
||||||
|
"customerNameLabel": "Customer Name:",
|
||||||
|
"cifNumberLabel": "CIF Number:",
|
||||||
|
"startingChequeNumberLabel": "Starting Cheque Number:",
|
||||||
|
"endingChequeNumberLabel": "Ending Cheque Number:",
|
||||||
|
"issueDateLabel": "Issue Date:",
|
||||||
|
"numberOfChequesLabel": "Number of Cheques:",
|
||||||
|
"instrumentTypeLabel": "Instrument Type:",
|
||||||
|
"closeButton": "Close",
|
||||||
|
"stopSingleChequeTitle": "Stop Single Cheque",
|
||||||
|
"chequeNumberHint": "Cheque Number *",
|
||||||
|
"pleaseEnterChequeNumberError": "Please enter a cheque number",
|
||||||
|
"invalidChequeNumberFormatError": "Invalid cheque number format",
|
||||||
|
"chequeNumberRangeError": "Cheque number must be between {from} and {to}",
|
||||||
|
"instrumentTypeHint": "Instrument Type *",
|
||||||
|
"stopIssueDateHint": "Stop Issue Date",
|
||||||
|
"stopExpiryDateHint": "Stop Expiry Date",
|
||||||
|
"stopAmountHint": "Stop Amount",
|
||||||
|
"stopCommentHint": "Stop Comment",
|
||||||
|
"chequebookIssueDateHint": "Chequebook Issue Date",
|
||||||
|
"successStatus": "Success",
|
||||||
|
"errorStatus": "Error",
|
||||||
|
"incorrectTpinErrorMessage": "The TPIN you entered is incorrect. Please try again.",
|
||||||
|
"internalServerError": "Internal Server Error",
|
||||||
|
"stopChequeButton": "Stop Cheque",
|
||||||
|
"stopMultipleChequesTitle": "Stop Multiple Cheques",
|
||||||
|
"fromChequeNumberHint": "From Cheque Number *",
|
||||||
|
"toChequeNumberHint": "To Cheque Number *"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -407,5 +407,154 @@
|
|||||||
"rbiCode2": "आरबीआई कोड 2",
|
"rbiCode2": "आरबीआई कोड 2",
|
||||||
"latitude": "अक्षांश",
|
"latitude": "अक्षांश",
|
||||||
"address": "ग्राहक का पता",
|
"address": "ग्राहक का पता",
|
||||||
"transactions": "लेनदेन"
|
"transactions": "लेनदेन",
|
||||||
|
"quickownsubtitle": "कांगड़ा बैंक के ज़रिए अपने प्रियजनों को आसानी से पैसे भेजें। तेज़, सुरक्षित और हमेशा आपकी उंगलियों पर",
|
||||||
|
"quickoutsidesubtitle": "भारत भर में किसी भी बैंक में आसानी से धनराशि स्थानांतरित करें। आपके लेन-देन सुरक्षित और शीघ्रता से संसाधित होते हैं",
|
||||||
|
"ftselfpaysubtitle": "अपने खातों के बीच आसानी से पैसे ट्रांसफर करें। आपका पैसा, आपकी सुविधानुसार",
|
||||||
|
"ftownsubtitle": "कांगड़ा बैंक के माध्यम से अपने बचत लाभार्थियों को पैसे भेजें",
|
||||||
|
"ftoutsidesubtitle": "भारत भर में किसी भी अन्य बैंक में अपने सहेजे गए लाभार्थियों को धनराशि हस्तांतरित करें",
|
||||||
|
"personaldetails": "व्यक्तिगत विवरण",
|
||||||
|
"kycdetails": "केवाईसी विवरण",
|
||||||
|
"viewall": "सभी देखें",
|
||||||
|
"branchlocator": "शाखा लोकेटर",
|
||||||
|
"atmlocator": "एटीएम लोकेटर",
|
||||||
|
"limitSetError": "निर्धारित की जाने वाली सीमा {maxAmount} से कम होनी चाहिए।",
|
||||||
|
"genericError": "त्रुटि: {errorMessage}",
|
||||||
|
"limitUpdatedSuccess": "सीमा अपडेट हो गई",
|
||||||
|
"remainingLimitToday": "आज की शेष सीमा",
|
||||||
|
"branchNameHint": "शाखा का नाम",
|
||||||
|
"noMatchingBranches": "कोई मेल खाने वाली शाखा नहीं मिली",
|
||||||
|
"selfPay": "स्वयं भुगतान",
|
||||||
|
"savingsAccountType": "बचत खाता",
|
||||||
|
"amountExceedsDailyLimit": "राशि की शेष दैनिक सीमा से अधिक है",
|
||||||
|
"incorrectTpinError": "कृपया सही टीपिन दर्ज करें",
|
||||||
|
"insufficientFundsError": "आपके खाते में पर्याप्त शेष राशि नहीं है",
|
||||||
|
"somethingWentWrongError": "कुछ गलत हो गया",
|
||||||
|
"currencyINR": "INR",
|
||||||
|
"remainingDailyLimit": "शेष दैनिक सीमा:",
|
||||||
|
"searchByNameOrAccount": "नाम या खाता संख्या से खोजें",
|
||||||
|
"beneficiaryCooldownMessage": "कूलडाउन अवधि के बाद लाभार्थी सक्षम हो जाएगा।",
|
||||||
|
"notApplicable": "लागू नहीं",
|
||||||
|
"savingsAccountLabel": "बचत खाता",
|
||||||
|
"loanAccountLabel": "ऋण खाता",
|
||||||
|
"termDepositLabel": "सावधि जमा",
|
||||||
|
"recurringDepositLabel": "आवर्ती जमा",
|
||||||
|
"currentAccountLabel": "चालू खाता",
|
||||||
|
"unknownAccountLabel": "अज्ञात खाता",
|
||||||
|
"selectAccountTitle": "खाता चुनें",
|
||||||
|
"noOtherAccounts": "कोई अन्य खाता नहीं मिला",
|
||||||
|
"kccbBankName": "कांगड़ा केंद्रीय सहकारी बैंक",
|
||||||
|
"fundTransferTitle": "धन हस्तांतरण",
|
||||||
|
"debitFromLabel": "से डेबिट करें",
|
||||||
|
"creditToLabel": "को क्रेडिट करें",
|
||||||
|
"remarksOptionalHint": "टिप्पणी (वैकल्पिक)",
|
||||||
|
"amountLabel": "राशि",
|
||||||
|
"amountRequiredError": "राशि आवश्यक है",
|
||||||
|
"validAmountError": "कृपया एक वैध राशि दर्ज करें",
|
||||||
|
"fetchingDailyLimitLoader": "दैनिक सीमा लाई जा रही है...",
|
||||||
|
"proceedButton": "आगे बढ़ें",
|
||||||
|
"enterKey": "दर्ज करें",
|
||||||
|
"backKey": "वापस",
|
||||||
|
"doneKey": "पूर्ण",
|
||||||
|
"transactionDate": "{date} को",
|
||||||
|
"paymentResultPng": "/payment_result.png",
|
||||||
|
"rubikFont": "Rubik",
|
||||||
|
"transactionDateLabel": "दिनांक: {date}",
|
||||||
|
"utrLabel": "UTR: {utr}",
|
||||||
|
"searchByNameOrAccountHint": "नाम या खाता संख्या से खोजें",
|
||||||
|
"savingsAccountDropdown": "बचत",
|
||||||
|
"beneficiaryExistsError": "लाभार्थी पहले से मौजूद है",
|
||||||
|
"somethingWentWrongShort": "कुछ गलत हो गया",
|
||||||
|
"currentAccountDropdown": "चालू",
|
||||||
|
"failedToDeleteBeneficiaryError": "लाभार्थी को हटाने में विफल: {error}",
|
||||||
|
"notAvailable": "उपलब्ध नहीं है",
|
||||||
|
"bankNameLabel": "बैंक का नाम",
|
||||||
|
"accountNumberLabel": "खाता संख्या",
|
||||||
|
"accountTypeLabel": "खाते का प्रकार",
|
||||||
|
"ifscCodeLabel": "IFSC कोड",
|
||||||
|
"branchNameLabel": "शाखा का नाम",
|
||||||
|
"enquiryEmailSubject": "पूछताछ",
|
||||||
|
"couldNotOpenEmailApp": "{email} के लिए ईमेल ऐप नहीं खोला जा सका",
|
||||||
|
"couldNotOpenDialer": "{phone} के लिए डायलर नहीं खोला जा सका",
|
||||||
|
"couldNotLaunchUrl": "{url} लॉन्च नहीं किया जा सका",
|
||||||
|
"complaintFormUrl": "https://kccbhp.bank.in/complaint-form/",
|
||||||
|
"complaintFormTitle": "शिकायत प्रपत्र",
|
||||||
|
"chairmanEmail": "chairman@kccb.in",
|
||||||
|
"chairmanPhone": "01892-222677",
|
||||||
|
"mdEmail": "md@kccb.in",
|
||||||
|
"mdPhone": "01892-224969",
|
||||||
|
"gmwEmail": "gmw@kccb.in",
|
||||||
|
"gmwPhone": "01892-223280",
|
||||||
|
"gmnEmail": "gmn@kccb.in",
|
||||||
|
"gmnPhone": "01892-224607",
|
||||||
|
"atmNameAddressHint": "नाम/पता",
|
||||||
|
"noMatchingAtms": "कोई मेल खाने वाला एटीएम नहीं मिला",
|
||||||
|
"faq1Question": "मैं मोबाइल बैंकिंग ऐप में कैसे लॉग इन करूं?",
|
||||||
|
"faq1Answer": "आप अपने ग्राहक नंबर और पासवर्ड का उपयोग करके लॉग इन कर सकते हैं। समर्थित उपकरणों के लिए बायोमेट्रिक लॉगिन (फिंगरप्रिंट) और एमपिन भी उ।",
|
||||||
|
"faq2Question": "क्या इस ऐप पर मेरी बैंकिंग जानकारी सुरक्षित है?",
|
||||||
|
"faq2Answer": "हां। हम आपके डेटा को सुरक्षित रखने के लिए उद्योग-मानक एन्क्रिप्शन और बहु-कारक प्रमाणीकरण का उपयोग करते हैं।",
|
||||||
|
"faq3Question": "मैं अपने खाते की शेष राशि कैसे देख सकता हूं?",
|
||||||
|
"faq3Answer": "लॉग इन करने के बाद, आपके खाते की शेष राशि होम स्क्रीन पर प्रदर्शित होगी। आप “खाते” अनुभाग के तहत विस्तृत शेष राशि भी देख सकते हैं।",
|
||||||
|
"faq4Question": "क्या मैं अन्य बैंक खातों में पैसे ट्रांसफर कर सकता हूं?",
|
||||||
|
"faq4Answer": "हां। आप भारत में किसी भी बैंक खाते में धनराशि स्थानांतरित करने के लिए एनईएफटी, आरटीजीएस या आईएमपीएस का उपयोग कर सकते हैं।",
|
||||||
|
"faq5Question": "मैं अपना लेनदेन इतिहास कैसे देखूं?",
|
||||||
|
"faq5Answer": "हाल के और पिछले लेनदेन देखने के लिए होम स्क्रीन के नीचे “खाता विवरण” आइकन पर क्लिक करें।",
|
||||||
|
"chequeEnquiryTitle": "चेक पूछताछ",
|
||||||
|
"chequeEnquirySubtitle": "आप अपनी जारी की गई चेक बुक, प्रस्तुत किए गए चेकों और रोके गए चेकों के विवरण देख सकते हैं, जिसमें प्रासंगिक तिथियां, चेक नं्य आवश्यक जानकारी शामिल है",
|
||||||
|
"stopChequeSubtitle": "अपनी जारी की गई चेकबुक से एक या अधिक चेकों के लिए स्टॉप आरंभ करें। यह आवश्यक सेवा अनधिकृत लेनदेन को रोकने और धोखाधड़ी से बचानद करती है।",
|
||||||
|
"chequeEnquiryFailedError": "चेक स्थिति लाने में विफल: {error}",
|
||||||
|
"accountNumberTitle": "खाता संख्या",
|
||||||
|
"noAccountsFound": "कोई खाता नहीं मिला",
|
||||||
|
"searchByChequeDetailsHint": "चेक विवरण द्वारा खोजें",
|
||||||
|
"noChequeStatusFound": "कोई चेक स्थिति नहीं मिली।",
|
||||||
|
"chequebookIssuedLabel": "चेकबुक जारी (CI)",
|
||||||
|
"branchCodeLabel": "शाखा कोड:",
|
||||||
|
"fromChequeLabel": "चेक से:",
|
||||||
|
"toChequeLabel": "चेक तक:",
|
||||||
|
"dateLabel": "दिनांक:",
|
||||||
|
"chequesCountLabel": "चेकों की संख्या:",
|
||||||
|
"presentedChequeLabel": "प्रस्तुत चेक (PR)",
|
||||||
|
"chequeNumberLabel": "चेक नंबर:",
|
||||||
|
"transactionCodeLabel": "लेनदेन कोड:",
|
||||||
|
"amountLabelWithColon": "राशि:",
|
||||||
|
"statusLabel": "स्थिति:",
|
||||||
|
"stopChequeLabel": "चेक रोकें (ST)",
|
||||||
|
"stopIssueDateLabel": "रोक जारी करने की तारीख:",
|
||||||
|
"stopExpiryDateLabel": "रोक समाप्ति तिथि:",
|
||||||
|
"emptyString": "",
|
||||||
|
"cashCreditAccountLabel": "नकद क्रेडिट खाता",
|
||||||
|
"stopChequeTitle": "चेक रोकें",
|
||||||
|
"noChequebookToStop": "से चेक रोकने के लिए कोई चेक बुक नहीं मिली।",
|
||||||
|
"stopSingleChequeButton": "एकल चेक रोकें",
|
||||||
|
"pleaseSelectAccountFirst": "कृपया पहले एक खाता चुनें।",
|
||||||
|
"stopMultipleChequesButton": "एकाधिक चेक रोकें",
|
||||||
|
"noChequeIssuedStatus": "कोई चेक जारी स्थिति नहीं मिली।",
|
||||||
|
"chequebookDetailsTitle": "चेकबुक विवरण",
|
||||||
|
"customerNameLabel": "ग्राहक का नाम:",
|
||||||
|
"cifNumberLabel": "CIF नंबर:",
|
||||||
|
"startingChequeNumberLabel": "प्रारंभिक चेक नंबर:",
|
||||||
|
"endingChequeNumberLabel": "अंतिम चेक नंबर:",
|
||||||
|
"issueDateLabel": "जारी करने की तारीख:",
|
||||||
|
"numberOfChequesLabel": "चेकों की संख्या:",
|
||||||
|
"instrumentTypeLabel": "उपकरण का प्रकार:",
|
||||||
|
"closeButton": "बंद करें",
|
||||||
|
"stopSingleChequeTitle": "एकल चेक रोकें",
|
||||||
|
"chequeNumberHint": "चेक नंबर *",
|
||||||
|
"pleaseEnterChequeNumberError": "कृपया एक चेक नंबर दर्ज करें",
|
||||||
|
"invalidChequeNumberFormatError": "अमान्य चेक नंबर प्रारूप",
|
||||||
|
"chequeNumberRangeError": "चेक नंबर {from} और {to} के बीच होना चाहिए",
|
||||||
|
"instrumentTypeHint": "उपकरण का प्रकार *",
|
||||||
|
"stopIssueDateHint": "रोक जारी करने की तारीख",
|
||||||
|
"stopExpiryDateHint": "रोक समाप्ति तिथि",
|
||||||
|
"stopAmountHint": "राशि रोकें",
|
||||||
|
"stopCommentHint": "टिप्पणी रोकें",
|
||||||
|
"chequebookIssueDateHint": "चेकबुक जारी करने की तारीख",
|
||||||
|
"successStatus": "सफलता",
|
||||||
|
"errorStatus": "त्रुटि",
|
||||||
|
"incorrectTpinErrorMessage": "आपके द्वारा दर्ज किया गया टीपिन गलत है। कृपया पुन: प्रयास करें।",
|
||||||
|
"internalServerError": "आंतरिक सर्वर त्रुटि",
|
||||||
|
"stopChequeButton": "चेक रोकें",
|
||||||
|
"stopMultipleChequesTitle": "एकाधिक चेक रोकें",
|
||||||
|
"fromChequeNumberHint": "चेक नंबर से *",
|
||||||
|
"toChequeNumberHint": "चेक नंबर तक *"
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -15,7 +15,7 @@ void main() async {
|
|||||||
]);
|
]);
|
||||||
|
|
||||||
// Check for device compromise
|
// Check for device compromise
|
||||||
//final compromisedMessage = await SecurityService.deviceCompromisedMessage;
|
// final compromisedMessage = await SecurityService.deviceCompromisedMessage;
|
||||||
// if (compromisedMessage != null) {
|
// if (compromisedMessage != null) {
|
||||||
// runApp(MaterialApp(
|
// runApp(MaterialApp(
|
||||||
// home: SecurityErrorScreen(message: compromisedMessage),
|
// home: SecurityErrorScreen(message: compromisedMessage),
|
||||||
|
|||||||
@@ -6,6 +6,7 @@ import FlutterMacOS
|
|||||||
import Foundation
|
import Foundation
|
||||||
|
|
||||||
import device_info_plus
|
import device_info_plus
|
||||||
|
import flutter_local_notifications
|
||||||
import flutter_secure_storage_macos
|
import flutter_secure_storage_macos
|
||||||
import local_auth_darwin
|
import local_auth_darwin
|
||||||
import package_info_plus
|
import package_info_plus
|
||||||
@@ -16,6 +17,7 @@ import url_launcher_macos
|
|||||||
|
|
||||||
func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
|
func RegisterGeneratedPlugins(registry: FlutterPluginRegistry) {
|
||||||
DeviceInfoPlusMacosPlugin.register(with: registry.registrar(forPlugin: "DeviceInfoPlusMacosPlugin"))
|
DeviceInfoPlusMacosPlugin.register(with: registry.registrar(forPlugin: "DeviceInfoPlusMacosPlugin"))
|
||||||
|
FlutterLocalNotificationsPlugin.register(with: registry.registrar(forPlugin: "FlutterLocalNotificationsPlugin"))
|
||||||
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"))
|
FPPPackageInfoPlusPlugin.register(with: registry.registrar(forPlugin: "FPPPackageInfoPlusPlugin"))
|
||||||
|
|||||||
56
pubspec.lock
56
pubspec.lock
@@ -137,6 +137,14 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.0.8"
|
version: "1.0.8"
|
||||||
|
dbus:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: dbus
|
||||||
|
sha256: "79e0c23480ff85dc68de79e2cd6334add97e48f7f4865d17686dd6ea81a47e8c"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.7.11"
|
||||||
device_info_plus:
|
device_info_plus:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@@ -238,6 +246,38 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "3.0.2"
|
version: "3.0.2"
|
||||||
|
flutter_local_notifications:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: flutter_local_notifications
|
||||||
|
sha256: "19ffb0a8bb7407875555e5e98d7343a633bb73707bae6c6a5f37c90014077875"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "19.5.0"
|
||||||
|
flutter_local_notifications_linux:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: flutter_local_notifications_linux
|
||||||
|
sha256: e3c277b2daab8e36ac5a6820536668d07e83851aeeb79c446e525a70710770a5
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "6.0.0"
|
||||||
|
flutter_local_notifications_platform_interface:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: flutter_local_notifications_platform_interface
|
||||||
|
sha256: "277d25d960c15674ce78ca97f57d0bae2ee401c844b6ac80fcd972a9c99d09fe"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "9.1.0"
|
||||||
|
flutter_local_notifications_windows:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: flutter_local_notifications_windows
|
||||||
|
sha256: "8d658f0d367c48bd420e7cf2d26655e2d1130147bca1eea917e576ca76668aaf"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "1.0.3"
|
||||||
flutter_localizations:
|
flutter_localizations:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description: flutter
|
description: flutter
|
||||||
@@ -541,6 +581,14 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "1.0.0"
|
version: "1.0.0"
|
||||||
|
open_filex:
|
||||||
|
dependency: "direct main"
|
||||||
|
description:
|
||||||
|
name: open_filex
|
||||||
|
sha256: "9976da61b6a72302cf3b1efbce259200cd40232643a467aac7370addf94d6900"
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "4.7.0"
|
||||||
package_info_plus:
|
package_info_plus:
|
||||||
dependency: "direct main"
|
dependency: "direct main"
|
||||||
description:
|
description:
|
||||||
@@ -874,6 +922,14 @@ packages:
|
|||||||
url: "https://pub.dev"
|
url: "https://pub.dev"
|
||||||
source: hosted
|
source: hosted
|
||||||
version: "0.7.6"
|
version: "0.7.6"
|
||||||
|
timezone:
|
||||||
|
dependency: transitive
|
||||||
|
description:
|
||||||
|
name: timezone
|
||||||
|
sha256: dd14a3b83cfd7cb19e7888f1cbc20f258b8d71b54c06f79ac585f14093a287d1
|
||||||
|
url: "https://pub.dev"
|
||||||
|
source: hosted
|
||||||
|
version: "0.10.1"
|
||||||
typed_data:
|
typed_data:
|
||||||
dependency: transitive
|
dependency: transitive
|
||||||
description:
|
description:
|
||||||
|
|||||||
@@ -61,6 +61,8 @@ dependencies:
|
|||||||
device_info_plus: ^11.3.0
|
device_info_plus: ^11.3.0
|
||||||
showcaseview: ^2.0.3
|
showcaseview: ^2.0.3
|
||||||
package_info_plus: ^4.2.0
|
package_info_plus: ^4.2.0
|
||||||
|
flutter_local_notifications: ^19.5.0
|
||||||
|
open_filex: ^4.7.0
|
||||||
# jailbreak_root_detection: "^1.1.6"
|
# jailbreak_root_detection: "^1.1.6"
|
||||||
|
|
||||||
|
|
||||||
@@ -114,6 +116,8 @@ flutter:
|
|||||||
- assets/images/yes_bank_logo.png
|
- assets/images/yes_bank_logo.png
|
||||||
- assets/images/uco_logo.png
|
- assets/images/uco_logo.png
|
||||||
- assets/images/ipos_logo.png
|
- assets/images/ipos_logo.png
|
||||||
|
- assets/images/profile.svg
|
||||||
|
- assets/images/profile.png
|
||||||
- assets/animations/rupee.json
|
- assets/animations/rupee.json
|
||||||
- assets/animations/error.json
|
- assets/animations/error.json
|
||||||
- assets/animations/done.json
|
- assets/animations/done.json
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ list(APPEND FLUTTER_PLUGIN_LIST
|
|||||||
)
|
)
|
||||||
|
|
||||||
list(APPEND FLUTTER_FFI_PLUGIN_LIST
|
list(APPEND FLUTTER_FFI_PLUGIN_LIST
|
||||||
|
flutter_local_notifications_windows
|
||||||
)
|
)
|
||||||
|
|
||||||
set(PLUGIN_BUNDLED_LIBRARIES)
|
set(PLUGIN_BUNDLED_LIBRARIES)
|
||||||
|
|||||||
Reference in New Issue
Block a user