mirror of
https://github.com/crskycode/GARbro.git
synced 2024-11-27 15:44:00 +08:00
(MCG): implemented 8bpp images.
This commit is contained in:
parent
6e9064bc50
commit
6604bbe698
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -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);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user