2023-08-27 05:07:35 +00:00
|
|
|
import 'dart:async';
|
|
|
|
|
2024-01-15 15:26:13 +00:00
|
|
|
import 'package:flutter_hooks/flutter_hooks.dart';
|
2023-08-27 05:07:35 +00:00
|
|
|
|
2024-01-15 15:26:13 +00:00
|
|
|
/// Used to debounce function calls with the [interval] provided.
|
|
|
|
class Debouncer {
|
|
|
|
Debouncer({required this.interval});
|
|
|
|
final Duration interval;
|
2023-08-27 05:07:35 +00:00
|
|
|
Timer? _timer;
|
2024-01-15 15:26:13 +00:00
|
|
|
FutureOr<void> Function()? _lastAction;
|
2023-08-27 05:07:35 +00:00
|
|
|
|
2024-01-15 15:26:13 +00:00
|
|
|
void run(FutureOr<void> Function() action) {
|
|
|
|
_lastAction = action;
|
2023-08-27 05:07:35 +00:00
|
|
|
_timer?.cancel();
|
2024-01-15 15:26:13 +00:00
|
|
|
_timer = Timer(interval, _callAndRest);
|
2023-08-27 05:07:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void _callAndRest() {
|
2024-01-15 15:26:13 +00:00
|
|
|
_lastAction?.call();
|
2023-08-27 05:07:35 +00:00
|
|
|
_timer = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
void dispose() {
|
|
|
|
_timer?.cancel();
|
|
|
|
_timer = null;
|
2024-01-15 15:26:13 +00:00
|
|
|
_lastAction = null;
|
2023-08-27 05:07:35 +00:00
|
|
|
}
|
|
|
|
}
|
2024-01-15 15:26:13 +00:00
|
|
|
|
|
|
|
/// Creates a [Debouncer] that will be disposed automatically. If no [interval] is provided, a
|
|
|
|
/// default interval of 300ms is used to debounce the function calls
|
|
|
|
Debouncer useDebouncer({
|
|
|
|
Duration interval = const Duration(milliseconds: 300),
|
|
|
|
List<Object?>? keys,
|
|
|
|
}) =>
|
|
|
|
use(_DebouncerHook(interval: interval, keys: keys));
|
|
|
|
|
|
|
|
class _DebouncerHook extends Hook<Debouncer> {
|
|
|
|
const _DebouncerHook({
|
|
|
|
required this.interval,
|
|
|
|
List<Object?>? keys,
|
|
|
|
}) : super(keys: keys);
|
|
|
|
|
|
|
|
final Duration interval;
|
|
|
|
|
|
|
|
@override
|
|
|
|
HookState<Debouncer, Hook<Debouncer>> createState() => _DebouncerHookState();
|
|
|
|
}
|
|
|
|
|
|
|
|
class _DebouncerHookState extends HookState<Debouncer, _DebouncerHook> {
|
|
|
|
late final debouncer = Debouncer(interval: hook.interval);
|
|
|
|
|
|
|
|
@override
|
|
|
|
Debouncer build(_) => debouncer;
|
|
|
|
|
|
|
|
@override
|
|
|
|
void dispose() => debouncer.dispose();
|
|
|
|
|
|
|
|
@override
|
|
|
|
String get debugLabel => 'useDebouncer';
|
|
|
|
}
|