use byteorder::NativeEndian;
use std::collections::BTreeMap;
use std::io::{Seek, Write};
use decoder::ifd::{self, Tag};
use error::{TiffError, TiffFormatError, TiffResult};
pub mod colortype;
mod writer;
use self::colortype::*;
use self::writer::*;
#[derive(Clone)]
pub struct Rational {
    pub n: u32,
    pub d: u32,
}
pub enum ResolutionUnit {
    None = 1,
    Inch = 2,
    Centimeter = 3,
}
pub trait TiffValue {
    const BYTE_LEN: u32;
    const FIELD_TYPE: ifd::Type;
    fn count(&self) -> u32;
    fn bytes(&self) -> u32 {
        self.count() * Self::BYTE_LEN
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()>;
}
impl TiffValue for [u8] {
    const BYTE_LEN: u32 = 1;
    const FIELD_TYPE: ifd::Type = ifd::Type::BYTE;
    fn count(&self) -> u32 {
        self.len() as u32
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        writer.write_bytes(self)?;
        Ok(())
    }
}
impl TiffValue for [u16] {
    const BYTE_LEN: u32 = 2;
    const FIELD_TYPE: ifd::Type = ifd::Type::SHORT;
    fn count(&self) -> u32 {
        self.len() as u32
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        
        let slice =
            unsafe { ::std::slice::from_raw_parts(self.as_ptr() as *const u8, self.len() * 2) };
        writer.write_bytes(slice)?;
        Ok(())
    }
}
impl TiffValue for [u32] {
    const BYTE_LEN: u32 = 4;
    const FIELD_TYPE: ifd::Type = ifd::Type::LONG;
    fn count(&self) -> u32 {
        self.len() as u32
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        
        let slice =
            unsafe { ::std::slice::from_raw_parts(self.as_ptr() as *const u8, self.len() * 4) };
        writer.write_bytes(slice)?;
        Ok(())
    }
}
impl TiffValue for [Rational] {
    const BYTE_LEN: u32 = 8;
    const FIELD_TYPE: ifd::Type = ifd::Type::RATIONAL;
    fn count(&self) -> u32 {
        self.len() as u32
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        for x in self {
            x.write(writer)?;
        }
        Ok(())
    }
}
impl TiffValue for u8 {
    const BYTE_LEN: u32 = 1;
    const FIELD_TYPE: ifd::Type = ifd::Type::BYTE;
    fn count(&self) -> u32 {
        1
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        writer.write_u8(*self)?;
        Ok(())
    }
}
impl TiffValue for u16 {
    const BYTE_LEN: u32 = 2;
    const FIELD_TYPE: ifd::Type = ifd::Type::SHORT;
    fn count(&self) -> u32 {
        1
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        writer.write_u16(*self)?;
        Ok(())
    }
}
impl TiffValue for u32 {
    const BYTE_LEN: u32 = 4;
    const FIELD_TYPE: ifd::Type = ifd::Type::LONG;
    fn count(&self) -> u32 {
        1
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        writer.write_u32(*self)?;
        Ok(())
    }
}
impl TiffValue for Rational {
    const BYTE_LEN: u32 = 8;
    const FIELD_TYPE: ifd::Type = ifd::Type::RATIONAL;
    fn count(&self) -> u32 {
        1
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        writer.write_u32(self.n)?;
        writer.write_u32(self.d)?;
        Ok(())
    }
}
impl TiffValue for str {
    const BYTE_LEN: u32 = 1;
    const FIELD_TYPE: ifd::Type = ifd::Type::ASCII;
    fn count(&self) -> u32 {
        self.len() as u32 + 1
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        if self.is_ascii() && !self.bytes().any(|b| b == 0) {
            writer.write_bytes(self.as_bytes())?;
            writer.write_u8(0)?;
            Ok(())
        } else {
            Err(TiffError::FormatError(TiffFormatError::InvalidTag))
        }
    }
}
impl<'a, T: TiffValue + ?Sized> TiffValue for &'a T {
    const BYTE_LEN: u32 = T::BYTE_LEN;
    const FIELD_TYPE: ifd::Type = T::FIELD_TYPE;
    fn count(&self) -> u32 {
        (*self).count()
    }
    fn write<W: Write>(&self, writer: &mut TiffWriter<W>) -> TiffResult<()> {
        (*self).write(writer)
    }
}
pub struct TiffEncoder<W> {
    writer: TiffWriter<W>,
}
impl<W: Write + Seek> TiffEncoder<W> {
    pub fn new(writer: W) -> TiffResult<TiffEncoder<W>> {
        let mut encoder = TiffEncoder {
            writer: TiffWriter::new(writer),
        };
        NativeEndian::write_header(&mut encoder.writer)?;
        Ok(encoder)
    }
    
