//! \file ImageHZC.cs //! \date Tue Dec 08 22:54:11 2015 //! \brief Favorite View Point image format. // // Copyright (C) 2015 by morkt // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING // FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS // IN THE SOFTWARE. // using GameRes.Compression; using GameRes.Utility; using System; using System.ComponentModel.Composition; using System.IO; using System.Windows.Media; using System.Windows.Media.Imaging; namespace GameRes.Formats.FVP { internal class HzcMetaData : ImageMetaData { public int Type; public int UnpackedSize; public int HeaderSize; } [Export(typeof(ImageFormat))] public class HzcFormat : ImageFormat { public override string Tag { get { return "HZC"; } } public override string Description { get { return "Favorite View Point image format"; } } public override uint Signature { get { return 0x31637A68; } } // 'HZC1' public override ImageMetaData ReadMetaData (IBinaryStream stream) { var header = stream.ReadHeader (0x2C); if (!header.AsciiEqual (0xC, "NVSG")) return null; int type = header.ToUInt16 (0x12); return new HzcMetaData { Width = header.ToUInt16 (0x14), Height = header.ToUInt16 (0x16), OffsetX = header.ToInt16 (0x18), OffsetY = header.ToInt16 (0x1A), BPP = 0 == type ? 24 : type > 2 ? 8 : 32, Type = type, UnpackedSize = header.ToInt32 (4), HeaderSize = header.ToInt32 (8), }; } public override ImageData Read (IBinaryStream stream, ImageMetaData info) { var meta = (HzcMetaData)info; stream.Position = 12 + meta.HeaderSize; using (var decoder = new HzcDecoder (stream, meta, true)) return decoder.Image; } public override void Write (Stream file, ImageData image) { throw new System.NotImplementedException ("HzcFormat.Write not implemented"); } } internal sealed class HzcDecoder : IImageDecoder { HzcMetaData m_info; ImageData m_image; int m_stride; long m_frame_offset; int m_frame_size; public Stream Source { get; private set; } public ImageFormat SourceFormat { get { return null; } } public ImageMetaData Info { get { return m_info; } } public PixelFormat Format { get; private set; } public BitmapPalette Palette { get; private set; } public ImageData Image { get { if (null == m_image) { var pixels = ReadPixels(); m_image = ImageData.Create (Info, Format, Palette, pixels, m_stride); } return m_image; } } public HzcDecoder (IBinaryStream input, HzcMetaData info, Entry entry) : this (input, info) { m_frame_offset = entry.Offset; m_frame_size = (int)entry.Size; } public HzcDecoder (IBinaryStream input, HzcMetaData info, bool leave_open = false) { m_info = info; m_stride = (int)m_info.Width * m_info.BPP / 8; switch (m_info.Type) { default: throw new NotSupportedException(); case 0: Format = PixelFormats.Bgr24; break; case 1: case 2: Format = PixelFormats.Bgra32; break; case 3: Format = PixelFormats.Gray8; break; case 4: { Format = PixelFormats.Indexed8; var colors = new Color[2] { Color.FromRgb (0,0,0), Color.FromRgb (0xFF,0xFF,0xFF) }; Palette = new BitmapPalette (colors); break; } } Source = new ZLibStream (input.AsStream, CompressionMode.Decompress, leave_open); m_frame_offset = 0; m_frame_size = m_stride * (int)Info.Height; } byte[] ReadPixels () { var pixels = new byte[m_frame_size]; long offset = 0; for (;;) { if (pixels.Length != Source.Read (pixels, 0, pixels.Length)) throw new EndOfStreamException(); if (offset >= m_frame_offset) break; offset += m_frame_size; } return pixels; } bool m_disposed = false; public void Dispose () { if (!m_disposed) { Source.Dispose(); m_disposed = true; } GC.SuppressFinalize (this); } } }