#![allow(unused, clippy::comparison_to_empty, clippy::manual_range_patterns)]
use crate::profile::{ProfileType, typedef};
use crate::proto::*;
#[derive(Debug, Clone)]
pub struct TimeInZone {
pub timestamp: typedef::DateTime,
pub reference_mesg: typedef::MesgNum,
pub reference_index: typedef::MessageIndex,
pub time_in_hr_zone: Vec<u32>,
pub time_in_speed_zone: Vec<u32>,
pub time_in_cadence_zone: Vec<u32>,
pub time_in_power_zone: Vec<u32>,
pub hr_zone_high_boundary: Vec<u8>,
pub speed_zone_high_boundary: Vec<u16>,
pub cadence_zone_high_boundary: Vec<u8>,
pub power_zone_high_boundary: Vec<u16>,
pub hr_calc_type: typedef::HrZoneCalc,
pub max_heart_rate: u8,
pub resting_heart_rate: u8,
pub threshold_heart_rate: u8,
pub pwr_calc_type: typedef::PwrZoneCalc,
pub functional_threshold_power: u16,
pub unknown_fields: Vec<Field>,
pub developer_fields: Vec<DeveloperField>,
}
impl TimeInZone {
pub const TIMESTAMP: u8 = 253;
pub const REFERENCE_MESG: u8 = 0;
pub const REFERENCE_INDEX: u8 = 1;
pub const TIME_IN_HR_ZONE: u8 = 2;
pub const TIME_IN_SPEED_ZONE: u8 = 3;
pub const TIME_IN_CADENCE_ZONE: u8 = 4;
pub const TIME_IN_POWER_ZONE: u8 = 5;
pub const HR_ZONE_HIGH_BOUNDARY: u8 = 6;
pub const SPEED_ZONE_HIGH_BOUNDARY: u8 = 7;
pub const CADENCE_ZONE_HIGH_BOUNDARY: u8 = 8;
pub const POWER_ZONE_HIGH_BOUNDARY: u8 = 9;
pub const HR_CALC_TYPE: u8 = 10;
pub const MAX_HEART_RATE: u8 = 11;
pub const RESTING_HEART_RATE: u8 = 12;
pub const THRESHOLD_HEART_RATE: u8 = 13;
pub const PWR_CALC_TYPE: u8 = 14;
pub const FUNCTIONAL_THRESHOLD_POWER: u8 = 15;
pub const fn new() -> Self {
Self {
timestamp: typedef::DateTime(u32::MAX),
reference_mesg: typedef::MesgNum(u16::MAX),
reference_index: typedef::MessageIndex(u16::MAX),
time_in_hr_zone: Vec::<u32>::new(),
time_in_speed_zone: Vec::<u32>::new(),
time_in_cadence_zone: Vec::<u32>::new(),
time_in_power_zone: Vec::<u32>::new(),
hr_zone_high_boundary: Vec::<u8>::new(),
speed_zone_high_boundary: Vec::<u16>::new(),
cadence_zone_high_boundary: Vec::<u8>::new(),
power_zone_high_boundary: Vec::<u16>::new(),
hr_calc_type: typedef::HrZoneCalc(u8::MAX),
max_heart_rate: u8::MAX,
resting_heart_rate: u8::MAX,
threshold_heart_rate: u8::MAX,
pwr_calc_type: typedef::PwrZoneCalc(u8::MAX),
functional_threshold_power: u16::MAX,
unknown_fields: Vec::new(),
developer_fields: Vec::new(),
}
}
pub fn time_in_hr_zone_scaled(&self) -> Vec<f64> {
if self.time_in_hr_zone == Vec::<u32>::new() {
return Vec::new();
}
let mut v = Vec::with_capacity(self.time_in_hr_zone.len());
for &x in &self.time_in_hr_zone {
v.push(x as f64 / 1000.0 - 0.0)
}
v
}
pub fn set_time_in_hr_zone_scaled(&mut self, v: &Vec<f64>) -> &mut TimeInZone {
if v.is_empty() {
self.time_in_hr_zone = Vec::new();
return self;
}
self.time_in_hr_zone = Vec::with_capacity(v.len());
for &x in v {
let unscaled = (x + 0.0) * 1000.0;
if unscaled.is_nan() || unscaled.is_infinite() || unscaled > u32::MAX as f64 {
self.time_in_hr_zone.push(u32::MAX);
continue;
}
self.time_in_hr_zone.push(unscaled as u32);
}
self
}
pub fn time_in_speed_zone_scaled(&self) -> Vec<f64> {
if self.time_in_speed_zone == Vec::<u32>::new() {
return Vec::new();
}
let mut v = Vec::with_capacity(self.time_in_speed_zone.len());
for &x in &self.time_in_speed_zone {
v.push(x as f64 / 1000.0 - 0.0)
}
v
}
pub fn set_time_in_speed_zone_scaled(&mut self, v: &Vec<f64>) -> &mut TimeInZone {
if v.is_empty() {
self.time_in_speed_zone = Vec::new();
return self;
}
self.time_in_speed_zone = Vec::with_capacity(v.len());
for &x in v {
let unscaled = (x + 0.0) * 1000.0;
if unscaled.is_nan() || unscaled.is_infinite() || unscaled > u32::MAX as f64 {
self.time_in_speed_zone.push(u32::MAX);
continue;
}
self.time_in_speed_zone.push(unscaled as u32);
}
self
}
pub fn time_in_cadence_zone_scaled(&self) -> Vec<f64> {
if self.time_in_cadence_zone == Vec::<u32>::new() {
return Vec::new();
}
let mut v = Vec::with_capacity(self.time_in_cadence_zone.len());
for &x in &self.time_in_cadence_zone {
v.push(x as f64 / 1000.0 - 0.0)
}
v
}
pub fn set_time_in_cadence_zone_scaled(&mut self, v: &Vec<f64>) -> &mut TimeInZone {
if v.is_empty() {
self.time_in_cadence_zone = Vec::new();
return self;
}
self.time_in_cadence_zone = Vec::with_capacity(v.len());
for &x in v {
let unscaled = (x + 0.0) * 1000.0;
if unscaled.is_nan() || unscaled.is_infinite() || unscaled > u32::MAX as f64 {
self.time_in_cadence_zone.push(u32::MAX);
continue;
}
self.time_in_cadence_zone.push(unscaled as u32);
}
self
}
pub fn time_in_power_zone_scaled(&self) -> Vec<f64> {
if self.time_in_power_zone == Vec::<u32>::new() {
return Vec::new();
}
let mut v = Vec::with_capacity(self.time_in_power_zone.len());
for &x in &self.time_in_power_zone {
v.push(x as f64 / 1000.0 - 0.0)
}
v
}
pub fn set_time_in_power_zone_scaled(&mut self, v: &Vec<f64>) -> &mut TimeInZone {
if v.is_empty() {
self.time_in_power_zone = Vec::new();
return self;
}
self.time_in_power_zone = Vec::with_capacity(v.len());
for &x in v {
let unscaled = (x + 0.0) * 1000.0;
if unscaled.is_nan() || unscaled.is_infinite() || unscaled > u32::MAX as f64 {
self.time_in_power_zone.push(u32::MAX);
continue;
}
self.time_in_power_zone.push(unscaled as u32);
}
self
}
pub fn speed_zone_high_boundary_scaled(&self) -> Vec<f64> {
if self.speed_zone_high_boundary == Vec::<u16>::new() {
return Vec::new();
}
let mut v = Vec::with_capacity(self.speed_zone_high_boundary.len());
for &x in &self.speed_zone_high_boundary {
v.push(x as f64 / 1000.0 - 0.0)
}
v
}
pub fn set_speed_zone_high_boundary_scaled(&mut self, v: &Vec<f64>) -> &mut TimeInZone {
if v.is_empty() {
self.speed_zone_high_boundary = Vec::new();
return self;
}
self.speed_zone_high_boundary = Vec::with_capacity(v.len());
for &x in v {
let unscaled = (x + 0.0) * 1000.0;
if unscaled.is_nan() || unscaled.is_infinite() || unscaled > u16::MAX as f64 {
self.speed_zone_high_boundary.push(u16::MAX);
continue;
}
self.speed_zone_high_boundary.push(unscaled as u16);
}
self
}
}
impl Default for TimeInZone {
fn default() -> Self {
Self::new()
}
}
impl From<&Message> for TimeInZone {
fn from(mesg: &Message) -> Self {
let mut vals: [&Value; 254] = [const { &Value::Invalid }; 254];
const KNOWN_NUMS: [u64; 4] = [65535, 0, 0, 2305843009213693952];
let mut n = 0u64;
for field in &mesg.fields {
n += (KNOWN_NUMS[field.num as usize >> 6] >> (field.num & 63)) & 1 ^ 1
}
let mut unknown_fields: Vec<Field> = Vec::with_capacity(n as usize);
for field in &mesg.fields {
if (KNOWN_NUMS[field.num as usize >> 6] >> (field.num & 63)) & 1 == 0 {
unknown_fields.push(field.clone());
continue;
}
vals[field.num as usize] = &field.value;
}
Self {
timestamp: typedef::DateTime(vals[253].as_u32()),
reference_mesg: typedef::MesgNum(vals[0].as_u16()),
reference_index: typedef::MessageIndex(vals[1].as_u16()),
time_in_hr_zone: vals[2].as_vec_u32(),
time_in_speed_zone: vals[3].as_vec_u32(),
time_in_cadence_zone: vals[4].as_vec_u32(),
time_in_power_zone: vals[5].as_vec_u32(),
hr_zone_high_boundary: vals[6].as_vec_u8(),
speed_zone_high_boundary: vals[7].as_vec_u16(),
cadence_zone_high_boundary: vals[8].as_vec_u8(),
power_zone_high_boundary: vals[9].as_vec_u16(),
hr_calc_type: typedef::HrZoneCalc(vals[10].as_u8()),
max_heart_rate: vals[11].as_u8(),
resting_heart_rate: vals[12].as_u8(),
threshold_heart_rate: vals[13].as_u8(),
pwr_calc_type: typedef::PwrZoneCalc(vals[14].as_u8()),
functional_threshold_power: vals[15].as_u16(),
unknown_fields,
developer_fields: mesg.developer_fields.clone(),
}
}
}
impl From<TimeInZone> for Message {
fn from(m: TimeInZone) -> Self {
let mut arr = [const {
Field {
num: 0,
profile_type: ProfileType(0),
value: Value::Invalid,
is_expanded: false,
}
}; 17];
let mut len = 0usize;
if m.timestamp != typedef::DateTime(u32::MAX) {
arr[len] = Field {
num: 253,
profile_type: ProfileType::DATE_TIME,
value: Value::Uint32(m.timestamp.0),
is_expanded: false,
};
len += 1;
}
if m.reference_mesg != typedef::MesgNum(u16::MAX) {
arr[len] = Field {
num: 0,
profile_type: ProfileType::MESG_NUM,
value: Value::Uint16(m.reference_mesg.0),
is_expanded: false,
};
len += 1;
}
if m.reference_index != typedef::MessageIndex(u16::MAX) {
arr[len] = Field {
num: 1,
profile_type: ProfileType::MESSAGE_INDEX,
value: Value::Uint16(m.reference_index.0),
is_expanded: false,
};
len += 1;
}
if m.time_in_hr_zone != Vec::<u32>::new() {
arr[len] = Field {
num: 2,
profile_type: ProfileType::UINT32,
value: Value::VecUint32(m.time_in_hr_zone),
is_expanded: false,
};
len += 1;
}
if m.time_in_speed_zone != Vec::<u32>::new() {
arr[len] = Field {
num: 3,
profile_type: ProfileType::UINT32,
value: Value::VecUint32(m.time_in_speed_zone),
is_expanded: false,
};
len += 1;
}
if m.time_in_cadence_zone != Vec::<u32>::new() {
arr[len] = Field {
num: 4,
profile_type: ProfileType::UINT32,
value: Value::VecUint32(m.time_in_cadence_zone),
is_expanded: false,
};
len += 1;
}
if m.time_in_power_zone != Vec::<u32>::new() {
arr[len] = Field {
num: 5,
profile_type: ProfileType::UINT32,
value: Value::VecUint32(m.time_in_power_zone),
is_expanded: false,
};
len += 1;
}
if m.hr_zone_high_boundary != Vec::<u8>::new() {
arr[len] = Field {
num: 6,
profile_type: ProfileType::UINT8,
value: Value::VecUint8(m.hr_zone_high_boundary),
is_expanded: false,
};
len += 1;
}
if m.speed_zone_high_boundary != Vec::<u16>::new() {
arr[len] = Field {
num: 7,
profile_type: ProfileType::UINT16,
value: Value::VecUint16(m.speed_zone_high_boundary),
is_expanded: false,
};
len += 1;
}
if m.cadence_zone_high_boundary != Vec::<u8>::new() {
arr[len] = Field {
num: 8,
profile_type: ProfileType::UINT8,
value: Value::VecUint8(m.cadence_zone_high_boundary),
is_expanded: false,
};
len += 1;
}
if m.power_zone_high_boundary != Vec::<u16>::new() {
arr[len] = Field {
num: 9,
profile_type: ProfileType::UINT16,
value: Value::VecUint16(m.power_zone_high_boundary),
is_expanded: false,
};
len += 1;
}
if m.hr_calc_type != typedef::HrZoneCalc(u8::MAX) {
arr[len] = Field {
num: 10,
profile_type: ProfileType::HR_ZONE_CALC,
value: Value::Uint8(m.hr_calc_type.0),
is_expanded: false,
};
len += 1;
}
if m.max_heart_rate != u8::MAX {
arr[len] = Field {
num: 11,
profile_type: ProfileType::UINT8,
value: Value::Uint8(m.max_heart_rate),
is_expanded: false,
};
len += 1;
}
if m.resting_heart_rate != u8::MAX {
arr[len] = Field {
num: 12,
profile_type: ProfileType::UINT8,
value: Value::Uint8(m.resting_heart_rate),
is_expanded: false,
};
len += 1;
}
if m.threshold_heart_rate != u8::MAX {
arr[len] = Field {
num: 13,
profile_type: ProfileType::UINT8,
value: Value::Uint8(m.threshold_heart_rate),
is_expanded: false,
};
len += 1;
}
if m.pwr_calc_type != typedef::PwrZoneCalc(u8::MAX) {
arr[len] = Field {
num: 14,
profile_type: ProfileType::PWR_ZONE_CALC,
value: Value::Uint8(m.pwr_calc_type.0),
is_expanded: false,
};
len += 1;
}
if m.functional_threshold_power != u16::MAX {
arr[len] = Field {
num: 15,
profile_type: ProfileType::UINT16,
value: Value::Uint16(m.functional_threshold_power),
is_expanded: false,
};
len += 1;
}
Message {
header: 0,
num: typedef::MesgNum::TIME_IN_ZONE,
fields: {
let mut fields: Vec<Field> = Vec::with_capacity(len + m.unknown_fields.len());
fields.extend_from_slice(&arr[..len]);
fields.extend_from_slice(&m.unknown_fields);
fields
},
developer_fields: m.developer_fields,
}
}
}