    pub fn new_directory(&mut self) -> TiffResult<DirectoryEncoder<W>> {
        DirectoryEncoder::new(&mut self.writer)
    }
    
    pub fn new_image<C: ColorType>(
        &mut self,
        width: u32,
        height: u32,
    ) -> TiffResult<ImageEncoder<W, C>> {
        let encoder = DirectoryEncoder::new(&mut self.writer)?;
        ImageEncoder::new(encoder, width, height)
    }
    
    pub fn write_image<C: ColorType>(
        &mut self,
        width: u32,
        height: u32,
        data: &[C::Inner],
    ) -> TiffResult<()>
    where
        [C::Inner]: TiffValue,
    {
        let num_pix = (width as usize).checked_mul(height as usize)
            .ok_or_else(|| ::std::io::Error::new(
                ::std::io::ErrorKind::InvalidInput,
                "Image width * height exceeds usize"))?;
        if data.len() < num_pix {
            return Err(::std::io::Error::new(
                ::std::io::ErrorKind::InvalidData,
                "Input data slice is undersized for provided dimensions").into());
        }
        let encoder = DirectoryEncoder::new(&mut self.writer)?;
        let mut image: ImageEncoder<W, C> = ImageEncoder::new(encoder, width, height)?;
        let mut idx = 0;
        while image.next_strip_sample_count() > 0 {
            let sample_count = image.next_strip_sample_count() as usize;
            image.write_strip(&data[idx..idx + sample_count])?;
            idx += sample_count;
        }
        image.finish()
    }
}
pub struct DirectoryEncoder<'a, W: 'a + Write + Seek> {
    writer: &'a mut TiffWriter<W>,
    dropped: bool,
    
    ifd_pointer_pos: u64,
    ifd: BTreeMap<u16, (u16, u32, Vec<u8>)>,
}
impl<'a, W: 'a + Write + Seek> DirectoryEncoder<'a, W> {
    fn new(writer: &'a mut TiffWriter<W>) -> TiffResult<DirectoryEncoder<'a, W>> {
        writer.pad_word_boundary()?;
        let ifd_pointer_pos = writer.offset();
        writer.write_u32(0)?;
        Ok(DirectoryEncoder {
            writer,
            dropped: false,
            ifd_pointer_pos,
            ifd: BTreeMap::new(),
        })
    }
    
    pub fn write_tag<T: TiffValue>(&mut self, tag: Tag, value: T) {
        let len = <T>::BYTE_LEN * value.count();
        let mut bytes = Vec::with_capacity(len as usize);
        {
            let mut writer = TiffWriter::new(&mut bytes);
            value.write(&mut writer).unwrap();
        }
        self.ifd
            .insert(tag.to_u16(), (<T>::FIELD_TYPE as u16, value.count(), bytes));
    }
    fn write_directory(&mut self) -> TiffResult<u64> {
        
        for &mut (_, _, ref mut bytes) in self.ifd.values_mut() {
            if bytes.len() > 4 {
                let offset = self.writer.offset();
                self.writer.write_bytes(bytes)?;
                *bytes = vec![0, 0, 0, 0];
                let mut writer = TiffWriter::new(bytes as &mut [u8]);
                writer.write_u32(offset as u32)?;
            } else {
                while bytes.len() < 4 {
                    bytes.push(0);
                }
            }
        }
        let offset = self.writer.offset();
        self.writer.write_u16(self.ifd.len() as u16)?;
        for (tag, &(ref field_type, ref count, ref offset)) in self.ifd.iter() {
            self.writer.write_u16(*tag)?;
            self.writer.write_u16(*field_type)?;
            self.writer.write_u32(*count)?;
            self.writer.write_bytes(offset)?;
        }
        Ok(offset)
    }
    
    
    
    pub fn write_data<T: TiffValue>(&mut self, value: T) -> TiffResult<u64> {
        let offset = self.writer.offset();
        value.write(&mut self.writer)?;
        Ok(offset)
    }
    fn finish_internal(&mut self) -> TiffResult<()> {
        let ifd_pointer = self.write_directory()?;
        let curr_pos = self.writer.offset();
        self.writer.goto_offset(self.ifd_pointer_pos)?;
        self.writer.write_u32(ifd_pointer as u32)?;
        self.writer.goto_offset(curr_pos)?;
        self.writer.write_u32(0)?;
        self.dropped = true;
        Ok(())
    }
    
    pub fn finish(mut self) -> TiffResult<()> {
        self.finish_internal()
    }
}
impl<'a, W: Write + Seek> Drop for DirectoryEncoder<'a, W> {
    fn drop(&mut self) {
        if !self.dropped {
            let _ = self.finish_internal();
        }
    }
}
pub struct ImageEncoder<'a, W: 'a + Write + Seek, C: ColorType> {
    encoder: DirectoryEncoder<'a, W>,
    strip_idx: u64,
    strip_count: u64,
    row_samples: u64,
    height: u32,
    rows_per_strip: u64,
    strip_offsets: Vec<u32>,
    strip_byte_count: Vec<u32>,
    dropped: bool,
    _phantom: ::std::marker::PhantomData<C>,
}
impl<'a, W: 'a + Write + Seek, T: ColorType> ImageEncoder<'a, W, T> {
    fn new(
        mut encoder: DirectoryEncoder<'a, W>,
        width: u32,
        height: u32,
    ) -> TiffResult<ImageEncoder<'a, W, T>> {
        let row_samples = u64::from(width) * <T>::BITS_PER_SAMPLE.len() as u64;
        let row_bytes = row_samples * u64::from(<T::Inner>::BYTE_LEN);
        
