use std::collections::VecDeque;
use crate::error::{Error, Result};
use crate::ohlcv::Candle;
use crate::traits::Indicator;
#[derive(Debug, Clone)]
pub struct WilliamsR {
period: usize,
candles: VecDeque<Candle>,
}
impl WilliamsR {
pub fn new(period: usize) -> Result<Self> {
if period == 0 {
return Err(Error::PeriodZero);
}
Ok(Self {
period,
candles: VecDeque::with_capacity(period),
})
}
pub const fn period(&self) -> usize {
self.period
}
}
impl Indicator for WilliamsR {
type Input = Candle;
type Output = f64;
fn update(&mut self, candle: Candle) -> Option<f64> {
if self.candles.len() == self.period {
self.candles.pop_front();
}
self.candles.push_back(candle);
if self.candles.len() < self.period {
return None;
}
let hh = self
.candles
.iter()
.map(|c| c.high)
.fold(f64::NEG_INFINITY, f64::max);
let ll = self
.candles
.iter()
.map(|c| c.low)
.fold(f64::INFINITY, f64::min);
let range = hh - ll;
if range == 0.0 {
return Some(-50.0);
}
Some(-100.0 * (hh - candle.close) / range)
}
fn reset(&mut self) {
self.candles.clear();
}
fn warmup_period(&self) -> usize {
self.period
}
fn is_ready(&self) -> bool {
self.candles.len() == self.period
}
fn name(&self) -> &'static str {
"WilliamsR"
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::traits::BatchExt;
use approx::assert_relative_eq;
fn c(h: f64, l: f64, cl: f64) -> Candle {
Candle::new(cl, h, l, cl, 1.0, 0).unwrap()
}
#[test]
fn close_at_high_yields_zero() {
let candles = vec![c(10.0, 8.0, 9.0), c(11.0, 9.0, 10.0), c(12.0, 10.0, 12.0)];
let mut w = WilliamsR::new(3).unwrap();
let out = w.batch(&candles);
assert_relative_eq!(out[2].unwrap(), 0.0, epsilon = 1e-12);
}
#[test]
fn close_at_low_yields_minus_100() {
let candles = vec![c(12.0, 10.0, 11.0), c(11.0, 9.0, 10.0), c(10.0, 8.0, 8.0)];
let mut w = WilliamsR::new(3).unwrap();
let out = w.batch(&candles);
assert_relative_eq!(out[2].unwrap(), -100.0, epsilon = 1e-12);
}
#[test]
fn within_range() {
let candles: Vec<Candle> = (0..100)
.map(|i| {
let m = 50.0 + (f64::from(i) * 0.3).sin() * 5.0;
c(m + 1.0, m - 1.0, m)
})
.collect();
let mut w = WilliamsR::new(14).unwrap();
for v in w.batch(&candles).into_iter().flatten() {
assert!((-100.0..=0.0).contains(&v), "%R out of range: {v}");
}
}
#[test]
fn batch_equals_streaming() {
let candles: Vec<Candle> = (0..30)
.map(|i| c(f64::from(i) + 2.0, f64::from(i), f64::from(i) + 1.0))
.collect();
let mut a = WilliamsR::new(5).unwrap();
let mut b = WilliamsR::new(5).unwrap();
assert_eq!(
a.batch(&candles),
candles.iter().map(|x| b.update(*x)).collect::<Vec<_>>()
);
}
#[test]
fn rejects_zero_period() {
assert!(WilliamsR::new(0).is_err());
}
}