Small knowledge, big challenge! This article is participating in the creation activity of “Essential Tips for Programmers”.

This article has participated in the “Digitalstar Project” and won a creative gift package to challenge the creative incentive money.

In the process of learning Flutter, there may be countdown and other requirements. At this time, a Timer is needed.

Timer

Timer, as an abstract class, provides two factory methods to call; Can be used as a single or repeated trigger countdown timer;

Case try

1. Timer()

factory Timer(Duration duration, void Function() callback) {
    if (Zone.current == Zone.root) {
      return Zone.current.createTimer(duration, callback);
    }
    return Zone.current.createTimer(duration, Zone.current.bindCallbackGuarded(callback));
}
Copy the code

Timer allows the execution after the specified delay time, Timeout is set to specify the delay time, and then callback is used to listen on the execution result.

Timer(Duration(seconds: 3), () { print("_timer01() -> Timer(Duration(seconds: 3) printed after 3 seconds"); Toast.show('Timer(Duration(seconds: 3) printed after 3 seconds ! ', context, duration: Toast.LENGTH_SHORT, gravity: Toast.BOTTOM); });Copy the code

2. Timer.periodic()

factory Timer.periodic(Duration duration, void callback(Timer timer)) {
    if (Zone.current == Zone.root) {
      return Zone.current.createPeriodicTimer(duration, callback);
    }
    var boundCallback = Zone.current.bindUnaryCallbackGuarded<Timer>(callback);
    return Zone.current.createPeriodicTimer(duration, boundCallback);
}
Copy the code

We can get from the named constructor timer.periodic (), which performs the Timer countdown again through the periodic binding callback.

Timer. Periodic () can countdown repeatedly and periodically. If you do not call it manually, it will always close.

In the command, timer. tick indicates the number of calls, and timer. isActive indicates whether the current Timer isActive.

Timer.periodic(Duration(seconds: 2), (timer) { print('_timer02() -> Timer.periodic() -> Timer.tick -> ${timer.tick} -> Timer.isActive -> ${timer.isActive}');  });Copy the code

3. Timer.run()

static void run(void Function() callback) {
    new Timer(Duration.zero, callback);
}
Copy the code

The execution of a Timer provides a convenient time.run () named constructor for Flutter to be executed as quickly as possible. This can be simply interpreted as counting down to 0.

The small dish tries the following sequence, executing synchronous A -> B -> C first, followed by asynchronous run() and duration.zero;

print('_timer03() -> A');
Timer.run(() {
  print('_timer03() -> Timer.run()');
});
print('_timer03() -> B');
Timer(Duration.zero, () {
  print('_timer03() -> Duration.zero');
});
print('_timer03() -> C');
Copy the code

4. cancel()

The Timer() Timer can be cancelled by cancel(), especially in the case of periodic timer.periodic () calls, which need to be cancelled at appropriate times; Try canceling inside the Timer() callback and calling outside the method.

4.1 Cancel in Timer() callback
Timer.periodic(Duration(seconds: 2), (timer) {
  if (timer.tick == 3) {
    timer.cancel();
  }
  print('_timer04() -> Timer.periodic() -> Timer.tick -> ${timer.tick} -> Timer.isActive -> ${timer.isActive}');
});
Copy the code

4.2 Out-of-method Cancellation
_timer = Timer.periodic(Duration(seconds: 2), (timer) { print('_timer05() -> Timer.periodic() -> Timer.tick -> ${timer.tick} -> Timer.isActive -> ${timer.isActive}');  }); _timer.cancel();Copy the code

ACETimerButton Custom timer

Side dish tried a simple timer, similar to a button to get a captcha; Timeout is the countdown time, color is used to customize the text color, and preName is the text content.

ACETimerButton(this.timeout, {this.color, this.preName});
Copy the code

The whole definition process is simple, just need to update the button text when the button is clicked and calculate the Timer periodic countdown, and cancel and destroy the Timer when the countdown ends and the Widget is destroyed.

class ACETimerButton extends StatefulWidget { final int timeout; final Color color; final String preName; ACETimerButton(this.timeout, {this.color, this.preName}); @override _ACETimerButtonState createState() => _ACETimerButtonState(); } class _ACETimerButtonState extends State<ACETimerButton> { Timer _timer; int _timeOut; String _name; bool _isClick = false; @override Widget build(BuildContext context) => GestureDetector( onTap: () { if (! _isClick) { _startTimer(); } _isClick = true; }, child: Container(padding: EdgeInsets. All (6.0), decoration: BoxDecoration(color: RGB (50, 50, 50)) Color.grey.withopacity (0.2), borderRadius: borderRadius. Circular (4.0)), child: Center(child: Text(_name?? 'click', style: TextStyle(color: widget.color ?? Colors. Blue, fontSize: 16.0))))); @override void initState() { super.initState(); _name = widget.preName; _timeOut = widget.timeout; } _startTimer() { _timer = Timer.periodic(const Duration(seconds: 1), (timer) { setState(() { _timeOut--; _name = '$_timeOut s'; }); if (_timeOut == 0) { _cancelTimer(); _isClick = false; _name = widget.preName; _timeOut = widget.timeout; }}); } _cancelTimer() { if (_timer ! = null) { _timer.cancel(); _timer = null; } _isClick = false; } @override void dispose() { super.dispose(); _cancelTimer(); }}Copy the code


Timer case source code


The learning of Timer has come to an end, and the ACETimerButton custom Timer button is not perfect enough. If there are mistakes, please give more guidance!

Source: Little Monk A Ce