diff options
Diffstat (limited to 'src/block')
-rw-r--r-- | src/block/block.rs | 5 | ||||
-rw-r--r-- | src/block/extension/graphiccontrol.rs | 59 | ||||
-rw-r--r-- | src/block/extension/mod.rs | 36 | ||||
-rw-r--r-- | src/block/mod.rs | 10 |
4 files changed, 102 insertions, 8 deletions
diff --git a/src/block/block.rs b/src/block/block.rs deleted file mode 100644 index 329361f..0000000 --- a/src/block/block.rs +++ /dev/null @@ -1,5 +0,0 @@ -use super::IndexedImage; - -pub enum Block { - IndexedImage(IndexedImage) -} \ No newline at end of file diff --git a/src/block/extension/graphiccontrol.rs b/src/block/extension/graphiccontrol.rs new file mode 100644 index 0000000..46f44fa --- /dev/null +++ b/src/block/extension/graphiccontrol.rs @@ -0,0 +1,59 @@ +pub struct GraphicControl { + pub(crate) packed: u8, + pub(crate) delay_time: u16, + pub(crate) transparency_index: u8 +} + +impl GraphicControl { + pub fn new(disposal_method: DisposalMethod, user_input_flag: bool, transparency_flag: bool, delay_time: u16, transparency_index: u8) -> Self { + let mut ret = Self { + packed: 0, + delay_time, + transparency_index + }; + + ret.disposal_method(disposal_method); + ret.user_input(user_input_flag); + ret.transparency(transparency_flag); + + ret + } + + pub fn disposal_method(&mut self, method: DisposalMethod) { + match method { + DisposalMethod::Clear => self.packed &= 0b111_000_1_1, + DisposalMethod::DoNotDispose => self.packed |= 0b000_100_0_0, + DisposalMethod::RestoreBackground => self.packed |= 0b000_010_0_0, + DisposalMethod::RestorePrevious => self.packed |= 0b000_110_0_0 + }; + } + + pub fn user_input(&mut self, flag: bool) { + if flag { + self.packed |= 0b000_000_1_0; + } else { + self.packed &= 0b111_111_0_1; + } + } + + pub fn transparency(&mut self, flag: bool) { + if flag { + self.packed |= 0b000_000_0_1; + } else { + self.packed &= 0b111_111_1_0; + } + } + + pub fn delay_time(&mut self, hundreths: u16) { + self.delay_time = hundreths; + } + + //TODO: Transparency index setter +} + +pub enum DisposalMethod { + Clear, + DoNotDispose, + RestoreBackground, + RestorePrevious +} \ No newline at end of file diff --git a/src/block/extension/mod.rs b/src/block/extension/mod.rs new file mode 100644 index 0000000..4d65c09 --- /dev/null +++ b/src/block/extension/mod.rs @@ -0,0 +1,36 @@ +mod graphiccontrol; + +pub use graphiccontrol::{DisposalMethod, GraphicControl}; + +pub enum Extension { + GraphicControl(GraphicControl), + Looping(u16) +} + +impl From<&Extension> for Box<[u8]> { + fn from(ext: &Extension) -> Self { + let mut vec = vec![]; + vec.push(0x21); // Push the extension introducer + + match ext { + Extension::GraphicControl(gc) => { + vec.push(0xF9); // Graphic control label + vec.push(0x04); // Block size for this extension is always 4 + vec.push(gc.packed); + vec.extend_from_slice(&gc.delay_time.to_le_bytes()); + vec.push(gc.transparency_index); + }, + Extension::Looping(count) => { + vec.push(0xFF); // Application extension label + vec.push(0x0B); // 11 bytes in this block + vec.extend_from_slice(b"NETSCAPE2.0"); // App. ident. and "auth code" + vec.push(0x03); // Sub-block length + vec.push(0x01); // Identifies netscape looping extension + vec.extend_from_slice(&count.to_le_bytes()); + } + } + + vec.push(0x00); // Zero-length data block indicates end of extension + vec.into_boxed_slice() + } +} \ No newline at end of file diff --git a/src/block/mod.rs b/src/block/mod.rs index a0c2454..38b10ea 100644 --- a/src/block/mod.rs +++ b/src/block/mod.rs @@ -1,13 +1,17 @@ -mod block; mod colortable; +pub mod extension; mod indexedimage; mod imagedescriptor; mod screendescriptor; mod version; -pub use block::Block; pub use colortable::ColorTable; pub use indexedimage::IndexedImage; pub use imagedescriptor::ImageDescriptor; pub use screendescriptor::ScreenDescriptor; -pub use version::Version; \ No newline at end of file +pub use version::Version; + +pub enum Block { + IndexedImage(IndexedImage), + Extension(extension::Extension) +} \ No newline at end of file |