|
|
|
|
@ -1,3 +1,48 @@
|
|
|
|
|
//! A generic event debouncer, for grouping events which occur close together in time into a single
|
|
|
|
|
//! event.
|
|
|
|
|
//!
|
|
|
|
|
//! For example, you may have some
|
|
|
|
|
//! [templates](https://www.arewewebyet.org/topics/templating/)
|
|
|
|
|
//! which you want to reload whenever a template file changes. However, if many small changes are
|
|
|
|
|
//! made to the template files in quick succession, it would be wasteful to reload the templates
|
|
|
|
|
//! for every change; instead, a debouncer could be used to group the changes that occur at a
|
|
|
|
|
//! similar time into a single change, so the templates are only reloaded once.
|
|
|
|
|
//!
|
|
|
|
|
//! A new debouncer can be created with the [`debouncer`](debouncer) function, which returns the
|
|
|
|
|
//! debouncer in two halves: a tx (send) half and rx (receive) half. The tx sends raw un-debounced
|
|
|
|
|
//! events to the debouncer, and the rx receives the debounced events from the debouncer. Both
|
|
|
|
|
//! halves can be cloned to allow for multiple senders and receivers.
|
|
|
|
|
//!
|
|
|
|
|
//! ```
|
|
|
|
|
//! # use std::{thread, time::Duration};
|
|
|
|
|
//! // Create a new debouncer which takes raw events of type `u32` and outputs
|
|
|
|
|
//! // debounced events of type `Vec<u32>`.
|
|
|
|
|
//! let (tx, rx) = treacle::debouncer::<u32, Vec<u32>, _>(
|
|
|
|
|
//! // Group events which occur in the same 500ms window.
|
|
|
|
|
//! Duration::from_millis(500),
|
|
|
|
|
//! // Combine raw events by pushing them to a vector.
|
|
|
|
|
//! |acc, raw_event| {
|
|
|
|
|
//! let mut events_vector = acc.unwrap_or_default();
|
|
|
|
|
//! events_vector.push(raw_event);
|
|
|
|
|
//! events_vector
|
|
|
|
|
//! });
|
|
|
|
|
//!
|
|
|
|
|
//! thread::spawn(move || {
|
|
|
|
|
//! // Send two raw events in quick succession.
|
|
|
|
|
//! tx.send(10).unwrap();
|
|
|
|
|
//! tx.send(20).unwrap();
|
|
|
|
|
//!
|
|
|
|
|
//! // Wait, then send more raw events.
|
|
|
|
|
//! thread::sleep(Duration::from_millis(500));
|
|
|
|
|
//! tx.send(30).unwrap();
|
|
|
|
|
//! tx.send(40).unwrap();
|
|
|
|
|
//! tx.send(50).unwrap();
|
|
|
|
|
//! });
|
|
|
|
|
//!
|
|
|
|
|
//! assert_eq!(rx.recv().unwrap(), &[10, 20]);
|
|
|
|
|
//! assert_eq!(rx.recv().unwrap(), &[30, 40, 50]);
|
|
|
|
|
//! ```
|
|
|
|
|
|
|
|
|
|
pub mod fold;
|
|
|
|
|
|
|
|
|
|
use std::{
|
|
|
|
|
@ -691,7 +736,7 @@ struct CountOverflowError;
|
|
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
|
|
|
|
use std::{time::Duration, thread};
|
|
|
|
|
use std::{time::{Duration, Instant}, thread};
|
|
|
|
|
|
|
|
|
|
use super::{debouncer, fold};
|
|
|
|
|
|
|
|
|
|
@ -728,4 +773,58 @@ mod tests {
|
|
|
|
|
assert_eq!(rx.recv().unwrap(), &[1, 2, 3]);
|
|
|
|
|
assert!(rx.recv().is_err());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_multi_tx() {
|
|
|
|
|
let (tx1, rx) = debouncer(Duration::from_millis(100), fold::fold_vec_push::<u8>);
|
|
|
|
|
|
|
|
|
|
let tx2 = tx1.clone();
|
|
|
|
|
|
|
|
|
|
tx1.send(1).unwrap();
|
|
|
|
|
tx2.send(2).unwrap();
|
|
|
|
|
assert_eq!(rx.recv().unwrap(), &[1, 2]);
|
|
|
|
|
|
|
|
|
|
drop(tx1);
|
|
|
|
|
|
|
|
|
|
tx2.send(3).unwrap();
|
|
|
|
|
tx2.send(4).unwrap();
|
|
|
|
|
assert_eq!(rx.recv().unwrap(), &[3, 4]);
|
|
|
|
|
|
|
|
|
|
let start_time = Instant::now();
|
|
|
|
|
tx2.send(5).unwrap();
|
|
|
|
|
tx2.send(6).unwrap();
|
|
|
|
|
assert_eq!(rx.recv().unwrap(), &[5, 6]);
|
|
|
|
|
assert!(Instant::now().duration_since(start_time) >= Duration::from_millis(100));
|
|
|
|
|
|
|
|
|
|
tx2.send(7).unwrap();
|
|
|
|
|
tx2.send(8).unwrap();
|
|
|
|
|
drop(tx2);
|
|
|
|
|
assert_eq!(rx.recv().unwrap(), &[7, 8]);
|
|
|
|
|
assert!(rx.recv().is_err());
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_multi_rx() {
|
|
|
|
|
let (tx, rx1) = debouncer(Duration::from_millis(100), fold::fold_vec_push::<u8>);
|
|
|
|
|
|
|
|
|
|
let rx2 = rx1.clone();
|
|
|
|
|
|
|
|
|
|
tx.send(1).unwrap();
|
|
|
|
|
tx.send(2).unwrap();
|
|
|
|
|
assert_eq!(rx1.recv().unwrap(), &[1, 2]);
|
|
|
|
|
|
|
|
|
|
tx.send(3).unwrap();
|
|
|
|
|
tx.send(4).unwrap();
|
|
|
|
|
assert_eq!(rx2.recv().unwrap(), &[3, 4]);
|
|
|
|
|
|
|
|
|
|
drop(rx1);
|
|
|
|
|
|
|
|
|
|
tx.send(5).unwrap();
|
|
|
|
|
tx.send(6).unwrap();
|
|
|
|
|
assert_eq!(rx2.recv().unwrap(), &[5, 6]);
|
|
|
|
|
|
|
|
|
|
drop(rx2);
|
|
|
|
|
|
|
|
|
|
assert!(tx.send(7).is_err());
|
|
|
|
|
}
|
|
|
|
|
}
|