use std::time::Duration;
use bevy::time::Stopwatch;
use pyo3::{exceptions::PyTypeError, prelude::*};
fn duration_from_py(py_duration: &Bound<'_, PyAny>) -> PyResult<Duration> {
if let Ok(duration) = py_duration.extract::<Duration>() {
return Ok(duration);
}
if let Ok(seconds) = py_duration.extract::<f64>() {
if seconds < 0.0 {
return Err(PyTypeError::new_err("Duration cannot be negative"));
}
return Ok(Duration::from_secs_f64(seconds));
}
if let Ok(seconds) = py_duration.extract::<u64>() {
return Ok(Duration::from_secs(seconds));
}
Err(PyTypeError::new_err(
"Duration must be a Duration object, float (seconds), or int (seconds)",
))
}
#[pyclass(name = "Stopwatch", eq)]
#[derive(Debug, Clone, Default, PartialEq)]
pub struct PyStopwatch {
pub(crate) stopwatch: Stopwatch,
}
#[pymethods]
impl PyStopwatch {
#[new]
pub fn new() -> Self {
Self {
stopwatch: Stopwatch::new(),
}
}
pub fn tick(&mut self, delta: &Bound<'_, PyAny>) -> PyResult<()> {
self.stopwatch.tick(duration_from_py(delta)?);
Ok(())
}
pub fn elapsed(&self) -> Duration {
self.stopwatch.elapsed()
}
pub fn elapsed_secs(&self) -> f32 {
self.stopwatch.elapsed_secs()
}
pub fn elapsed_secs_f64(&self) -> f64 {
self.stopwatch.elapsed_secs_f64()
}
pub fn set_elapsed(&mut self, time: &Bound<'_, PyAny>) -> PyResult<()> {
self.stopwatch.set_elapsed(duration_from_py(time)?);
Ok(())
}
pub fn pause(&mut self) {
self.stopwatch.pause();
}
pub fn unpause(&mut self) {
self.stopwatch.unpause();
}
pub fn is_paused(&self) -> bool {
self.stopwatch.is_paused()
}
pub fn reset(&mut self) {
self.stopwatch.reset();
}
pub fn __repr__(&self) -> String {
format!("{:?}", self.stopwatch)
}
}