Skip to main content

wickra_core/indicators/
evwma.rs

1//! Elastic Volume-Weighted Moving Average (EVWMA).
2
3use std::collections::VecDeque;
4
5use crate::error::{Error, Result};
6use crate::ohlcv::Candle;
7use crate::traits::Indicator;
8
9/// Christian P. Fries' Elastic Volume-Weighted Moving Average.
10///
11/// Unlike `VWMA` which is a per-bar weighted mean, `EVWMA` runs an
12/// "elastic" recurrence whose smoothing weight is the bar's volume relative
13/// to the running window-volume:
14///
15/// ```text
16/// V_sum_t  = Σ volume_i over the last `period` candles
17/// EVWMA_t  = ((V_sum_t - volume_t) * EVWMA_{t-1} + volume_t * close_t) / V_sum_t
18/// ```
19///
20/// A bar whose volume is small compared to the window total barely moves the
21/// average; a bar whose volume dominates the window pulls it strongly toward
22/// the bar's close. The series is seeded with the close of the first candle
23/// after the volume window has filled (i.e. after `period` candles).
24///
25/// If `V_sum_t == 0` (every candle in the window has zero volume), the
26/// recurrence is undefined; the indicator holds its previous value.
27///
28/// Reference: Christian P. Fries, *Wilmott Magazine*, 2001.
29///
30/// # Example
31///
32/// ```
33/// use wickra_core::{Candle, Evwma, Indicator};
34///
35/// let mut evwma = Evwma::new(20).unwrap();
36/// let mut last = None;
37/// for i in 0..40 {
38///     let p = 100.0 + f64::from(i);
39///     let candle = Candle::new(p, p + 1.0, p - 1.0, p, 10.0, i64::from(i)).unwrap();
40///     last = evwma.update(candle);
41/// }
42/// assert!(last.is_some());
43/// ```
44#[derive(Debug, Clone)]
45pub struct Evwma {
46    period: usize,
47    /// Rolling window of `(close, volume)` pairs, oldest at the front.
48    window: VecDeque<(f64, f64)>,
49    sum_v: f64,
50    current: Option<f64>,
51}
52
53impl Evwma {
54    /// # Errors
55    /// Returns [`Error::PeriodZero`] if `period == 0`.
56    pub fn new(period: usize) -> Result<Self> {
57        if period == 0 {
58            return Err(Error::PeriodZero);
59        }
60        Ok(Self {
61            period,
62            window: VecDeque::with_capacity(period),
63            sum_v: 0.0,
64            current: None,
65        })
66    }
67
68    /// Configured period.
69    pub const fn period(&self) -> usize {
70        self.period
71    }
72
73    /// Current value if available.
74    pub const fn value(&self) -> Option<f64> {
75        self.current
76    }
77}
78
79impl Indicator for Evwma {
80    type Input = Candle;
81    type Output = f64;
82
83    fn update(&mut self, candle: Candle) -> Option<f64> {
84        let close = candle.close;
85        let volume = candle.volume;
86        if self.window.len() == self.period {
87            let (_, old_v) = self.window.pop_front().expect("window is non-empty");
88            self.sum_v -= old_v;
89        }
90        self.window.push_back((close, volume));
91        self.sum_v += volume;
92        if self.window.len() < self.period {
93            return None;
94        }
95        // The volume sum may be zero (every bar in the window had zero
96        // volume); the recurrence is undefined, so seed/hold instead.
97        if self.sum_v <= 0.0 {
98            if self.current.is_none() {
99                self.current = Some(close);
100            }
101            return self.current;
102        }
103        let prev = self.current.unwrap_or(close);
104        let next = ((self.sum_v - volume) * prev + volume * close) / self.sum_v;
105        self.current = Some(next);
106        Some(next)
107    }
108
109    fn reset(&mut self) {
110        self.window.clear();
111        self.sum_v = 0.0;
112        self.current = None;
113    }
114
115    fn warmup_period(&self) -> usize {
116        self.period
117    }
118
119    fn is_ready(&self) -> bool {
120        self.current.is_some()
121    }
122
123    fn name(&self) -> &'static str {
124        "EVWMA"
125    }
126}
127
128#[cfg(test)]
129mod tests {
130    use super::*;
131    use crate::traits::BatchExt;
132    use approx::assert_relative_eq;
133
134    fn candle(close: f64, volume: f64, ts: i64) -> Candle {
135        Candle::new(close, close, close, close, volume, ts).unwrap()
136    }
137
138    #[test]
139    fn rejects_zero_period() {
140        assert!(matches!(Evwma::new(0), Err(Error::PeriodZero)));
141    }
142
143    #[test]
144    fn accessors_and_metadata() {
145        let mut e = Evwma::new(5).unwrap();
146        assert_eq!(e.period(), 5);
147        assert_eq!(e.warmup_period(), 5);
148        assert_eq!(e.name(), "EVWMA");
149        assert_eq!(e.value(), None);
150        for i in 0..5 {
151            e.update(candle(10.0, 1.0, i));
152        }
153        assert!(e.value().is_some());
154    }
155
156    #[test]
157    fn constant_series_yields_the_constant() {
158        // A flat close — every (V_sum - v) * prev + v * close reduces to
159        // V_sum * close, so the recurrence preserves the constant after the
160        // first seeded sample.
161        let mut e = Evwma::new(5).unwrap();
162        let candles: Vec<Candle> = (0..30).map(|i| candle(42.0, 3.0, i)).collect();
163        let out = e.batch(&candles);
164        for v in out.iter().skip(4).flatten() {
165            assert_relative_eq!(*v, 42.0, epsilon = 1e-12);
166        }
167    }
168
169    #[test]
170    fn reference_value_period_2() {
171        // EVWMA(2). Bars: (close, volume) = (10, 1), (20, 3), (30, 1).
172        //   Bar 1: window not full (size 1) -> None.
173        //   Bar 2: window full, sum_v = 4, prev seeds to 20.
174        //          EVWMA = ((4 - 3) * 20 + 3 * 20) / 4 = 80 / 4 = 20.
175        //   Bar 3: window slides, sum_v = 4 (drops the 1, gains the 1).
176        //          EVWMA = ((4 - 1) * 20 + 1 * 30) / 4 = (60 + 30) / 4 = 22.5.
177        let mut e = Evwma::new(2).unwrap();
178        assert_eq!(e.update(candle(10.0, 1.0, 0)), None);
179        assert_relative_eq!(
180            e.update(candle(20.0, 3.0, 1)).unwrap(),
181            20.0,
182            epsilon = 1e-12
183        );
184        assert_relative_eq!(
185            e.update(candle(30.0, 1.0, 2)).unwrap(),
186            22.5,
187            epsilon = 1e-12
188        );
189    }
190
191    #[test]
192    fn warmup_emits_first_value_at_period() {
193        let mut e = Evwma::new(4).unwrap();
194        for i in 0..3 {
195            assert_eq!(e.update(candle(10.0, 1.0, i)), None);
196        }
197        assert!(e.update(candle(10.0, 1.0, 3)).is_some());
198    }
199
200    #[test]
201    fn zero_volume_window_holds_value() {
202        // Every bar has zero volume: no participation, so the recurrence
203        // can't move and EVWMA simply seeds to the first close.
204        let mut e = Evwma::new(3).unwrap();
205        e.update(candle(10.0, 0.0, 0));
206        e.update(candle(15.0, 0.0, 1));
207        let v = e.update(candle(20.0, 0.0, 2)).unwrap();
208        assert_relative_eq!(v, 20.0, epsilon = 1e-12);
209        // Next bar still flat-zero volume: holds 20.
210        let v2 = e.update(candle(50.0, 0.0, 3)).unwrap();
211        assert_relative_eq!(v2, 20.0, epsilon = 1e-12);
212    }
213
214    #[test]
215    fn batch_equals_streaming() {
216        let candles: Vec<Candle> = (0..60_i64)
217            .map(|i| {
218                let c = 100.0 + (i as f64 * 0.3).sin() * 8.0;
219                candle(c, 1.0 + (i % 7) as f64, i)
220            })
221            .collect();
222        let batch = Evwma::new(10).unwrap().batch(&candles);
223        let mut b = Evwma::new(10).unwrap();
224        let streamed: Vec<_> = candles.iter().map(|c| b.update(*c)).collect();
225        assert_eq!(batch, streamed);
226    }
227
228    #[test]
229    fn reset_clears_state() {
230        let mut e = Evwma::new(3).unwrap();
231        let candles: Vec<Candle> = (0..10).map(|i| candle(10.0 + i as f64, 2.0, i)).collect();
232        e.batch(&candles);
233        assert!(e.is_ready());
234        e.reset();
235        assert!(!e.is_ready());
236        assert_eq!(e.update(candle(10.0, 1.0, 0)), None);
237    }
238}