1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
|
use crate::block::{Block, ColorTable, extension::Extension, ScreenDescriptor, Version};
pub struct Gif {
pub header: Version,
pub screen_descriptor: ScreenDescriptor,
pub global_color_table: Option<ColorTable>,
pub blocks: Vec<Block>
// Trailer at the end of this struct is 0x3B //
}
impl Gif {
pub fn to_vec(&self) -> Vec<u8> {
let mut out = vec![];
out.extend_from_slice((&self.header).into());
let mut boxed: Box<[u8]> = (&self.screen_descriptor).into();
out.extend_from_slice(&*boxed);
// While we output the color table, grab it's length to use when
// outputting the image, or 0 if we don't have a GCT
let mcs = if let Some(gct) = &self.global_color_table {
boxed = gct.into();
out.extend_from_slice(&*boxed);
gct.packed_len()
} else {
0
};
for block in self.blocks.iter() {
match block {
Block::IndexedImage(image) => {
boxed = image.as_boxed_slice(mcs);
out.extend_from_slice(&*boxed);
},
Block::Extension(ext) => {
boxed = ext.into();
out.extend_from_slice(&*boxed);
}
}
}
// Write Trailer
out.push(0x3B);
out
}
}
#[cfg(test)]
pub mod gif {
use crate::Color;
use crate::writer::{GifBuilder, ImageBuilder};
use crate::block::extension::{DisposalMethod, GraphicControl};
use super::*;
#[test]
fn to_vec_gif87a() {
let gct = vec![
Color::new(1, 2, 3), Color::new(253, 254, 255)
];
let colortable = vec![
Color::new(0, 0, 0), Color::new(128, 0, 255)
];
let indicies = vec![0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0];
let expected_out = vec![
0x47, 0x49, 0x46, 0x38, 0x37, 0x61, // Version - GIF87a
0x04, 0x00, 0x04, 0x00, 0b1000_0000, 0x00, 0x00, // Logical Screen Descriptor
1, 2, 3, 253, 254, 255, // Global Color Table
0x2C, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x04, 0x00, 0b1000_0000, // Image Descriptor 1
0, 0, 0, 128, 0, 255, // Color Table
0x02, 0x05, 0x84, 0x1D, 0x81, 0x7A, 0x50, 0x00, // Image Data 1
0x2C, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x04, 0x00, 0b0000_0000, // Image Descriptor 2
0x02, 0x05, 0x84, 0x1D, 0x81, 0x7A, 0x50, 0x00, // Image Data 2
0x3B // Trailer
];
let actual_out = GifBuilder::new(Version::Gif87a, 4, 4)
.global_color_table(gct.into())
.image(ImageBuilder::new(4, 4)
.color_table(colortable.into())
.indicies(indicies.clone())
).image(ImageBuilder::new(4, 4)
.indicies(indicies)
).build().to_vec();
assert_eq!(actual_out, expected_out);
}
#[test]
fn to_vec_gif89a() {
let gct = vec![
Color::new(1, 2, 3), Color::new(253, 254, 255)
];
let colortable = vec![
Color::new(0, 0, 0), Color::new(128, 0, 255)
];
let indicies = vec![0, 0, 0, 0, 0, 1, 1, 0, 0, 1, 1, 0, 0, 0, 0, 0];
let expected_out = vec![
0x47, 0x49, 0x46, 0x38, 0x37, 0x61, // Version - GIF87a
0x04, 0x00, 0x04, 0x00, 0b1000_0000, 0x00, 0x00, // Logical Screen Descriptor
1, 2, 3, 253, 254, 255, // Global Color Table
0x2C, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x04, 0x00, 0b1000_0000, // Image Descriptor 1
0, 0, 0, 128, 0, 255, // Color Table
0x02, 0x05, 0x84, 0x1D, 0x81, 0x7A, 0x50, 0x00, // Image Data 1
0x21, 0xF9, 0x04, 0b000_010_0_0, 0x40, 0x00, 0x00, 0x00, // Graphic Control Extension
0x2C, 0x00, 0x00, 0x00, 0x00, 0x04, 0x00, 0x04, 0x00, 0b0000_0000, // Image Descriptor 2
0x02, 0x05, 0x84, 0x1D, 0x81, 0x7A, 0x50, 0x00, // Image Data 2
0x3B // Trailer
];
let actual_out = GifBuilder::new(Version::Gif87a, 4, 4)
.global_color_table(gct.into())
.image(ImageBuilder::new(4, 4)
.color_table(colortable.into())
.indicies(indicies.clone())
).extension(Extension::GraphicControl(GraphicControl::new(DisposalMethod::RestoreBackground, false, false, 64, 0)))
.image(ImageBuilder::new(4, 4)
.indicies(indicies)
).build().to_vec();
assert_eq!(actual_out, expected_out);
}
}
|