use super::{unpack_type, Error, Reader, ReaderIterator};
use crate::{BitWidth, Buffer, FlexBufferType};
pub struct VectorReader<B> {
pub(super) reader: Reader<B>,
pub(super) length: usize,
}
impl<B: Buffer> Clone for VectorReader<B> {
fn clone(&self) -> Self {
VectorReader { reader: self.reader.clone(), ..*self }
}
}
impl<B: Buffer> Default for VectorReader<B> {
fn default() -> Self {
VectorReader { reader: Reader::default(), length: usize::default() }
}
}
impl<B: Buffer> VectorReader<B> {
pub fn len(&self) -> usize {
self.length
}
pub fn is_empty(&self) -> bool {
self.length == 0
}
fn get_elem_type(&self, i: usize) -> Result<(FlexBufferType, BitWidth), Error> {
if let Some(ty) = self.reader.fxb_type.typed_vector_type() {
Ok((ty, self.reader.width))
} else {
let types_addr = self.reader.address + self.length * self.reader.width.n_bytes();
self.reader
.buffer
.get(types_addr + i)
.ok_or(Error::FlexbufferOutOfBounds)
.and_then(|&t| unpack_type(t))
}
}
pub fn idx(&self, i: usize) -> Reader<B> {
self.index(i).unwrap_or_default()
}
pub fn index(&self, i: usize) -> Result<Reader<B>, Error> {
if i >= self.length {
return Err(Error::IndexOutOfBounds);
}
let (fxb_type, bw) = self.get_elem_type(i)?;
let data_address = self.reader.address + self.reader.width.n_bytes() * i;
Reader::new(
self.reader.buffer.shallow_copy(),
data_address,
fxb_type,
bw,
self.reader.width,
)
}
pub fn iter(&self) -> ReaderIterator<B> {
ReaderIterator::new(self.clone())
}
}