diff options
Diffstat (limited to 'src/block/colortable.rs')
-rw-r--r-- | src/block/colortable.rs | 101 |
1 files changed, 58 insertions, 43 deletions
diff --git a/src/block/colortable.rs b/src/block/colortable.rs index 97b428f..da9458c 100644 --- a/src/block/colortable.rs +++ b/src/block/colortable.rs @@ -1,67 +1,82 @@ -use std::ops::Deref; pub use crate::Color; +use std::{ + convert::{TryFrom, TryInto}, + ops::Deref, +}; pub struct ColorTable { - table: Vec<Color> + table: Vec<Color>, } impl ColorTable { - pub fn new() -> Self { - Self { - table: vec![] - } - } + pub fn new() -> Self { + Self { table: vec![] } + } - /// Returns the number of colors in the color table as used by the packed - /// fields in the Logical Screen Descriptor and Image Descriptor. You can - /// get the actual size with the [`len`](struct.ColorTable.html#method.len) method. - pub fn packed_len(&self) -> u8 { - ((self.table.len() as f32).log2().ceil() - 1f32) as u8 - } + /// Returns the number of colors in the color table as used by the packed + /// fields in the Logical Screen Descriptor and Image Descriptor. You can + /// get the actual size with the [`len`](struct.ColorTable.html#method.len) method. + pub fn packed_len(&self) -> u8 { + ((self.table.len() as f32).log2().ceil() - 1f32) as u8 + } - /// Returns the number of items in the table - pub fn len(&self) -> usize { - self.table.len() - } + /// Returns the number of items in the table + pub fn len(&self) -> usize { + self.table.len() + } - /// Pushes a color on to the end of the table - pub fn push(&mut self, color: Color) { - self.table.push(color); - } + /// Pushes a color on to the end of the table + pub fn push(&mut self, color: Color) { + self.table.push(color); + } } impl Deref for ColorTable { - type Target = [Color]; + type Target = [Color]; - fn deref(&self) -> &Self::Target { - &self.table - } + fn deref(&self) -> &Self::Target { + &self.table + } } impl From<Vec<Color>> for ColorTable { - fn from(table: Vec<Color>) -> Self{ - ColorTable { - table - } - } + fn from(table: Vec<Color>) -> Self { + ColorTable { table } + } } impl From<&ColorTable> for Box<[u8]> { - fn from(table: &ColorTable) -> Self { - let mut vec = vec![]; + fn from(table: &ColorTable) -> Self { + let mut vec = vec![]; - for color in table.iter() { - vec.extend_from_slice(&[color.r, color.g, color.b]); - } + for color in table.iter() { + vec.extend_from_slice(&[color.r, color.g, color.b]); + } - let packed_len = 2u8.pow(table.packed_len() as u32 + 1); - let padding = (packed_len as usize - table.len()) * 3; - if padding > 0 { - vec.extend_from_slice(&vec![0; padding]); - } + let packed_len = 2usize.pow(table.packed_len() as u32 + 1); + let padding = (packed_len as usize - table.len()) * 3; + if padding > 0 { + vec.extend_from_slice(&vec![0; padding]); + } - vec.into_boxed_slice() - } + vec.into_boxed_slice() + } } -//TODO: TryFrom Vec<u8> (must be multiple of 3 len) and From Vec<Color> \ No newline at end of file +//TODO: TryFrom Vec<u8> (must be multiple of 3 len) and From Vec<Color> +impl TryFrom<&[u8]> for ColorTable { + type Error = (); + + fn try_from(value: &[u8]) -> Result<Self, Self::Error> { + if value.len() % 3 != 0 { + return Err(()); + } else { + Ok(Self { + table: value + .chunks(3) + .map(|slice| Color::from(TryInto::<[u8; 3]>::try_into(slice).unwrap())) + .collect::<Vec<Color>>(), + }) + } + } +} |