// ignore_for_file: library_private_types_in_public_api import 'package:flutter/material.dart'; /* */ class LoginPage extends StatefulWidget { const LoginPage({Key? key}) : super(key: key); @override _LoginPageState createState() { return _LoginPageState(); } } /// 登录相关 class _LoginPageState extends State{ // 焦点 final FocusNode _focusNodeUserName = FocusNode(); final FocusNode _focusNodePassWord = FocusNode(); // 用户名输入框控制器,final 此控制器可以监听用户名输入框的操作 final TextEditingController _userNameController = TextEditingController(); // 表单状态 final GlobalKey _formKey = GlobalKey (); // 用户名、密码 String _username = "", _password = ""; // 是否显示密码 bool _isShowPwd = false; // 是否显示输入框尾部的清除按钮 bool _isShowClear = false; /// 插入到渲染树时调用,只执行一次。(类似Android Fragment的onCreateView函数) @override void initState() { // 设置焦点监听 _focusNodeUserName.addListener(_focusNodeListener); _focusNodePassWord.addListener(_focusNodeListener); // 监听用户名框的输入改变 _focusNodeUserName.addListener(() { print("监听用户名框的输入改变 ${_userNameController.text}"); // 监听输入变化,当有内容的时候,显示尾部清除按钮,否则不显示 if (_userNameController.text.isNotEmpty) { _isShowClear = true; } else { _isShowClear = false; } // 调用setState 方法,重新调用build 进行渲染界面 setState(() {}); }); super.initState(); } @override Widget build(BuildContext context) { // 返回按钮 Widget backImageArea = Container( height: 45, alignment: Alignment.centerLeft, margin: const EdgeInsets.only(left: 5, top: 5), child: IconButton( icon: const ImageIcon(AssetImage("assets/images/dot.png")), tooltip: "返回", onPressed: () { print("返回------"); Navigator.pop(context); }, ), ); // logo图片区域 Widget logoImageArea = Container( margin: const EdgeInsets.only(top: 20), alignment: Alignment.topCenter, child: Image.asset( "assets/images/logo.png", width: 226, height: 130, ), ); // 文本输入框区域 Widget inputTextArea = Container( margin: const EdgeInsets.only(left: 20, right: 20), // 文本输入框区域的装饰 decoration: const BoxDecoration( borderRadius: BorderRadius.all(Radius.circular(8)), color: Colors.white), // 使用Form将两个输入框包起来做控制 child: Form( key: _formKey, // Form里面是一个垂直布局 child: Column( mainAxisSize: MainAxisSize.min, // 控件 children: [ // 用户名 TextFormField( controller: _userNameController, // 焦点控制 focusNode: _focusNodeUserName, // 设置键盘类型 keyboardType: TextInputType.number, // 输入框的装饰 decoration: InputDecoration( labelText: "用户名", hintText: "手机号或身份证号", prefixIcon: const Icon(Icons.person), // 尾部添加清除按钮 suffixIcon: (_isShowClear) ? IconButton( icon: const Icon(Icons.clear), onPressed: () { // 清空文本框的内容 _userNameController.clear(); }) : null), // 校验用户名 validator: (value) { if (value == null || value.isEmpty) { return '用户名不能为空!'; } return null; }, // 保持数据 onSaved: (String? value) { _username = value.toString(); }, ), // 间隔 const SizedBox(height: 10), // 密码 TextFormField( focusNode: _focusNodePassWord, decoration: InputDecoration( labelText: "密码", hintText: "请输入密码", prefixIcon: Icon(Icons.lock), // 是否显示密码 suffixIcon: IconButton( icon: Icon( _isShowPwd ? Icons.visibility : Icons.visibility_off), onPressed: () { setState(() { _isShowPwd = !_isShowPwd; }); })), obscureText: !_isShowPwd, // 校验密码 validator: (pwd) { // 正则验证密码 RegExp exp = RegExp("[a-zA-Z0-9_]{6,20}"); if (pwd == null || pwd.isEmpty) { return "密码不能为空"; } else if (!exp.hasMatch(pwd)) { return "请输入正确的6-20位数字、字母或下划线的密码"; } else { return null; } }, // 保存数据 onSaved: (String? value) { _password = value.toString(); }, ) ], ), ), ); // 忘记密码 Widget forgetPwdArea = Container( margin: const EdgeInsets.only(right: 20, top: 5), child: Row( mainAxisSize: MainAxisSize.max, mainAxisAlignment: MainAxisAlignment.end, children: [ TextButton( child: const Text( "忘记密码?", style: TextStyle(color: Colors.black54), ), onPressed: () { print('找回密码'); }) ], ), ); // 登录按钮区域 Widget loginButtonArea = Container( margin: const EdgeInsets.only(left: 20, right: 20), height: 45, child: ElevatedButton( style: ButtonStyle( backgroundColor: MaterialStateProperty.all(Colors.lightGreen), ), onPressed: () { // 点击登录按钮,解除焦点,回收键盘 _focusNodeUserName.unfocus(); _focusNodePassWord.unfocus(); if (_formKey.currentState!.validate()) { // 输入验证通过 _formKey.currentState!.save(); // todo 等录接口相关操作 print("登陆------$_username;$_password"); ScaffoldMessenger.of(context).showSnackBar( SnackBar(content: Text("登陆------$_username;$_password")), ); } else { print("登陆-----noValidate-"); } }, child: const Text( "登录", style: TextStyle(color: Colors.white), )), ); // 组装widget组件,形成界面 return Scaffold( backgroundColor: Colors.white, // 外层添加一个手势,用于点击空白部分,回收键盘 body: GestureDetector( onTap: () { // 点击空白区域,回收键盘 _focusNodeUserName.unfocus(); _focusNodePassWord.unfocus(); }, child: ListView( children: [ backImageArea, logoImageArea, const SizedBox(height: 60), inputTextArea, forgetPwdArea, const SizedBox(height: 14), loginButtonArea ], ), )); } /// 销毁(类似于Android的onDestroy, 在执行Navigator.pop后会调用该办法, 表示组件已销毁;) @override void dispose() { // 移除焦点监听 _focusNodeUserName.removeListener(_focusNodeListener); _focusNodePassWord.removeListener(_focusNodeListener); _userNameController.dispose(); super.dispose(); } // 监听焦点 _focusNodeListener() async { if (_focusNodeUserName.hasFocus) { print("用户名框获取焦点"); // 取消密码框的焦点状态 _focusNodePassWord.unfocus(); } if (_focusNodePassWord.hasFocus) { // 取消用户名框焦点状态 _focusNodeUserName.unfocus(); } } // 验证用户名 String validateUserName(String value) { // 验证手机号 if (value.isEmpty) { return "用户名不能为空!"; } else { return ""; } } //验证密码 String validatePassWord(String? pwd) { // 正则验证密码 RegExp exp = RegExp("[a-zA-Z0-9_]{6,20}"); if (pwd == null || pwd.isEmpty) { return "密码不能为空"; } else if (!exp.hasMatch(pwd)) { return "请输入正确的6-20位数字、字母或下划线的密码"; } else { return ""; } } }