use crate::{BackgroundExecutor, Task};
use std::{
future::Future,
pin::Pin,
sync::atomic::{AtomicUsize, Ordering::SeqCst},
task,
time::Duration,
};
pub trait FluentBuilder {
fn map<U>(self, f: impl FnOnce(Self) -> U) -> U
where
Self: Sized,
{
f(self)
}
fn when(self, condition: bool, then: impl FnOnce(Self) -> Self) -> Self
where
Self: Sized,
{
self.map(|this| if condition { then(this) } else { this })
}
fn when_else(
self,
condition: bool,
then: impl FnOnce(Self) -> Self,
else_fn: impl FnOnce(Self) -> Self,
) -> Self
where
Self: Sized,
{
self.map(|this| if condition { then(this) } else { else_fn(this) })
}
fn when_some<T>(self, option: Option<T>, then: impl FnOnce(Self, T) -> Self) -> Self
where
Self: Sized,
{
self.map(|this| {
if let Some(value) = option {
then(this, value)
} else {
this
}
})
}
fn when_none<T>(self, option: &Option<T>, then: impl FnOnce(Self) -> Self) -> Self
where
Self: Sized,
{
self.map(|this| if option.is_some() { this } else { then(this) })
}
}
pub trait FutureExt {
fn with_timeout(self, timeout: Duration, executor: &BackgroundExecutor) -> WithTimeout<Self>
where
Self: Sized;
}
impl<T: Future> FutureExt for T {
fn with_timeout(self, timeout: Duration, executor: &BackgroundExecutor) -> WithTimeout<Self>
where
Self: Sized,
{
WithTimeout {
future: self,
timer: executor.timer(timeout),
}
}
}
#[pin_project::pin_project]
pub struct WithTimeout<T> {
#[pin]
future: T,
#[pin]
timer: Task<()>,
}
#[derive(Debug, thiserror::Error)]
#[error("Timed out before future resolved")]
pub struct Timeout;
impl<T: Future> Future for WithTimeout<T> {
type Output = Result<T::Output, Timeout>;
fn poll(self: Pin<&mut Self>, cx: &mut task::Context) -> task::Poll<Self::Output> {
let this = self.project();
if let task::Poll::Ready(output) = this.future.poll(cx) {
task::Poll::Ready(Ok(output))
} else if this.timer.poll(cx).is_ready() {
task::Poll::Ready(Err(Timeout))
} else {
task::Poll::Pending
}
}
}
pub(crate) fn atomic_incr_if_not_zero(counter: &AtomicUsize) -> usize {
let mut loaded = counter.load(SeqCst);
loop {
if loaded == 0 {
return 0;
}
match counter.compare_exchange_weak(loaded, loaded + 1, SeqCst, SeqCst) {
Ok(x) => return x + 1,
Err(actual) => loaded = actual,
}
}
}
#[inline]
pub(crate) fn round_half_toward_zero(value: f32) -> f32 {
(value.abs() - 0.5).ceil().copysign(value)
}
#[inline]
pub(crate) fn round_half_toward_zero_f64(value: f64) -> f64 {
(value.abs() - 0.5).ceil().copysign(value)
}
#[inline]
pub(crate) fn round_to_device_pixel(logical: f32, scale_factor: f32) -> f32 {
round_half_toward_zero(logical * scale_factor)
}
#[inline]
pub(crate) fn round_stroke_to_device_pixel(logical: f32, scale_factor: f32) -> f32 {
if logical == 0.0 {
0.0
} else {
round_to_device_pixel(logical.max(0.0), scale_factor).max(1.0)
}
}
#[inline]
pub(crate) fn floor_to_device_pixel(logical: f32, scale_factor: f32) -> f32 {
(logical * scale_factor).floor()
}
#[inline]
pub(crate) fn ceil_to_device_pixel(logical: f32, scale_factor: f32) -> f32 {
(logical * scale_factor).ceil()
}
#[cfg(test)]
mod tests {
use crate::TestAppContext;
use super::*;
#[test]
fn test_round_half_toward_zero() {
assert_eq!(round_half_toward_zero(0.5), 0.0);
assert_eq!(round_half_toward_zero(1.5), 1.0);
assert_eq!(round_half_toward_zero(2.5), 2.0);
assert_eq!(round_half_toward_zero(-0.5), 0.0);
assert_eq!(round_half_toward_zero(-1.5), -1.0);
assert_eq!(round_half_toward_zero(-2.5), -2.0);
assert_eq!(round_half_toward_zero(1.5001), 2.0);
assert_eq!(round_half_toward_zero(1.4999), 1.0);
assert_eq!(round_half_toward_zero(-1.5001), -2.0);
assert_eq!(round_half_toward_zero(-1.4999), -1.0);
assert_eq!(round_half_toward_zero(0.0), 0.0);
assert_eq!(round_half_toward_zero(3.0), 3.0);
assert_eq!(round_half_toward_zero(-3.0), -3.0);
}
#[test]
fn test_device_pixel_helpers() {
assert_eq!(round_to_device_pixel(1.0, 1.5), 1.0);
assert_eq!(round_to_device_pixel(0.3, 2.0), 1.0);
assert_eq!(round_to_device_pixel(1.4, 1.0), 1.0);
assert_eq!(round_to_device_pixel(1.6, 1.0), 2.0);
assert_eq!(round_stroke_to_device_pixel(0.0, 1.0), 0.0);
assert_eq!(round_stroke_to_device_pixel(0.4, 1.0), 1.0);
assert_eq!(round_stroke_to_device_pixel(0.5, 1.0), 1.0);
assert_eq!(round_stroke_to_device_pixel(1.0, 1.5), 1.0);
assert_eq!(round_stroke_to_device_pixel(1.6, 1.0), 2.0);
assert_eq!(floor_to_device_pixel(0.3, 2.0), 0.0);
assert_eq!(ceil_to_device_pixel(0.3, 2.0), 1.0);
assert_eq!(floor_to_device_pixel(2.1, 1.0), 2.0);
assert_eq!(ceil_to_device_pixel(2.1, 1.0), 3.0);
assert_eq!(round_to_device_pixel(2.0, 2.0), 4.0);
assert_eq!(floor_to_device_pixel(2.0, 2.0), 4.0);
assert_eq!(ceil_to_device_pixel(2.0, 2.0), 4.0);
}
#[test]
fn test_round_half_toward_zero_f64() {
assert_eq!(round_half_toward_zero_f64(0.5), 0.0);
assert_eq!(round_half_toward_zero_f64(-0.5), 0.0);
assert_eq!(round_half_toward_zero_f64(1.5), 1.0);
assert_eq!(round_half_toward_zero_f64(-1.5), -1.0);
assert_eq!(round_half_toward_zero_f64(2.5001), 3.0);
}
#[gpui::test]
async fn test_with_timeout(cx: &mut TestAppContext) {
Task::ready(())
.with_timeout(Duration::from_secs(1), &cx.executor())
.await
.expect("Timeout should be noop");
let long_duration = Duration::from_secs(6000);
let short_duration = Duration::from_secs(1);
cx.executor()
.timer(long_duration)
.with_timeout(short_duration, &cx.executor())
.await
.expect_err("timeout should have triggered");
let fut = cx
.executor()
.timer(long_duration)
.with_timeout(short_duration, &cx.executor());
cx.executor().advance_clock(short_duration * 2);
futures::FutureExt::now_or_never(fut)
.unwrap_or_else(|| panic!("timeout should have triggered"))
.expect_err("timeout");
}
}