mirror of
https://github.com/crskycode/GARbro.git
synced 2025-01-15 14:23:54 +08:00
213 lines
8.1 KiB
C#
213 lines
8.1 KiB
C#
|
//! \file ImageQNT.cs
|
||
|
//! \date Thu Apr 09 21:37:18 2015
|
||
|
//! \brief AliceSoft RGB 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 System;
|
||
|
using System.ComponentModel.Composition;
|
||
|
using System.IO;
|
||
|
using System.Windows;
|
||
|
using System.Windows.Media;
|
||
|
using System.Windows.Media.Imaging;
|
||
|
using GameRes.Utility;
|
||
|
using ZLibNet;
|
||
|
|
||
|
namespace GameRes.Formats.AliceSoft
|
||
|
{
|
||
|
public class QntMetaData : ImageMetaData
|
||
|
{
|
||
|
public uint RGBSize;
|
||
|
public uint AlphaSize;
|
||
|
}
|
||
|
|
||
|
[Export(typeof(ImageFormat))]
|
||
|
public class QntFormat : ImageFormat
|
||
|
{
|
||
|
public override string Tag { get { return "QNT"; } }
|
||
|
public override string Description { get { return "AliceSoft System image format"; } }
|
||
|
public override uint Signature { get { return 0x544e51; } } // 'QNT'
|
||
|
|
||
|
public override void Write (Stream file, ImageData image)
|
||
|
{
|
||
|
throw new System.NotImplementedException ("QntFormat.Write not implemented");
|
||
|
}
|
||
|
|
||
|
public override ImageMetaData ReadMetaData (Stream stream)
|
||
|
{
|
||
|
var header = new byte[0x44];
|
||
|
if (header.Length != stream.Read (header, 0, header.Length))
|
||
|
return null;
|
||
|
int version = LittleEndian.ToInt32 (header, 4);
|
||
|
if (version <= 0 || version > 2)
|
||
|
return null;
|
||
|
if (0x44 != LittleEndian.ToUInt32 (header, 8))
|
||
|
return null;
|
||
|
uint width = LittleEndian.ToUInt32 (header, 0x14);
|
||
|
uint height = LittleEndian.ToUInt32 (header, 0x18);
|
||
|
if (0 == width || 0 == height)
|
||
|
return null;
|
||
|
return new QntMetaData
|
||
|
{
|
||
|
Width = width,
|
||
|
Height = height,
|
||
|
OffsetX = LittleEndian.ToInt32 (header, 0x0c),
|
||
|
OffsetY = LittleEndian.ToInt32 (header, 0x10),
|
||
|
BPP = LittleEndian.ToInt32 (header, 0x1c),
|
||
|
RGBSize = LittleEndian.ToUInt32 (header, 0x24),
|
||
|
AlphaSize = LittleEndian.ToUInt32 (header, 0x28),
|
||
|
};
|
||
|
}
|
||
|
|
||
|
public override ImageData Read (Stream stream, ImageMetaData info)
|
||
|
{
|
||
|
var meta = info as QntMetaData;
|
||
|
if (null == meta)
|
||
|
throw new ArgumentException ("QntFormat.Read should be supplied with QntMetaData", "info");
|
||
|
stream.Position = 0x44;
|
||
|
using (var reader = new Reader (stream, meta))
|
||
|
{
|
||
|
reader.Unpack();
|
||
|
var pixels = reader.Data;
|
||
|
int stride = (int)info.Width * (reader.BPP / 8);
|
||
|
PixelFormat format = 24 == reader.BPP ? PixelFormats.Bgr24 : PixelFormats.Bgra32;
|
||
|
var bitmap = BitmapSource.Create ((int)info.Width, (int)info.Height, 96, 96,
|
||
|
format, null, pixels, stride);
|
||
|
bitmap.Freeze();
|
||
|
return new ImageData (bitmap, info);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
internal class Reader : IDisposable
|
||
|
{
|
||
|
byte[] m_input;
|
||
|
byte[] m_alpha;
|
||
|
byte[] m_output;
|
||
|
int m_bpp;
|
||
|
int m_width;
|
||
|
int m_height;
|
||
|
|
||
|
public byte[] Data { get { return m_output; } }
|
||
|
public int BPP { get { return m_bpp*8; } }
|
||
|
|
||
|
public Reader (Stream stream, QntMetaData info)
|
||
|
{
|
||
|
m_width = (int)info.Width;
|
||
|
m_height = (int)info.Height;
|
||
|
int w = (m_width + 1) & ~1;
|
||
|
int h = (m_height + 1) & ~1;
|
||
|
int rgb_size = h * w * 3;
|
||
|
m_bpp = info.AlphaSize != 0 ? 4 : 3;
|
||
|
m_input = new byte[rgb_size];
|
||
|
var alpha_pos = stream.Position + info.RGBSize;
|
||
|
using (var zstream = new ZLibStream (stream, CompressionMode.Decompress, true))
|
||
|
if (rgb_size != zstream.Read (m_input, 0, rgb_size))
|
||
|
throw new InvalidFormatException ("Unexpected end of file");
|
||
|
if (info.AlphaSize != 0)
|
||
|
{
|
||
|
int alpha_size = w * m_height;
|
||
|
m_alpha = new byte[alpha_size];
|
||
|
stream.Position = alpha_pos;
|
||
|
using (var zstream = new ZLibStream (stream, CompressionMode.Decompress, true))
|
||
|
if (alpha_size != zstream.Read (m_alpha, 0, alpha_size))
|
||
|
throw new InvalidFormatException ("Unexpected end of file");
|
||
|
}
|
||
|
m_output = new byte[info.Width*info.Height*m_bpp];
|
||
|
}
|
||
|
|
||
|
public void Unpack ()
|
||
|
{
|
||
|
int src = 0;
|
||
|
int dst;
|
||
|
int stride = m_bpp * m_width;
|
||
|
for (int channel = 0; channel < 3; ++channel)
|
||
|
{
|
||
|
dst = channel;
|
||
|
for (int y = m_height >> 1; y != 0; --y)
|
||
|
{
|
||
|
for (int x = 0; x < m_width; ++x)
|
||
|
{
|
||
|
m_output[dst] = m_input[src++];
|
||
|
m_output[dst+stride] = m_input[src++];
|
||
|
dst += m_bpp;
|
||
|
}
|
||
|
dst += stride;
|
||
|
src += 2 * (m_width & 1);
|
||
|
}
|
||
|
if (0 != (m_height & 1))
|
||
|
{
|
||
|
for (int x = 0; x < m_width; ++x)
|
||
|
{
|
||
|
m_output[dst] = m_input[src];
|
||
|
src += 2;
|
||
|
dst += m_bpp;
|
||
|
}
|
||
|
src += 2 * (m_width & 1);
|
||
|
}
|
||
|
}
|
||
|
if (3 != m_bpp)
|
||
|
{
|
||
|
src = 0;
|
||
|
dst = 3;
|
||
|
for (int y = 0; y < m_height; ++y)
|
||
|
{
|
||
|
for (int x = 0; x < m_width; ++x)
|
||
|
{
|
||
|
m_output[dst] = m_alpha[src++];
|
||
|
dst += 4;
|
||
|
}
|
||
|
src += m_width & 1;
|
||
|
}
|
||
|
}
|
||
|
dst = m_bpp;
|
||
|
int i;
|
||
|
for (i = stride-m_bpp; i != 0; --i)
|
||
|
{
|
||
|
int b = m_output[dst-m_bpp] - m_output[dst];
|
||
|
m_output[dst++] = (byte)b;
|
||
|
}
|
||
|
for (int j = m_height - 1; j != 0; --j)
|
||
|
{
|
||
|
for (i = 0; i != m_bpp; ++i)
|
||
|
{
|
||
|
m_output[dst] = (byte)(m_output[dst-stride] - m_output[dst]);
|
||
|
++dst;
|
||
|
}
|
||
|
for (i = stride-m_bpp; i != 0; --i)
|
||
|
{
|
||
|
int b = ((int)m_output[dst-stride] + m_output[dst-m_bpp]) >> 1;
|
||
|
b -= m_output[dst];
|
||
|
m_output[dst++] = (byte)b;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
#region IDisposable Members
|
||
|
public void Dispose ()
|
||
|
{
|
||
|
GC.SuppressFinalize (this);
|
||
|
}
|
||
|
#endregion
|
||
|
}
|
||
|
}
|
||
|
}
|