        let rows_per_strip = (8000 + row_bytes - 1) / row_bytes;
        let strip_count = (u64::from(height) + rows_per_strip - 1) / rows_per_strip;
        encoder.write_tag(Tag::ImageWidth, width);
        encoder.write_tag(Tag::ImageLength, height);
        encoder.write_tag(Tag::Compression, 1u16);
        encoder.write_tag(Tag::BitsPerSample, <T>::BITS_PER_SAMPLE);
        encoder.write_tag(Tag::PhotometricInterpretation, <T>::TIFF_VALUE as u16);
        encoder.write_tag(Tag::RowsPerStrip, rows_per_strip as u32);
        encoder.write_tag(Tag::SamplesPerPixel, <T>::BITS_PER_SAMPLE.len() as u16);
        encoder.write_tag(Tag::XResolution, Rational { n: 1, d: 1 });
        encoder.write_tag(Tag::YResolution, Rational { n: 1, d: 1 });
        encoder.write_tag(Tag::ResolutionUnit, 1u16);
        Ok(ImageEncoder {
            encoder,
            strip_count,
            strip_idx: 0,
            row_samples,
            rows_per_strip,
            height,
            strip_offsets: Vec::new(),
            strip_byte_count: Vec::new(),
            dropped: false,
            _phantom: ::std::marker::PhantomData,
        })
    }
    
    pub fn next_strip_sample_count(&self) -> u64 {
        if self.strip_idx >= self.strip_count {
            return 0;
        }
        let start_row = ::std::cmp::min(u64::from(self.height), self.strip_idx * self.rows_per_strip);
        let end_row = ::std::cmp::min(
            u64::from(self.height),
            (self.strip_idx + 1) * self.rows_per_strip,
        );
        (end_row - start_row) * self.row_samples
    }
    
    pub fn write_strip(&mut self, value: &[T::Inner]) -> TiffResult<()>
    where
        [T::Inner]: TiffValue,
    {
        
        let samples = self.next_strip_sample_count();
        if value.len() as u64 != samples {
            return Err(::std::io::Error::new(
                ::std::io::ErrorKind::InvalidData,
                "Slice is wrong size for strip").into());
        }
        let offset = self.encoder.write_data(value)?;
        self.strip_offsets.push(offset as u32);
        self.strip_byte_count.push(value.bytes() as u32);
        self.strip_idx += 1;
        Ok(())
    }
    
    pub fn resolution(&mut self, unit: ResolutionUnit, value: Rational) {
        self.encoder.write_tag(Tag::ResolutionUnit, unit as u16);
        self.encoder.write_tag(Tag::XResolution, value.clone());
        self.encoder.write_tag(Tag::YResolution, value);
    }
    
    pub fn resolution_unit(&mut self, unit: ResolutionUnit) {
        self.encoder.write_tag(Tag::ResolutionUnit, unit as u16);
    }
    
    pub fn x_resolution(&mut self, value: Rational) {
        self.encoder.write_tag(Tag::XResolution, value);
    }
    
    pub fn y_resolution(&mut self, value: Rational) {
        self.encoder.write_tag(Tag::YResolution, value);
    }
    fn finish_internal(&mut self) -> TiffResult<()> {
        self.encoder
            .write_tag(Tag::StripOffsets, &*self.strip_offsets);
        self.encoder
            .write_tag(Tag::StripByteCounts, &*self.strip_byte_count);
        self.dropped = true;
        self.encoder.finish_internal()
    }
    
    pub fn encoder(&mut self) -> &mut DirectoryEncoder<'a, W> {
        &mut self.encoder
    }
    
    pub fn finish(mut self) -> TiffResult<()> {
        self.finish_internal()
    }
}
impl<'a, W: Write + Seek, C: ColorType> Drop for ImageEncoder<'a, W, C> {
    fn drop(&mut self) {
        if !self.dropped {
            let _ = self.finish_internal();
        }
    }
}