Flutter布局锦囊---验证码倒计时

设计给的效果如下:

UI布局图

拿到设计后,先把整体拆分成几个部分:

  1. “获取验证码按钮”,可以通过点击按钮来获取验证码。

然后就可以开始进行编码了。

第1步:绘制组件树

验证码倒计时的组件树

第2步:实现“获取验证码按钮”

获取验证码的按钮需要显示在文本字段的上面,所以你可以直接使用Flutter的墨水瓶(InkWell)组件,一个可以响应触摸的矩形区域。然后通过dart:async库的计时器(Timer)类,使其控制墨水瓶(InkWell)组件的状态。

import 'dart:async';
import 'package:flutter/material.dart';

/// 墨水瓶(`InkWell`)可用时使用的字体样式。
final TextStyle _availableStyle = TextStyle(
  fontSize: 16.0,
  color: const Color(0xFF00CACE),
);

/// 墨水瓶(`InkWell`)不可用时使用的样式。
final TextStyle _unavailableStyle = TextStyle(
  fontSize: 16.0,
  color: const Color(0xFFCCCCCC),
);

class LoginFormCode extends StatefulWidget {
  /// 倒计时的秒数,默认60秒。
  final int countdown;
  /// 用户点击时的回调函数。
  final Function onTapCallback;
  /// 是否可以获取验证码,默认为`false`。
  final bool available;

  LoginFormCode({
    this.countdown: 60,
    this.onTapCallback,
    this.available: false,
  });

  @override
  _LoginFormCodeState createState() => _LoginFormCodeState();
}

class _LoginFormCodeState extends State {
  /// 倒计时的计时器。
  Timer _timer;
  /// 当前倒计时的秒数。
  int _seconds;
  /// 当前墨水瓶(`InkWell`)的字体样式。
  TextStyle inkWellStyle = _availableStyle;
  /// 当前墨水瓶(`InkWell`)的文本。
  String _verifyStr = '获取验证码';

  @override
  void initState() {
    super.initState();
    _seconds = widget.countdown;
  }

  /// 启动倒计时的计时器。
  void _startTimer() {
    // 计时器(`Timer`)组件的定期(`periodic`)构造函数,创建一个新的重复计时器。
    _timer = Timer.periodic(
      Duration(seconds: 1),
      (timer) {
      if (_seconds == 0) {
        _cancelTimer();
        _seconds = widget.countdown;
        inkWellStyle = _availableStyle;
        setState(() {});
        return;
      }
      _seconds--;
      _verifyStr = '已发送$_seconds'+'s';
      setState(() {});
      if (_seconds == 0) {
         _verifyStr = '重新发送';
      }
    });
  }

  /// 取消倒计时的计时器。
  void _cancelTimer() {
    // 计时器(`Timer`)组件的取消(`cancel`)方法,取消计时器。
    _timer?.cancel();
  }

  @override
  Widget build(BuildContext context) {
    // 墨水瓶(`InkWell`)组件,响应触摸的矩形区域。
    return widget.available ? InkWell(
      child: Text(
        '  $_verifyStr  ',
        style: inkWellStyle,
      ),
      onTap: (_seconds == widget.countdown) ? () {
        _startTimer();
        inkWellStyle = _unavailableStyle;
        _verifyStr = '已发送$_seconds'+'s';
        setState(() {});
        widget.onTapCallback();
      } : null,
    ): InkWell(
      child: Text(
        '  获取验证码  ',
        style: _unavailableStyle,
      ),
    );
  }
}

第3步:还原效果

验证码倒计时的还原效果

你可能感兴趣的:(Flutter布局锦囊---验证码倒计时)