use std::io::Write;
use byteorder::{LittleEndian, WriteBytesExt};
use crate::error::Result;
use crate::types::Handle;
use super::stream_writer::DxfStreamWriter;
const BINARY_DXF_SENTINEL: &[u8] = b"AutoCAD Binary DXF\r\n\x1a\x00";
pub struct DxfBinaryWriter<W: Write> {
writer: W,
hex_buf: [u8; 17],
}
impl<W: Write> DxfBinaryWriter<W> {
pub fn new(mut writer: W) -> Result<Self> {
writer.write_all(BINARY_DXF_SENTINEL)?;
Ok(Self { writer, hex_buf: [0u8; 17] })
}
fn write_code(&mut self, code: i32) -> Result<()> {
self.writer.write_i16::<LittleEndian>(code as i16)?;
Ok(())
}
fn write_null_string(&mut self, value: &str) -> Result<()> {
self.writer.write_all(value.as_bytes())?;
self.writer.write_u8(0)?;
Ok(())
}
pub fn into_inner(self) -> W {
self.writer
}
}
impl<W: Write> DxfStreamWriter for DxfBinaryWriter<W> {
fn write_string(&mut self, code: i32, value: &str) -> Result<()> {
self.write_code(code)?;
self.write_null_string(value)?;
Ok(())
}
fn write_byte(&mut self, code: i32, value: u8) -> Result<()> {
self.write_code(code)?;
self.writer.write_i16::<LittleEndian>(value as i16)?;
Ok(())
}
fn write_i16(&mut self, code: i32, value: i16) -> Result<()> {
self.write_code(code)?;
self.writer.write_i16::<LittleEndian>(value)?;
Ok(())
}
fn write_i32(&mut self, code: i32, value: i32) -> Result<()> {
self.write_code(code)?;
self.writer.write_i32::<LittleEndian>(value)?;
Ok(())
}
fn write_i64(&mut self, code: i32, value: i64) -> Result<()> {
self.write_code(code)?;
self.writer.write_i64::<LittleEndian>(value)?;
Ok(())
}
fn write_double(&mut self, code: i32, value: f64) -> Result<()> {
self.write_code(code)?;
self.writer.write_f64::<LittleEndian>(value)?;
Ok(())
}
fn write_bool(&mut self, code: i32, value: bool) -> Result<()> {
self.write_code(code)?;
self.writer.write_u8(if value { 1 } else { 0 })?;
Ok(())
}
fn write_handle(&mut self, code: i32, handle: Handle) -> Result<()> {
self.write_code(code)?;
let val = handle.value();
if val == 0 {
self.writer.write_all(b"0\0")?;
} else {
let mut pos = 16usize;
let mut v = val;
while v > 0 {
pos -= 1;
let digit = (v & 0xF) as u8;
self.hex_buf[pos] = if digit < 10 { b'0' + digit } else { b'A' + digit - 10 };
v >>= 4;
}
let hex_len = 16 - pos;
self.hex_buf.copy_within(pos..16, 0);
self.hex_buf[hex_len] = 0; self.writer.write_all(&self.hex_buf[..hex_len + 1])?;
}
Ok(())
}
fn write_binary(&mut self, code: i32, data: &[u8]) -> Result<()> {
self.write_code(code)?;
self.writer.write_u8(data.len() as u8)?;
self.writer.write_all(data)?;
Ok(())
}
fn flush(&mut self) -> Result<()> {
self.writer.flush()?;
Ok(())
}
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
fn test_binary_sentinel() {
let mut buf = Vec::new();
{
let _writer = DxfBinaryWriter::new(&mut buf).unwrap();
}
assert!(buf.starts_with(BINARY_DXF_SENTINEL));
}
#[test]
fn test_write_string() {
let mut buf = Vec::new();
{
let mut writer = DxfBinaryWriter::new(&mut buf).unwrap();
writer.write_string(0, "LINE").unwrap();
}
let sentinel_len = BINARY_DXF_SENTINEL.len();
assert_eq!(buf[sentinel_len..sentinel_len+2], [0, 0]); assert_eq!(&buf[sentinel_len+2..sentinel_len+6], b"LINE");
assert_eq!(buf[sentinel_len+6], 0); }
#[test]
fn test_write_double() {
let mut buf = Vec::new();
{
let mut writer = DxfBinaryWriter::new(&mut buf).unwrap();
writer.write_double(10, 1.5).unwrap();
}
let sentinel_len = BINARY_DXF_SENTINEL.len();
assert_eq!(buf[sentinel_len..sentinel_len+2], [10, 0]); let expected: [u8; 8] = 1.5f64.to_le_bytes();
assert_eq!(&buf[sentinel_len+2..sentinel_len+10], &expected);
}
#[test]
fn test_write_i16() {
let mut buf = Vec::new();
{
let mut writer = DxfBinaryWriter::new(&mut buf).unwrap();
writer.write_i16(62, 7).unwrap();
}
let sentinel_len = BINARY_DXF_SENTINEL.len();
assert_eq!(buf[sentinel_len..sentinel_len+2], [62, 0]); assert_eq!(buf[sentinel_len+2..sentinel_len+4], [7, 0]); }
}