(MCG): implemented 8bpp images.

This commit is contained in:
morkt 2017-01-14 12:24:11 +04:00
parent 6e9064bc50
commit 6604bbe698
2 changed files with 86 additions and 27 deletions

View File

@ -30,17 +30,22 @@ using System.Collections.Generic;
using System.ComponentModel.Composition; using System.ComponentModel.Composition;
using System.IO; using System.IO;
using System.Windows.Media; using System.Windows.Media;
using System.Windows.Media.Imaging;
namespace GameRes.Formats.FC01 namespace GameRes.Formats.FC01
{ {
internal class McaArchive : ArcFile internal class McaArchive : ArcFile
{ {
public readonly byte Key; public readonly byte Key;
public readonly int BPP;
public readonly BitmapPalette Palette;
public McaArchive (ArcView arc, ArchiveFormat impl, ICollection<Entry> dir, byte key) public McaArchive (ArcView arc, ArchiveFormat impl, ICollection<Entry> dir, byte key, int bpp = 24, BitmapPalette palette = null)
: base (arc, impl, dir) : base (arc, impl, dir)
{ {
Key = key; Key = key;
BPP = bpp;
Palette = palette;
} }
} }
@ -60,6 +65,13 @@ namespace GameRes.Formats.FC01
if (index_offset >= file.MaxOffset || !IsSaneCount (count)) if (index_offset >= file.MaxOffset || !IsSaneCount (count))
return null; return null;
int bpp = file.View.ReadInt32 (0x14);
BitmapPalette palette = null;
if (8 == bpp)
{
palette = ReadPalette (file, index_offset);
index_offset += 0x400;
}
string base_name = Path.GetFileNameWithoutExtension (file.Name); string base_name = Path.GetFileNameWithoutExtension (file.Name);
long next_offset = file.View.ReadUInt32 (index_offset); long next_offset = file.View.ReadUInt32 (index_offset);
var dir = new List<Entry> (count); var dir = new List<Entry> (count);
@ -82,7 +94,7 @@ namespace GameRes.Formats.FC01
if (0 == dir.Count) if (0 == dir.Count)
return null; return null;
var options = Query<McgOptions> (arcStrings.MCAEncryptedNotice); var options = Query<McgOptions> (arcStrings.MCAEncryptedNotice);
return new McaArchive (file, this, dir, options.Key); return new McaArchive (file, this, dir, options.Key, bpp, palette);
} }
public override IImageDecoder OpenImage (ArcFile arc, Entry entry) public override IImageDecoder OpenImage (ArcFile arc, Entry entry)
@ -91,7 +103,7 @@ namespace GameRes.Formats.FC01
var input = arc.File.CreateStream (entry.Offset, entry.Size); var input = arc.File.CreateStream (entry.Offset, entry.Size);
try try
{ {
return new McaDecoder (input, mca.Key); return new McaDecoder (input, mca.Key, mca.BPP, mca.Palette);
} }
catch catch
{ {
@ -100,6 +112,19 @@ namespace GameRes.Formats.FC01
} }
} }
BitmapPalette ReadPalette (ArcView file, uint offset)
{
var palette = file.View.ReadBytes (offset, 0x400);
int src = 0;
var colors = new Color[0x100];
for (int i = 0; i < 0x100; ++i)
{
colors[i] = Color.FromRgb (palette[src+2], palette[src+1], palette[src]);
src += 4;
}
return new BitmapPalette (colors);
}
public override ResourceOptions GetDefaultOptions () public override ResourceOptions GetDefaultOptions ()
{ {
return new McgOptions { Key = Settings.Default.MCGLastKey }; return new McgOptions { Key = Settings.Default.MCGLastKey };
@ -122,22 +147,25 @@ namespace GameRes.Formats.FC01
internal sealed class McaDecoder : BinaryImageDecoder internal sealed class McaDecoder : BinaryImageDecoder
{ {
byte m_key; byte m_key;
int m_method; bool m_compressed;
int m_packed_size; int m_packed_size;
int m_unpacked_size; int m_unpacked_size;
BitmapPalette m_palette;
public McaDecoder (IBinaryStream input, byte key) : base (input) public McaDecoder (IBinaryStream input, byte key, int bpp, BitmapPalette palette) : base (input)
{ {
m_key = key; m_key = key;
var header = m_input.ReadHeader (0x20); var header = m_input.ReadHeader (0x20);
m_method = header.ToInt32 (0); int method = header.ToInt32 (0);
if (m_method < 0 || m_method > 1) if (method < 0 || method > 1)
throw new InvalidFormatException(); throw new InvalidFormatException();
m_compressed = method != 0;
uint width = header.ToUInt32 (0xC); uint width = header.ToUInt32 (0xC);
uint height = header.ToUInt32 (0x10); uint height = header.ToUInt32 (0x10);
m_packed_size = header.ToInt32 (0x14); m_packed_size = header.ToInt32 (0x14);
m_unpacked_size = header.ToInt32 (0x18); m_unpacked_size = header.ToInt32 (0x18);
Info = new ImageMetaData { Width = width, Height = height, BPP = 24 }; Info = new ImageMetaData { Width = width, Height = height, BPP = bpp };
m_palette = palette;
} }
protected override ImageData GetImageData () protected override ImageData GetImageData ()
@ -145,7 +173,7 @@ namespace GameRes.Formats.FC01
m_input.Position = 0x20; m_input.Position = 0x20;
var data = m_input.ReadBytes (m_packed_size); var data = m_input.ReadBytes (m_packed_size);
MrgOpener.Decrypt (data, 0, data.Length, m_key); MrgOpener.Decrypt (data, 0, data.Length, m_key);
if (m_method > 0) if (m_compressed)
{ {
using (var input = new BinMemoryStream (data)) using (var input = new BinMemoryStream (data))
using (var lzss = new MrgLzssReader (input, data.Length, m_unpacked_size)) using (var lzss = new MrgLzssReader (input, data.Length, m_unpacked_size))
@ -154,8 +182,9 @@ namespace GameRes.Formats.FC01
data = lzss.Data; data = lzss.Data;
} }
} }
int stride = ((int)Info.Width * 3 + 3) & ~3; int stride = ((int)Info.Width * Info.BPP / 8 + 3) & ~3;
return ImageData.Create (Info, PixelFormats.Bgr24, null, data, stride); var format = 8 == Info.BPP ? PixelFormats.Indexed8 : PixelFormats.Bgr24;
return ImageData.Create (Info, format, m_palette, data, stride);
} }
} }
} }

