use crate::utils::{check_begin_idx3, make_vec};
use serde::Deserialize;
use talib_sys::{
TA_ATR_Lookback, TA_Integer, TA_NATR_Lookback, TA_RetCode, TA_TRANGE_Lookback, TA_ATR, TA_NATR,
TA_TRANGE,
};
use derive_builder::Builder;
#[derive(Builder, Deserialize)]
pub struct ATRKwargs {
#[builder(default = "14")]
pub timeperiod: i32,
}
pub fn ta_atr(
high_ptr: *const f64,
low_ptr: *const f64,
close_ptr: *const f64,
len: usize,
kwargs: &ATRKwargs,
) -> Result<Vec<f64>, TA_RetCode> {
let mut out_begin: TA_Integer = 0;
let mut out_size: TA_Integer = 0;
let begin_idx = check_begin_idx3(len, high_ptr, low_ptr, close_ptr) as i32;
let end_idx = len as i32 - begin_idx - 1;
let lookback = begin_idx + unsafe { TA_ATR_Lookback(kwargs.timeperiod) };
let (mut out, ptr) = make_vec(len, lookback);
let ret_code = unsafe {
TA_ATR(
0,
end_idx,
high_ptr.offset(begin_idx as isize),
low_ptr.offset(begin_idx as isize),
close_ptr.offset(begin_idx as isize),
kwargs.timeperiod,
&mut out_begin,
&mut out_size,
ptr,
)
};
let out_size_begin = (begin_idx + out_begin + out_size) as usize;
match ret_code {
TA_RetCode::TA_SUCCESS => {
if out_size != 0 {
unsafe {
out.set_len(out_size_begin);
}
} else {
unsafe {
out.set_len(len);
}
}
Ok(out)
}
_ => Err(ret_code),
}
}
pub fn ta_trange(
high_ptr: *const f64,
low_ptr: *const f64,
close_ptr: *const f64,
len: usize,
) -> Result<Vec<f64>, TA_RetCode> {
let mut out_begin: TA_Integer = 0;
let mut out_size: TA_Integer = 0;
let begin_idx = check_begin_idx3(len, high_ptr, low_ptr, close_ptr) as i32;
let end_idx = len as i32 - begin_idx - 1;
let lookback = begin_idx + unsafe { TA_TRANGE_Lookback() };
let (mut out, ptr) = make_vec(len, lookback);
let ret_code = unsafe {
TA_TRANGE(
0,
end_idx,
high_ptr.offset(begin_idx as isize),
low_ptr.offset(begin_idx as isize),
close_ptr.offset(begin_idx as isize),
&mut out_begin,
&mut out_size,
ptr,
)
};
let out_size_begin = (begin_idx + out_begin + out_size) as usize;
match ret_code {
TA_RetCode::TA_SUCCESS => {
if out_size != 0 {
unsafe {
out.set_len(out_size_begin);
}
} else {
unsafe {
out.set_len(len);
}
}
Ok(out)
}
_ => Err(ret_code),
}
}
#[derive(Builder, Deserialize)]
pub struct NATRKwargs {
#[builder(default = "14")]
pub timeperiod: i32,
}
pub fn ta_natr(
high_ptr: *const f64,
low_ptr: *const f64,
close_ptr: *const f64,
len: usize,
kwargs: &NATRKwargs,
) -> Result<Vec<f64>, TA_RetCode> {
let mut out_begin: TA_Integer = 0;
let mut out_size: TA_Integer = 0;
let begin_idx = check_begin_idx3(len, high_ptr, low_ptr, close_ptr) as i32;
let end_idx = len as i32 - begin_idx - 1;
let lookback = begin_idx + unsafe { TA_NATR_Lookback(kwargs.timeperiod) };
let (mut out, ptr) = make_vec(len, lookback);
let ret_code = unsafe {
TA_NATR(
0,
end_idx,
high_ptr.offset(begin_idx as isize),
low_ptr.offset(begin_idx as isize),
close_ptr.offset(begin_idx as isize),
kwargs.timeperiod,
&mut out_begin,
&mut out_size,
ptr,
)
};
let out_size_begin = (begin_idx + out_begin + out_size) as usize;
match ret_code {
TA_RetCode::TA_SUCCESS => {
if out_size != 0 {
unsafe {
out.set_len(out_size_begin);
}
} else {
unsafe {
out.set_len(len);
}
}
Ok(out)
}
_ => Err(ret_code),
}
}