Yimaru-Mobile/lib/ui/views/login/screens/login_otp_screen.dart

158 lines
5.4 KiB
Dart

import 'package:flutter/material.dart';
import 'package:flutter_timer_countdown/flutter_timer_countdown.dart';
import 'package:pinput/pinput.dart';
import 'package:stacked/stacked.dart';
import 'package:yimaru_app/ui/widgets/custom_cursor.dart';
import '../../../common/app_colors.dart';
import '../../../common/ui_helpers.dart';
import '../../../widgets/custom_elevated_button.dart';
import '../login_viewmodel.dart';
import '../login_view.form.dart';
class LoginOtpScreen extends ViewModelWidget<LoginViewModel> {
final TextEditingController otpController;
final TextEditingController phoneNumberController;
const LoginOtpScreen(
{super.key,
required this.otpController,
required this.phoneNumberController});
@override
Widget build(BuildContext context, LoginViewModel viewModel) =>
_buildBody(viewModel);
Widget _buildBody(LoginViewModel viewModel) => Column(
crossAxisAlignment: CrossAxisAlignment.start,
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: _buildBodyChildren(viewModel),
);
List<Widget> _buildBodyChildren(LoginViewModel viewModel) =>
[_buildColumnScroller(viewModel), _buildContinueButtonWrapper(viewModel)];
Widget _buildColumnScroller(LoginViewModel viewModel) =>
SingleChildScrollView(
child: _buildUpperColumn(viewModel),
);
Widget _buildUpperColumn(LoginViewModel viewModel) => Column(
mainAxisSize: MainAxisSize.min,
crossAxisAlignment: CrossAxisAlignment.start,
children: _buildUpperColumnChildren(viewModel),
);
List<Widget> _buildUpperColumnChildren(LoginViewModel viewModel) => [
verticalSpaceMedium,
_buildTitle(),
verticalSpaceMedium,
_buildSubtitleWrapper(),
verticalSpaceMedium,
_buildPinPutWrapper(viewModel),
if (viewModel.hasOtpValidationMessage && viewModel.focusOtp)
verticalSpaceTiny,
if (viewModel.hasOtpValidationMessage && viewModel.focusOtp)
_buildOtpValidatorWrapper(viewModel),
verticalSpaceSmall,
_buildTimerWrapper(viewModel)
];
Widget _buildTitle() => Text(
'Verification Code',
style: style25DG600,
);
Widget _buildSubtitleWrapper() =>
phoneNumberController.text.length == 9 ? _buildSubtitle() : Container();
Widget _buildSubtitle() => Text(
'Code sent to your number +251${phoneNumberController.text.substring(0, 5)}****',
style: style14DG400,
);
Widget _buildPinPutWrapper(LoginViewModel viewModel) => Center(
child: _buildPinPut(viewModel),
);
Widget _buildPinPut(LoginViewModel viewModel) => Pinput(
controller: otpController,
defaultPinTheme: defaultPin,
cursor: const CustomCursor(),
errorPinTheme: errorPinTheme,
onTap: viewModel.setOtpFocus,
focusNode: viewModel.focusNode,
errorTextStyle: validationStyle,
//smsRetriever: locator<KewedeSmsRetriever>(),
focusedPinTheme: focusedThemePin,
submittedPinTheme: submittedThemePin,
hapticFeedbackType: HapticFeedbackType.heavyImpact,
separatorBuilder: (index) => const SizedBox(width: 25),
onCompleted: (otp) async => await viewModel.validateOtp(otp),
);
Widget _buildOtpValidatorWrapper(LoginViewModel viewModel) =>
viewModel.hasOtpValidationMessage
? _buildOtpValidator(viewModel)
: Container();
Widget _buildOtpValidator(LoginViewModel viewModel) => Text(
viewModel.otpValidationMessage!,
style: style12R700,
);
Widget _buildTimerWrapper(LoginViewModel viewModel) => !viewModel.buttonActive
? _buildTimerSection(viewModel)
: _buildResendButton();
Widget _buildResendButton() =>
TextButton(onPressed: () {}, child: _buildResendText());
Widget _buildResendText() => Text(
'Resend code',
style: style14P600.copyWith(fontStyle: FontStyle.italic),
);
Widget _buildTimerSection(LoginViewModel viewModel) => Row(
children: [
_buildCountdownText(),
horizontalSpaceSmall,
_buildTimer(viewModel)
],
);
Widget _buildCountdownText() => Text('Resend code in ', style: style14DG400);
Widget _buildTimer(LoginViewModel viewModel) => TimerCountdown(
enableDescriptions: false,
timeTextStyle: style14P600,
endTime: DateTime.now().add(const Duration(minutes: 3, seconds: 0)),
onEnd: viewModel.setResendButton,
format: CountDownTimerFormat.minutesSeconds,
colonsTextStyle: const TextStyle(color: kcPrimaryColor),
);
Widget _buildContinueButtonWrapper(LoginViewModel viewModel) => Padding(
padding: const EdgeInsets.only(bottom: 50),
child: _buildContinueButton(viewModel),
);
Widget _buildContinueButton(LoginViewModel viewModel) => CustomElevatedButton(
height: 55,
text: 'Continue',
borderRadius: 12,
foregroundColor: kcWhite,
backgroundColor: viewModel.focusOtp &&
otpController.text.length == 4 &&
!viewModel.hasOtpValidationMessage
? kcPrimaryColor
: kcPrimaryColor.withOpacity(0.1),
onTap: viewModel.focusOtp &&
otpController.text.length == 4 &&
!viewModel.hasOtpValidationMessage
? () => viewModel.replaceWithHome()
: null,
);
}