login_page.dart 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355
  1. // ignore_for_file: slash_for_doc_comments
  2. import 'package:cs_resources/constants/color_constants.dart';
  3. import 'package:cs_resources/generated/assets.dart';
  4. import 'package:flutter/cupertino.dart';
  5. import 'package:flutter/gestures.dart';
  6. import 'package:flutter/material.dart';
  7. import 'package:flutter/widgets.dart';
  8. import 'package:get/get.dart';
  9. import 'package:plugin_basic/base/base_stateful_page.dart';
  10. import 'package:plugin_basic/base/base_state.dart';
  11. import 'package:plugin_basic/base/mixin_state_lifecycle.dart';
  12. import 'package:plugin_basic/service/app_config_service.dart';
  13. import 'package:plugin_basic/utils/ext_get_nav.dart';
  14. import 'package:router/path/router_path.dart';
  15. import 'package:shared/utils/device_utils.dart';
  16. import 'package:shared/utils/log_utils.dart';
  17. import 'package:shared/utils/screen_util.dart';
  18. import 'package:widgets/ext/ex_widget.dart';
  19. import 'package:widgets/my_button.dart';
  20. import 'package:widgets/my_load_image.dart';
  21. import 'package:widgets/my_text_field.dart';
  22. import 'package:widgets/my_text_view.dart';
  23. import 'package:widgets/shatter/custom_radio_check.dart';
  24. import 'package:widgets/widget_export.dart';
  25. import 'login_controller.dart';
  26. import 'login_state.dart';
  27. /*
  28. App 登录页面
  29. */
  30. class LoginPage extends BaseStatefulPage<LoginController> {
  31. LoginPage({super.key});
  32. //启动当前页面
  33. static void startInstance() {
  34. return Get.start(RouterPath.authLogin, launchModel: LaunchModel.standard);
  35. }
  36. static void startWithPopAll() {
  37. Get.offAllNamed(RouterPath.authLogin);
  38. }
  39. @override
  40. State<LoginPage> createState() => _LoginPageState();
  41. @override
  42. LoginController createRawController() {
  43. return LoginController();
  44. }
  45. }
  46. /**
  47. * 页面
  48. */
  49. class _LoginPageState extends BaseState<LoginPage, LoginController> with StateLifecycle {
  50. late LoginState state;
  51. @override
  52. void initState() {
  53. super.initState();
  54. state = controller.state;
  55. }
  56. @override
  57. void dispose() {
  58. Get.delete<LoginController>();
  59. super.dispose();
  60. }
  61. @override
  62. Widget build(BuildContext context) {
  63. Log.d("LoginPage Lifecycle - build走了一遍");
  64. return autoCtlGetBuilder(
  65. builder: (controller) {
  66. return Scaffold(
  67. body: Container(
  68. width: double.infinity,
  69. height: double.infinity,
  70. decoration: const BoxDecoration(
  71. gradient: LinearGradient(
  72. colors: [
  73. Color(0xFF091D44),
  74. Color(0xFF245A8A),
  75. Color(0xFF7F7CEC),
  76. ],
  77. begin: Alignment.topCenter,
  78. end: Alignment.bottomCenter,
  79. ),
  80. ),
  81. child: Column(
  82. mainAxisSize: MainAxisSize.max,
  83. crossAxisAlignment: CrossAxisAlignment.center,
  84. children: [
  85. //当前选择的国家,这个不跟随滚动
  86. Align(
  87. alignment: Alignment.topRight,
  88. child: Row(
  89. mainAxisSize: MainAxisSize.min,
  90. crossAxisAlignment: CrossAxisAlignment.center,
  91. children: [
  92. Obx(() {
  93. return MyTextView(
  94. ConfigService.to.curSelectCountry == 0 ? "Vietnam".tr : "Singapore".tr,
  95. textColor: ColorConstants.white,
  96. isFontMedium: true,
  97. marginRight: 5,
  98. fontSize: 15,
  99. );
  100. }),
  101. const MyAssetImage(
  102. Assets.baseServiceTriangleDropDown,
  103. width: 8,
  104. height: 5,
  105. ),
  106. ],
  107. ).onTap(() => controller.gotoSelectCountryPage()).marginOnly(right: 20, top: ScreenUtil.getStatusBarH(context) + 20),
  108. ),
  109. // 底部滚动的布局
  110. SingleChildScrollView(
  111. scrollDirection: Axis.vertical,
  112. physics: const BouncingScrollPhysics(),
  113. child: Column(
  114. mainAxisSize: MainAxisSize.max,
  115. crossAxisAlignment: CrossAxisAlignment.center,
  116. children: [
  117. MyTextView(
  118. "Casual Labour System",
  119. textColor: ColorConstants.white,
  120. isFontBold: true,
  121. fontSize: 35,
  122. marginTop: 20,
  123. fontStyle: FontStyle.italic,
  124. textAlign: TextAlign.center,
  125. marginLeft: 50,
  126. marginRight: 50,
  127. ),
  128. //中间的输入框布局
  129. Container(
  130. width: double.infinity,
  131. margin: EdgeInsets.symmetric(vertical: 45, horizontal: 15),
  132. padding: EdgeInsets.symmetric(vertical: 33, horizontal: 20),
  133. decoration: BoxDecoration(
  134. color: Color(0xFF4DCFF6).withOpacity(0.2), // 设置背景颜色和不透明度
  135. borderRadius: BorderRadius.circular(5.0), // 设置圆角
  136. ),
  137. child: Center(
  138. child: Column(
  139. crossAxisAlignment: CrossAxisAlignment.start,
  140. children: [
  141. //登录文本
  142. MyTextView(
  143. "Email".tr,
  144. isFontMedium: true,
  145. fontSize: 17,
  146. textColor: ColorConstants.white,
  147. ),
  148. // 登录表单 - 账号
  149. _buildInputLayout(
  150. "code",
  151. textInputAction: TextInputAction.next,
  152. errorText: state.codeErrorText,
  153. onSubmit: (formKey, value) {
  154. state.formData[formKey]!['focusNode'].unfocus();
  155. FocusScope.of(context).requestFocus(state.formData['password']!['focusNode']);
  156. },
  157. ),
  158. //密码文本
  159. MyTextView(
  160. "Password".tr,
  161. isFontMedium: true,
  162. fontSize: 17,
  163. textColor: ColorConstants.white,
  164. marginTop: 18,
  165. ),
  166. // 登录表单 - 密码
  167. _buildInputLayout(
  168. "password",
  169. obscureText: !state.pwdVisibility,
  170. errorText: state.passwordErrorText,
  171. showRightIcon: true,
  172. rightWidget: IconButton(
  173. highlightColor: Colors.transparent,
  174. splashColor: Colors.transparent,
  175. icon: state.pwdVisibility
  176. ? const MyAssetImage(
  177. Assets.cptAuthPasswordShowIcon,
  178. width: 21,
  179. height: 21,
  180. color: ColorConstants.white,
  181. )
  182. : const MyAssetImage(
  183. Assets.cptAuthPasswordHideIcon,
  184. width: 21,
  185. height: 21,
  186. color: ColorConstants.white,
  187. ),
  188. onPressed: () {
  189. controller.switchPwdVisibility();
  190. },
  191. ),
  192. onSubmit: (formKey, value) {
  193. state.formData[formKey]!['focusNode'].unfocus();
  194. controller.doInputLogin();
  195. },
  196. ),
  197. //选择签到功能还是全功能
  198. Obx(() {
  199. return Visibility(
  200. visible: ConfigService.to.curSelectCountry == 0,
  201. child: CustomRadioCheck(
  202. options: state.loginOption,
  203. onOptionSelected: (index, text) {
  204. state.loginOptionPosition = index;
  205. },
  206. ).marginOnly(top: 20),
  207. );
  208. }),
  209. //登录按钮
  210. MyButton(
  211. type: ClickType.throttle,
  212. milliseconds: 500,
  213. onPressed: () {
  214. FocusScope.of(context).unfocus();
  215. controller.doInputLogin();
  216. },
  217. text: "Log in".tr,
  218. textColor: ColorConstants.white,
  219. fontSize: 16,
  220. radius: 22.5,
  221. backgroundColor: hexToColor("#FFBB1B"),
  222. fontWeight: FontWeight.w500,
  223. ).marginOnly(top: 20),
  224. //注册入口
  225. Visibility(
  226. visible: !DeviceUtils.isAndroid,
  227. child: _buildRichText(),
  228. ),
  229. ],
  230. ),
  231. ), // 传递子部件
  232. ),
  233. ],
  234. )).expanded(),
  235. ],
  236. ),
  237. ),
  238. );
  239. },
  240. );
  241. }
  242. @override
  243. void onStart() {
  244. Log.d("LoginPage Lifecycle - onStart");
  245. }
  246. @override
  247. void onStop() {
  248. Log.d("LoginPage Lifecycle - onStop");
  249. }
  250. @override
  251. void onResume() {
  252. Log.d("LoginPage Lifecycle - onResume");
  253. }
  254. @override
  255. void onPause() {
  256. Log.d("LoginPage Lifecycle - onPause");
  257. }
  258. //底部的注册文本
  259. Widget _buildRichText() {
  260. return Align(
  261. alignment: Alignment.center,
  262. child: RichText(
  263. text: TextSpan(
  264. style: TextStyle(fontSize: 14.0, fontWeight: FontWeight.w500, color: ColorConstants.textGrayAECAE5),
  265. children: <TextSpan>[
  266. TextSpan(
  267. text: "Don’t have an account?".tr,
  268. style: TextStyle(color: ColorConstants.textGrayAECAE5),
  269. ),
  270. TextSpan(
  271. text: "Sign up".tr,
  272. style: TextStyle(color: ColorConstants.textYellowFFBB1B),
  273. recognizer: TapGestureRecognizer()
  274. ..onTap = () {
  275. //去登录页面
  276. controller.gotoSignUpPage();
  277. },
  278. ),
  279. ],
  280. ),
  281. ),
  282. ).marginOnly(top: 23);
  283. }
  284. /// 输入框 账号与密码
  285. Widget _buildInputLayout(
  286. String key, {
  287. double marginTop = 0,
  288. bool? showRightIcon = false, //是否展示右侧的布局
  289. Widget? rightWidget, //右侧的布局
  290. TextInputType textInputType = TextInputType.text,
  291. String? errorText,
  292. bool obscureText = false,
  293. TextInputAction textInputAction = TextInputAction.done,
  294. Function? onSubmit,
  295. }) {
  296. return IgnoreKeyboardDismiss(
  297. child: MyTextField(
  298. key,
  299. state.formData[key]!['value'],
  300. hintText: state.formData[key]!['hintText'],
  301. hintStyle: TextStyle(
  302. color: hexToColor("#AECAE5"),
  303. fontSize: 15.0,
  304. fontWeight: FontWeight.w500,
  305. ),
  306. controller: state.formData[key]!['controller'],
  307. focusNode: state.formData[key]!['focusNode'],
  308. margin: EdgeInsets.only(top: marginTop),
  309. showDivider: true,
  310. dividerColor: hexToColor("#7BABC8"),
  311. style: TextStyle(
  312. color: ColorConstants.white,
  313. fontSize: 15.0,
  314. fontWeight: FontWeight.w500,
  315. ),
  316. inputType: textInputType,
  317. height: 35,
  318. textInputAction: textInputAction,
  319. onSubmit: onSubmit,
  320. cursorColor: ColorConstants.white,
  321. obscureText: obscureText,
  322. errorText: errorText,
  323. showLeftIcon: true,
  324. showRightIcon: showRightIcon,
  325. rightWidget: rightWidget,
  326. ),
  327. );
  328. }
  329. }