GARbro-mirror/ArcFormats/Masys/ImageAG.cs
2018-02-08 10:54:11 +04:00

260 lines
8.2 KiB
C#

//! \file ImageAG.cs
//! \date Sun May 10 23:53:34 2015
//! \brief Masys Enhanced Game Unit image format.
//
// Copyright (C) 2015-2018 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.Windows.Media;
using System.IO;
namespace GameRes.Formats.Megu
{
[Export(typeof(ImageFormat))]
public class AgFormat : ImageFormat
{
public override string Tag { get { return "ACG"; } }
public override string Description { get { return "Masys image format"; } }
public override uint Signature { get { return 0x00644741u; } } // 'AGd'
public override ImageMetaData ReadMetaData (IBinaryStream file)
{
file.Position = 4;
var info = new ImageMetaData();
info.Width = file.ReadUInt32();
info.Height = file.ReadUInt32();
file.Position = 0x38;
int alpha_size = file.ReadInt32();
info.BPP = 0 == alpha_size ? 24 : 32;
return info;
}
public override ImageData Read (IBinaryStream stream, ImageMetaData info)
{
var reader = new AgReader (stream, info);
reader.Unpack();
return ImageData.Create (info, reader.Format, null, reader.Data);
}
public override void Write (Stream file, ImageData image)
{
throw new NotImplementedException ("AgFormat.Write not implemented");
}
}
internal class AgReader
{
AgBitStream in1;
AgBitStream in2;
AgBitStream in3;
AgBitStream in4;
AgBitStream in5;
byte[] m_alpha;
byte[] m_output;
int m_width;
int m_height;
int m_pixel_size;
byte[] m_first = new byte[3];
public byte[] Data { get { return m_output; } }
public PixelFormat Format { get; private set; }
public AgReader (IBinaryStream input, ImageMetaData info)
{
m_width = (int)info.Width;
m_height = (int)info.Height;
input.Position = 0x0c;
uint offset1 = input.ReadUInt32();
int size1 = input.ReadInt32();
uint offset2 = input.ReadUInt32();
int size2 = input.ReadInt32();
uint offset3 = input.ReadUInt32();
int size3 = input.ReadInt32();
uint offset4 = input.ReadUInt32();
int size4 = input.ReadInt32();
uint offset5 = input.ReadUInt32();
int size5 = input.ReadInt32();
uint offset6 = input.ReadUInt32();
int size6 = input.ReadInt32();
input.Read (m_first, 0, 3);
if (size1 != 0)
in1 = new AgBitStream (input, offset1, size1);
if (size2 != 0)
in2 = new AgBitStream (input, offset2, size2);
if (size3 != 0)
in3 = new AgBitStream (input, offset3, size3);
if (size4 != 0)
in4 = new AgBitStream (input, offset4, size4);
if (size5 != 0)
in5 = new AgBitStream (input, offset5, size5);
if (size6 != 0)
{
input.Position = offset6;
m_alpha = new byte[m_height*m_width];
RleDecode (input, m_alpha);
Format = PixelFormats.Bgra32;
m_pixel_size = 4;
}
else
{
Format = PixelFormats.Bgr24;
m_pixel_size = 3;
}
m_output = new byte[m_width*m_height*m_pixel_size];
}
static internal byte[] ReadSection (IBinaryStream input, long offset, int size)
{
input.Position = offset;
var buf = new byte[size + 4];
if (size != input.Read (buf, 0, size))
throw new InvalidFormatException ("Unexpected end of file");
return buf;
}
public void Unpack ()
{
int dst = 0;
int stride = m_width * m_pixel_size;
for (int y = 0; y < m_height; ++y)
{
for (int x = 0; x < stride; x += m_pixel_size)
{
byte B = ReadColor (0);
byte G = ReadColor (1);
byte R = ReadColor (2);
m_output[dst+x ] = B;
m_output[dst+x+1] = G;
m_output[dst+x+2] = R;
m_first[0] = B;
m_first[1] = G;
m_first[2] = R;
}
m_first[0] = m_output[dst];
m_first[1] = m_output[dst+1];
m_first[2] = m_output[dst+2];
dst += stride;
}
if (m_alpha != null)
ApplyAlpha();
}
private byte ReadColor (int channel)
{
byte c;
if (0 != in1.GetBit())
{
c = in5.GetByte();
}
else if (0 != in3.GetBit())
{
c = m_first[channel];
}
else
{
c = (byte)(in4.GetNibble() + 1);
if (0 != in2.GetBit())
c = (byte)(m_first[channel] - c);
else
c += m_first[channel];
}
return c;
}
private void ApplyAlpha ()
{
int src = 0;
for (int i = 3; i < m_output.Length; i += 4)
{
int alpha = Math.Min (m_alpha[src++]*0xff/0x40, 0xff);
m_output[i] = (byte)alpha;
}
}
private static void RleDecode (IBinaryStream src, byte[] dst_buf)
{
int remaining = dst_buf.Length;
int dst = 0;
while (remaining > 0)
{
byte v = src.ReadUInt8();
int count;
if (0 != (v & 0x80))
{
v &= 0x7F;
count = src.ReadUInt16();
for (int j = 0; j < count; ++j)
{
dst_buf[dst++] = v;
}
}
else
{
dst_buf[dst++] = v;
count = 1;
}
remaining -= count;
}
}
}
internal class AgBitStream
{
byte[] m_input;
int m_src = 0;
int m_bits = 1;
public AgBitStream (IBinaryStream input, long offset, int size)
{
m_input = AgReader.ReadSection (input, offset, size);
}
public int GetBit ()
{
if (1 == m_bits)
{
m_bits = m_input[m_src++] | 0x100;
}
int bit = m_bits & 1;
m_bits >>= 1;
return bit;
}
public int GetNibble ()
{
if (1 == m_bits)
{
m_bits = m_input[m_src++] | 0x100;
}
int bits = m_bits & 0xF;
m_bits >>= 4;
return bits;
}
public byte GetByte ()
{
return m_input[m_src++];
}
}
}