use std::io;
use std::collections::{HashMap};
use super::decoder::{ByteOrder, SmartReader};
macro_rules! tags {
{$(
$tag:ident
$val:expr;
)*} => {
#[deriving(PartialEq, Eq, Show, Hash)]
pub enum Tag {
$($tag,)*
Unknown(u16)
}
impl Tag {
pub fn from_u16(n: u16) -> Tag {
$(if n == $val { Tag::$tag } else)* {
Tag::Unknown(n)
}
}
}
}
}
tags!{
PhotometricInterpretation 262;
Compression 259;
ImageLength 257;
ImageWidth 256;
ResolutionUnit 296; XResolution 282;
YResolution 283;
RowsPerStrip 278;
StripOffsets 273;
StripByteCounts 279;
BitsPerSample 258;
ColorMap 320;
SamplesPerPixel 277;
}
#[deriving(Show, FromPrimitive)]
pub enum Type {
BYTE = 1,
ASCII = 2,
SHORT = 3,
LONG = 4,
RATIONAL = 5,
}
#[deriving(Show)]
pub enum Value {
Unsigned(u32),
List(Vec<Value>)
}
impl Value {
pub fn as_u32(self) -> ::image::ImageResult<u32> {
match self {
Value::Unsigned(val) => Ok(val),
val => Err(::image::ImageError::FormatError(format!(
"Expected unsigned integer, {} found.", val
)))
}
}
pub fn as_u32_vec(self) -> ::image::ImageResult<Vec<u32>> {
match self {
Value::List(vec) => {
let mut new_vec = Vec::with_capacity(vec.len());
for v in vec.into_iter() {
new_vec.push(try!(v.as_u32()))
}
Ok(new_vec)
},
Value::Unsigned(val) => Ok(vec![val]),
}
}
}
pub struct Entry {
type_: Type,
count: u32,
offset: [u8, ..4],
}
impl ::std::fmt::Show for Entry {
fn fmt(&self, fmt: &mut ::std::fmt::Formatter) -> Result<(), ::std::fmt::Error> {
fmt.write(format!("Entry {{ type: {}, count: {}, offset: {} }}",
self.type_,
self.count,
self.offset.as_slice()
).as_bytes())
}
}
impl Entry {
pub fn new(type_: Type, count: u32, offset: [u8, ..4]) -> Entry {
Entry {
type_: type_,
count: count,
offset: offset
}
}
fn r(&self, byte_order: ByteOrder) -> SmartReader<io::MemReader> {
SmartReader::wrap(
io::MemReader::new(self.offset.as_slice().to_vec()),
byte_order
)
}
pub fn val<R: Reader + Seek>(&self, decoder: &mut super::TIFFDecoder<R>)
-> ::image::ImageResult<Value> {
let bo = decoder.byte_order();
match (self.type_, self.count) {
(Type::BYTE, 1) => Ok(Value::Unsigned(self.offset[0] as u32)),
(Type::SHORT, 1) => Ok(Value::Unsigned(try!(self.r(bo).read_u16()) as u32)),
(Type::LONG, 1) => Ok(Value::Unsigned(try!(self.r(bo).read_u32()))),
(Type::LONG, n) => {
let mut v = Vec::with_capacity(n as uint);
try!(decoder.goto_offset(try!(self.r(bo).read_u32())));
for _ in range(0, n) {
v.push(Value::Unsigned(try!(decoder.read_long())))
}
Ok(Value::List(v))
}
_ => Err(::image::ImageError::UnsupportedError("Unsupported data type.".to_string()))
}
}
}
pub type Directory = HashMap<Tag, Entry>;