import 'dart:async'; import 'dart:math' as math; import 'package:flutter/rendering.dart'; import 'package:logging/logging.dart'; import 'package:nc_photos/int_util.dart'; /// Throttle how many times an event could be triggered /// /// Events can be filtered by 2 ways: /// 1. Time passed after the last event /// 2. Number of events class Throttler { Throttler({ required this.onTriggered, this.logTag, }); /// Post an event /// /// [data] can be used to provide optional data. When the stream of events /// eventually trigger, a list of all data received will be passed to the /// callback. Nulls are ignored void trigger({ Duration maxResponceTime = const Duration(seconds: 1), int? maxPendingCount, T? data, }) { _count += 1; if (data != null) { _data.add(data); } _subscription?.cancel(); _subscription = null; if (maxPendingCount != null) { _maxCount = math.min(maxPendingCount, _maxCount ?? int32Max); } if (_maxCount != null && _count >= _maxCount!) { _log.info("[trigger]$_logTag Triggered after $_count events"); _doTrigger(); } else { final responseTime = _minDuration( maxResponceTime, _currentResponseTime ?? Duration(days: 1)); _subscription = Future.delayed(responseTime).asStream().listen((event) { _log.info("[trigger]$_logTag Triggered after $responseTime"); _doTrigger(); }); _currentResponseTime = responseTime; } } /// Drop all pending triggers, this may be useful in places like [dispose] void clear() { _subscription?.cancel(); _subscription = null; _currentResponseTime = null; _count = 0; _maxCount = null; _data = []; } void _doTrigger() { onTriggered?.call(_data); clear(); } String get _logTag => logTag == null ? "" : "[$logTag]"; final ValueChanged>? onTriggered; /// Extra tag printed with logs from this class final String? logTag; StreamSubscription? _subscription; Duration? _currentResponseTime; int _count = 0; int? _maxCount; var _data = []; static final _log = Logger("throttler.Throttler"); } Duration _minDuration(Duration a, Duration b) { return a.compareTo(b) < 0 ? a : b; }