View File

@ -28,6 +28,7 @@ using System.Collections.Generic;
using System.ComponentModel.Composition; using System.ComponentModel.Composition;
using System.IO; using System.IO;
using System.Windows.Media; using System.Windows.Media;
using System.Windows.Media.Imaging;
using GameRes.Formats.Properties; using GameRes.Formats.Properties;
using GameRes.Formats.Strings; using GameRes.Formats.Strings;
using GameRes.Utility; using GameRes.Utility;
@ -79,7 +80,7 @@ namespace GameRes.Formats.FC01
if (header_size < 0x40) if (header_size < 0x40)
return null; return null;
int bpp = header.ToInt32 (0x24); int bpp = header.ToInt32 (0x24);
if (24 != bpp) if (24 != bpp && 8 != bpp)
throw new NotSupportedException ("Not supported MCG image bitdepth"); throw new NotSupportedException ("Not supported MCG image bitdepth");
return new McgMetaData return new McgMetaData
{ {
@ -111,11 +112,11 @@ namespace GameRes.Formats.FC01
else else
key = LastKey.Value; key = LastKey.Value;
} }
var reader = new McgDecoder (stream.AsStream, meta, key); var reader = new McgDecoder (stream, meta, key);
reader.Unpack(); reader.Unpack();
if (reader.Key != 0) if (reader.Key != 0)
LastKey = reader.Key; LastKey = reader.Key;
return ImageData.Create (info, PixelFormats.Bgr24, null, reader.Data, reader.Stride); return ImageData.Create (info, reader.Format, reader.Palette, reader.Data, reader.Stride);
} }
public override void Write (Stream file, ImageData image) public override void Write (Stream file, ImageData image)
@ -152,33 +153,49 @@ namespace GameRes.Formats.FC01
int m_height; int m_height;
int m_pixels; int m_pixels;
byte m_key; byte m_key;
int m_version; IBinaryStream m_file;
McgMetaData m_info;
public byte[] Data { get { return m_output; } } public byte Key { get { return m_key; } }
public int Stride { get; private set; } public byte[] Data { get { return m_output; } }
public byte Key { get { return m_key; } } public int Stride { get; private set; }
public PixelFormat Format { get; private set; }
public BitmapPalette Palette { get; private set; }
public McgDecoder (Stream input, McgMetaData info, byte key) public McgDecoder (IBinaryStream input, McgMetaData info, byte key)
{ {
input.Position = info.DataOffset; m_file = input;
m_input = new byte[info.PackedSize]; m_info = info;
if (m_input.Length != input.Read (m_input, 0, m_input.Length))
throw new InvalidFormatException ("Unexpected end of file");
m_width = (int)info.Width; m_width = (int)info.Width;
m_height = (int)info.Height; m_height = (int)info.Height;
m_pixels = m_width*m_height; m_pixels = m_width*m_height;
m_key = key; m_key = key;
m_version = info.Version; Stride = m_width * m_info.BPP / 8;
Stride = 3 * m_width; if (101 == m_info.Version)
if (101 == m_version)
Stride = (Stride + 3) & -4; Stride = (Stride + 3) & -4;
if (24 == m_info.BPP)
Format = PixelFormats.Bgr24;
else if (8 == m_info.BPP)
Format = PixelFormats.Indexed8;
else
throw new InvalidFormatException();
} }
static readonly byte[] ChannelOrder = { 1, 0, 2 }; static readonly byte[] ChannelOrder = { 1, 0, 2 };
public void Unpack () public void Unpack ()
{ {
if (200 == m_version) m_file.Position = m_info.DataOffset;
int input_size = m_info.PackedSize;
if (8 == m_info.BPP)
{
ReadPalette();
input_size -= 0x400;
}
m_input = m_file.ReadBytes (input_size);
if (m_input.Length != input_size)
throw new InvalidFormatException ("Unexpected end of file");
if (200 == m_info.Version)
UnpackV200(); UnpackV200();
else else
UnpackV101(); UnpackV101();
@ -297,5 +314,18 @@ namespace GameRes.Formats.FC01
m_output[dst++] = (byte)(r + g); m_output[dst++] = (byte)(r + g);
} }
} }
void ReadPalette ()
{
var palette_data = m_file.ReadBytes (0x400);
int src = 0;
var colors = new Color[0x100];
for (int i = 0; i < 0x100; ++i)
{
colors[i] = Color.FromRgb (palette_data[src+2], palette_data[src+1], palette_data[src]);
src += 4;
}
Palette = new BitmapPalette (colors);
}
} }
} }