Feature_MPIN_OTP #46
@ -82,7 +82,7 @@ class LoginController extends MyController {
|
|||||||
final bool isMpinEnabled = LocalStorage.getIsMpin();
|
final bool isMpinEnabled = LocalStorage.getIsMpin();
|
||||||
|
|
||||||
if (isMpinEnabled) {
|
if (isMpinEnabled) {
|
||||||
Get.toNamed('/auth/mpin-auth');
|
Get.offAllNamed('/home');
|
||||||
} else {
|
} else {
|
||||||
Get.offAllNamed('/home');
|
Get.offAllNamed('/home');
|
||||||
}
|
}
|
||||||
@ -90,8 +90,10 @@ class LoginController extends MyController {
|
|||||||
|
|
||||||
Future<void> _handleRememberMe() async {
|
Future<void> _handleRememberMe() async {
|
||||||
if (isChecked.value) {
|
if (isChecked.value) {
|
||||||
await LocalStorage.setToken('username', basicValidator.getController('username')!.text);
|
await LocalStorage.setToken(
|
||||||
await LocalStorage.setToken('password', basicValidator.getController('password')!.text);
|
'username', basicValidator.getController('username')!.text);
|
||||||
|
await LocalStorage.setToken(
|
||||||
|
'password', basicValidator.getController('password')!.text);
|
||||||
await LocalStorage.setBool('remember_me', true);
|
await LocalStorage.setBool('remember_me', true);
|
||||||
} else {
|
} else {
|
||||||
await LocalStorage.removeToken('username');
|
await LocalStorage.removeToken('username');
|
||||||
|
@ -20,7 +20,7 @@ class MPINController extends GetxController {
|
|||||||
|
|
||||||
final retypeControllers = List.generate(6, (_) => TextEditingController());
|
final retypeControllers = List.generate(6, (_) => TextEditingController());
|
||||||
final retypeFocusNodes = List.generate(6, (_) => FocusNode());
|
final retypeFocusNodes = List.generate(6, (_) => FocusNode());
|
||||||
|
final RxInt failedAttempts = 0.obs;
|
||||||
@override
|
@override
|
||||||
void onInit() {
|
void onInit() {
|
||||||
super.onInit();
|
super.onInit();
|
||||||
@ -77,7 +77,12 @@ class MPINController extends GetxController {
|
|||||||
|
|
||||||
if (success) {
|
if (success) {
|
||||||
logger.i("[MPINController] MPIN generation successful.");
|
logger.i("[MPINController] MPIN generation successful.");
|
||||||
_navigateToDashboard(message: "MPIN Set Successfully");
|
showAppSnackbar(
|
||||||
|
title: "Success",
|
||||||
|
message: "MPIN generated successfully. Please login again.",
|
||||||
|
type: SnackbarType.success,
|
||||||
|
);
|
||||||
|
await LocalStorage.logout();
|
||||||
} else {
|
} else {
|
||||||
logger.w("[MPINController] MPIN generation failed.");
|
logger.w("[MPINController] MPIN generation failed.");
|
||||||
clearFields();
|
clearFields();
|
||||||
@ -186,9 +191,15 @@ class MPINController extends GetxController {
|
|||||||
isLoading.value = false;
|
isLoading.value = false;
|
||||||
|
|
||||||
if (response == null) {
|
if (response == null) {
|
||||||
logger.i(
|
logger.i("[MPINController] MPIN generated successfully");
|
||||||
"[MPINController] MPIN generated successfully (null response treated as success)");
|
|
||||||
Get.toNamed('/auth/mpin-auth');
|
showAppSnackbar(
|
||||||
|
title: "Success",
|
||||||
|
message: "MPIN generated successfully. Please login again.",
|
||||||
|
type: SnackbarType.success,
|
||||||
|
);
|
||||||
|
|
||||||
|
await LocalStorage.logout();
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
} else {
|
} else {
|
||||||
@ -261,6 +272,7 @@ class MPINController extends GetxController {
|
|||||||
type: SnackbarType.error,
|
type: SnackbarType.error,
|
||||||
);
|
);
|
||||||
clearFields();
|
clearFields();
|
||||||
|
onInvalidMPIN();
|
||||||
}
|
}
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
isLoading.value = false;
|
isLoading.value = false;
|
||||||
@ -273,4 +285,15 @@ class MPINController extends GetxController {
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void onInvalidMPIN() {
|
||||||
|
failedAttempts.value++;
|
||||||
|
if (failedAttempts.value >= 3) {
|
||||||
|
showAppSnackbar(
|
||||||
|
title: "Error",
|
||||||
|
message: "Too many failed attempts. Consider logging in again.",
|
||||||
|
type: SnackbarType.error,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -125,7 +125,7 @@ class OTPController extends GetxController {
|
|||||||
print('MPIN Enabled? $isMpinEnabled');
|
print('MPIN Enabled? $isMpinEnabled');
|
||||||
|
|
||||||
if (isMpinEnabled) {
|
if (isMpinEnabled) {
|
||||||
Get.toNamed('/auth/mpin-auth');
|
Get.offAllNamed('/home');
|
||||||
} else {
|
} else {
|
||||||
Get.offAllNamed('/home');
|
Get.offAllNamed('/home');
|
||||||
}
|
}
|
||||||
|
@ -277,27 +277,20 @@ class _MPINAuthScreenState extends State<MPINAuthScreen> with UIMixin {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Widget _buildFooterOptions(MPINController controller, bool isNewUser) {
|
Widget _buildFooterOptions(MPINController controller, bool isNewUser) {
|
||||||
|
return Obx(() {
|
||||||
|
final showBackToLogin =
|
||||||
|
controller.failedAttempts.value >= 3 && !isNewUser;
|
||||||
|
|
||||||
return Column(
|
return Column(
|
||||||
children: [
|
children: [
|
||||||
if (isNewUser)
|
|
||||||
TextButton.icon(
|
|
||||||
onPressed: controller.switchToEnterMPIN,
|
|
||||||
icon: const Icon(Icons.arrow_back, size: 18, color: Colors.black87),
|
|
||||||
label: MyText.bodyMedium(
|
|
||||||
'Back to Enter MPIN',
|
|
||||||
color: Colors.black87,
|
|
||||||
fontWeight: 600,
|
|
||||||
fontSize: 14,
|
|
||||||
),
|
|
||||||
),
|
|
||||||
if (isNewUser)
|
if (isNewUser)
|
||||||
TextButton.icon(
|
TextButton.icon(
|
||||||
onPressed: () async {
|
onPressed: () async {
|
||||||
Get.delete<MPINController>();
|
Get.delete<MPINController>();
|
||||||
Get.toNamed('/dashboard');
|
Get.toNamed('/dashboard');
|
||||||
},
|
},
|
||||||
icon:
|
icon: const Icon(Icons.arrow_back,
|
||||||
const Icon(Icons.arrow_back, size: 18, color: Colors.redAccent),
|
size: 18, color: Colors.redAccent),
|
||||||
label: MyText.bodyMedium(
|
label: MyText.bodyMedium(
|
||||||
'Back to Home Page',
|
'Back to Home Page',
|
||||||
color: contentTheme.brandRed,
|
color: contentTheme.brandRed,
|
||||||
@ -305,7 +298,23 @@ class _MPINAuthScreenState extends State<MPINAuthScreen> with UIMixin {
|
|||||||
fontSize: 14,
|
fontSize: 14,
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
if (showBackToLogin)
|
||||||
|
TextButton.icon(
|
||||||
|
onPressed: () async {
|
||||||
|
Get.delete<MPINController>();
|
||||||
|
await LocalStorage.logout();
|
||||||
|
},
|
||||||
|
icon: const Icon(Icons.arrow_back,
|
||||||
|
size: 18, color: Colors.redAccent),
|
||||||
|
label: MyText.bodyMedium(
|
||||||
|
'Go back to Login Screen',
|
||||||
|
color: contentTheme.brandRed,
|
||||||
|
fontWeight: 600,
|
||||||
|
fontSize: 14,
|
||||||
|
),
|
||||||
|
),
|
||||||
],
|
],
|
||||||
);
|
);
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user