aboutsummaryrefslogtreecommitdiff
path: root/src/encode.rs
blob: 30edc49e2eb4c7cfb882040fc1aac9446ddf0e73 (plain)
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
use std::borrow::Borrow;

use bitvec::{prelude::*, vec::BitVec};
use byteorder::{LittleEndian, WriteBytesExt};

use crate::{
    error::{OBIError, OBIResult},
    rle::RLE,
    CompressionType, Image,
};

pub fn encode_image<I>(obi_image: I) -> OBIResult<Vec<u8>>
where
    I: Borrow<Image>,
{
    let obi_image = obi_image.borrow();
    let mut obi_data = Vec::with_capacity(obi_image.file_header.file_size as usize);

    // file header
    let file_header = &obi_image.file_header;
    obi_data
        .write_u16::<LittleEndian>(file_header.version)
        .map_err(|_| OBIError::Encode)?;
    obi_data
        .write_u32::<LittleEndian>(file_header.file_size)
        .map_err(|_| OBIError::Encode)?;
    obi_data
        .write_u32::<LittleEndian>(file_header.data_offset)
        .map_err(|_| OBIError::Encode)?;

    // image info header
    let image_info_header = &obi_image.image_info_header;
    obi_data
        .write_u32::<LittleEndian>(image_info_header.width)
        .map_err(|_| OBIError::Encode)?;
    obi_data
        .write_u32::<LittleEndian>(image_info_header.height)
        .map_err(|_| OBIError::Encode)?;
    obi_data
        .write_u32::<LittleEndian>(image_info_header.compression_type)
        .map_err(|_| OBIError::Encode)?;
    obi_data
        .write_u32::<LittleEndian>(image_info_header.post_compression_size)
        .map_err(|_| OBIError::Encode)?;

    let write_pixel_data = |pixels: &Vec<bool>, obi_data: &mut Vec<u8>| -> OBIResult<()> {
        for byte in pixels.chunks(8) {
            let mut bv = BitVec::<Lsb0, u8>::new();
            for &b in byte {
                bv.push(b)
            }
            obi_data
                .write_u8(bv.load::<u8>())
                .map_err(|_| OBIError::Encode)?;
        }
        Ok(())
    };

    // pixmap data
    let pixmap = &obi_image.data;
    match CompressionType::from_u32(obi_image.image_info_header.compression_type) {
        CompressionType::RLE => {
            let (data_points, lengths): (Vec<_>, Vec<_>) = pixmap.compress().into_iter().unzip();
            for l in lengths {
                obi_data
                    .write_u32::<LittleEndian>(l as u32)
                    .map_err(|_| OBIError::Encode)?;
            }
            // end length sequence with zero
            obi_data
                .write_u32::<LittleEndian>(0)
                .map_err(|_| OBIError::Encode)?;
            // begin data point sequence
            write_pixel_data(&data_points, &mut obi_data)?;
        }
        _ => write_pixel_data(pixmap, &mut obi_data)?,
    };

    return Ok(obi_data);
}