2019-03-11 10:46:01 +04:00

136 lines
5.0 KiB
C#

//! \file ImageHG2.cs
//! \date Sun Nov 29 06:33:49 2015
//! \brief CatSystem HG2 image format implementation.
//
// 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 System.ComponentModel.Composition;
using System.IO;
using System.Windows.Media;
namespace GameRes.Formats.CatSystem
{
internal class Hg2MetaData : HgMetaData
{
public int Version;
public int DataPacked;
public int DataUnpacked;
public int CtlPacked;
public int CtlUnpacked;
public int Depth;
}
[Export(typeof(ImageFormat))]
public class Hg2Format : ImageFormat
{
public override string Tag { get { return "HG2"; } }
public override string Description { get { return "CatSystem engine image format"; } }
public override uint Signature { get { return 0x322D4748; } } // 'HG-2'
public override ImageMetaData ReadMetaData (IBinaryStream stream)
{
stream.Position = 8;
var info = new Hg2MetaData();
int version = stream.ReadInt32();
if (0x25 == version)
info.HeaderSize = 0x58;
else if (0x20 == version)
info.HeaderSize = 0x50;
else if (0x10 == version)
info.HeaderSize = 0x30;
else
return null;
info.Version = version;
info.Width = stream.ReadUInt32();
info.Height = stream.ReadUInt32();
info.BPP = stream.ReadInt16();
info.Depth = stream.ReadInt16();
stream.Seek (8, SeekOrigin.Current);
info.DataPacked = stream.ReadInt32();
info.DataUnpacked = stream.ReadInt32();
info.CtlPacked = stream.ReadInt32();
info.CtlUnpacked = stream.ReadInt32();
if (info.HeaderSize > 0x30)
{
stream.Seek (8, SeekOrigin.Current);
info.CanvasWidth = stream.ReadUInt32();
info.CanvasHeight = stream.ReadUInt32();
info.OffsetX = stream.ReadInt32();
info.OffsetY = stream.ReadInt32();
}
return info;
}
public override ImageData Read (IBinaryStream stream, ImageMetaData info)
{
using (var reader = new Hg2Reader (stream, (Hg2MetaData)info))
return reader.Image;
}
public override void Write (Stream file, ImageData image)
{
throw new System.NotImplementedException ("Hg2Format.Write not implemented");
}
}
internal sealed class Hg2Reader : HgReader
{
Hg2MetaData m_hg2;
public override ImageData Image
{
get { return m_image ?? (m_image = CreateImage()); }
}
public Hg2Reader (IBinaryStream input, Hg2MetaData info) : base (input, info)
{
m_hg2 = info;
}
ImageData CreateImage ()
{
var pixels = UnpackStream (m_hg2.HeaderSize, m_hg2.DataPacked, m_hg2.DataUnpacked,
m_hg2.CtlPacked, m_hg2.CtlUnpacked);
var format = 24 == m_hg2.BPP ? PixelFormats.Bgr24 : PixelFormats.Bgra32;
if (m_hg2.Version > 0x10)
return ImageData.CreateFlipped (m_hg2, format, null, pixels, Stride);
if (m_hg2.Depth > 0 && m_hg2.Depth < 8)
{
byte max = (byte)((1 << m_hg2.Depth) - 1);
for (int i = 0; i < pixels.Length; i += m_pixel_size)
{
pixels[i ] = (byte)(pixels[i ] * 0xFF / max);
pixels[i+1] = (byte)(pixels[i+1] * 0xFF / max);
pixels[i+2] = (byte)(pixels[i+2] * 0xFF / max);
}
}
if (32 == m_hg2.BPP)
{
for (int i = 3; i < pixels.Length; i += 4)
pixels[i] ^= 0xFF;
}
return ImageData.Create (m_hg2, format, null, pixels, Stride);
}
}
}