mirror of
https://github.com/lifegpc/GARbro.git
synced 2026-06-06 13:39:09 +08:00
implemented PPAC archives, TIM2 and TX images.
This commit is contained in:
@@ -151,6 +151,10 @@
|
||||
<Compile Include="Cyberworks\ArcP8.cs" />
|
||||
<Compile Include="Cyberworks\ImageTB1.cs" />
|
||||
<Compile Include="DaiSystem\ArcPAC.cs" />
|
||||
<Compile Include="DigitalWorks\ArcBIN.cs" />
|
||||
<Compile Include="DigitalWorks\ArcPAC.cs" />
|
||||
<Compile Include="DigitalWorks\ImageTM2.cs" />
|
||||
<Compile Include="DigitalWorks\ImageTX.cs" />
|
||||
<Compile Include="DirectDraw\DxtDecoder.cs" />
|
||||
<Compile Include="DjSystem\ArcDAT.cs" />
|
||||
<Compile Include="elf\ArcAi5DAT.cs" />
|
||||
|
||||
180
ArcFormats/DigitalWorks/ArcBIN.cs
Normal file
180
ArcFormats/DigitalWorks/ArcBIN.cs
Normal file
@@ -0,0 +1,180 @@
|
||||
//! \file ArcBIN.cs
|
||||
//! \date 2018 Sep 19
|
||||
//! \brief Digital Works resource archive.
|
||||
//
|
||||
// Copyright (C) 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.Collections.Generic;
|
||||
using System.ComponentModel.Composition;
|
||||
using System.IO;
|
||||
using System.Linq;
|
||||
using GameRes.Utility;
|
||||
|
||||
namespace GameRes.Formats.DigitalWorks
|
||||
{
|
||||
[Serializable]
|
||||
public class IndexEntry
|
||||
{
|
||||
public uint Offset;
|
||||
public uint Size;
|
||||
public bool IsPacked;
|
||||
public ushort Id;
|
||||
|
||||
public IndexEntry (uint offset, uint size, bool is_packed, ushort id)
|
||||
{
|
||||
Offset = offset;
|
||||
Size = size;
|
||||
IsPacked = is_packed;
|
||||
Id = id;
|
||||
}
|
||||
}
|
||||
|
||||
[Serializable]
|
||||
public class BinScheme
|
||||
{
|
||||
public string Extension;
|
||||
public long Size;
|
||||
public IList<IndexEntry> Index;
|
||||
}
|
||||
|
||||
[Serializable]
|
||||
public class PacScheme : ResourceScheme
|
||||
{
|
||||
public IDictionary<string, IDictionary<string, BinScheme>> KnownSchemes;
|
||||
}
|
||||
|
||||
[Export(typeof(ArchiveFormat))]
|
||||
public class BinOpener : PacOpener
|
||||
{
|
||||
public override string Tag { get { return "BIN/PAC"; } }
|
||||
public override string Description { get { return "Digital Works resource archive"; } }
|
||||
public override uint Signature { get { return 0; } }
|
||||
public override bool IsHierarchic { get { return false; } }
|
||||
public override bool CanWrite { get { return false; } }
|
||||
|
||||
public override ArcFile TryOpen (ArcView file)
|
||||
{
|
||||
if (!file.Name.HasAnyOfExtensions ("bin", "pac"))
|
||||
return null;
|
||||
var scheme = FindScheme (file);
|
||||
if (null == scheme)
|
||||
return null;
|
||||
var pac_name = Path.GetFileNameWithoutExtension (file.Name);
|
||||
var dir = scheme.Index.Select (e => new PackedEntry {
|
||||
Name = string.Format ("{0}{1:D5}.{2}", pac_name, e.Id, scheme.Extension),
|
||||
Offset = e.Offset,
|
||||
Size = e.Size,
|
||||
} as Entry).ToList();
|
||||
dir.ForEach (e => e.Type = FormatCatalog.Instance.GetTypeFromName (e.Name));
|
||||
return new ArcFile (file, this, dir);
|
||||
}
|
||||
|
||||
BinScheme FindScheme (ArcView bin_file)
|
||||
{
|
||||
var bin_name = Path.GetFileName (bin_file.Name).ToUpperInvariant();
|
||||
foreach (var game in KnownSchemes.Values)
|
||||
{
|
||||
BinScheme scheme;
|
||||
if (game.TryGetValue (bin_name, out scheme) && bin_file.MaxOffset == scheme.Size)
|
||||
return scheme;
|
||||
}
|
||||
if (bin_file.MaxOffset >= uint.MaxValue)
|
||||
return null;
|
||||
var bin_dir = VFS.GetDirectoryName (bin_file.Name);
|
||||
var game_dir = Directory.GetParent (bin_dir).FullName;
|
||||
var exe_files = VFS.GetFiles (VFS.CombinePath (game_dir, "*.exe"));
|
||||
if (!exe_files.Any())
|
||||
return null;
|
||||
var last_idx = new byte[12];
|
||||
LittleEndian.Pack ((uint)bin_file.MaxOffset, last_idx, 0);
|
||||
LittleEndian.Pack ((uint)bin_file.MaxOffset, last_idx, 4);
|
||||
foreach (var exe_entry in exe_files)
|
||||
{
|
||||
using (var exe_file = VFS.OpenView (exe_entry))
|
||||
{
|
||||
var exe = new ExeFile (exe_file);
|
||||
if (!exe.ContainsSection (".data"))
|
||||
continue;
|
||||
var data_section = exe.Sections[".data"];
|
||||
var idx_pos = exe.FindString (data_section, last_idx, 4);
|
||||
if (idx_pos > 0)
|
||||
return ParseIndexTable (exe_file, data_section, idx_pos, bin_name);
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
BinScheme ParseIndexTable (ArcView exe_file, ExeFile.Section data, long pos, string bin_name)
|
||||
{
|
||||
uint pac_size = exe_file.View.ReadUInt32 (pos);
|
||||
long last_offset = pac_size;
|
||||
var dir = new List<IndexEntry>();
|
||||
for (pos -= 12; pos >= data.Offset && last_offset != 0; pos -= 12)
|
||||
{
|
||||
long offset = exe_file.View.ReadUInt32 (pos);
|
||||
uint size = exe_file.View.ReadUInt32 (pos+4);
|
||||
ushort is_packed = exe_file.View.ReadUInt16 (pos+8);
|
||||
ushort id = exe_file.View.ReadUInt16 (pos+10);
|
||||
if (0 == size || offset + size > last_offset || is_packed != 0 && is_packed != 1)
|
||||
return null;
|
||||
var entry = new IndexEntry ((uint)offset, size, is_packed != 0, id);
|
||||
dir.Add (entry);
|
||||
last_offset = offset;
|
||||
}
|
||||
bin_name = Path.GetFileNameWithoutExtension (bin_name).ToUpperInvariant();
|
||||
string ext;
|
||||
if (!PacExtensionMap.TryGetValue (bin_name, out ext))
|
||||
ext = "";
|
||||
return new BinScheme {
|
||||
Extension = ext,
|
||||
Size = pac_size,
|
||||
Index = dir,
|
||||
};
|
||||
}
|
||||
|
||||
static readonly Dictionary<string, string> PacExtensionMap = new Dictionary<string, string> {
|
||||
{ "ANM", "BIN" },
|
||||
{ "MOV", "MPG" },
|
||||
{ "STR", "OGG" },
|
||||
{ "TAK", "BIN" },
|
||||
{ "VCE", "OGG" },
|
||||
{ "VIS", "TMX" },
|
||||
{ "_SE", "OGG" },
|
||||
};
|
||||
|
||||
PacScheme DefaultScheme = new PacScheme {
|
||||
KnownSchemes = new Dictionary<string, IDictionary<string, BinScheme>>()
|
||||
};
|
||||
|
||||
public IDictionary<string, IDictionary<string, BinScheme>> KnownSchemes
|
||||
{
|
||||
get { return DefaultScheme.KnownSchemes; }
|
||||
}
|
||||
|
||||
public override ResourceScheme Scheme
|
||||
{
|
||||
get { return DefaultScheme; }
|
||||
set { DefaultScheme = (PacScheme)value; }
|
||||
}
|
||||
}
|
||||
}
|
||||
99
ArcFormats/DigitalWorks/ArcPAC.cs
Normal file
99
ArcFormats/DigitalWorks/ArcPAC.cs
Normal file
@@ -0,0 +1,99 @@
|
||||
//! \file ArcPAC.cs
|
||||
//! \date 2018 Sep 18
|
||||
//! \brief Digital Works resource archive.
|
||||
//
|
||||
// Copyright (C) 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.Collections.Generic;
|
||||
using System.ComponentModel.Composition;
|
||||
using System.IO;
|
||||
using GameRes.Compression;
|
||||
|
||||
namespace GameRes.Formats.DigitalWorks
|
||||
{
|
||||
[Export(typeof(ArchiveFormat))]
|
||||
public class PacOpener : ArchiveFormat
|
||||
{
|
||||
public override string Tag { get { return "PAC/HED"; } }
|
||||
public override string Description { get { return "Digital Works resource archive"; } }
|
||||
public override uint Signature { get { return 0x43415050; } } // 'PPAC-PAC'
|
||||
public override bool IsHierarchic { get { return false; } }
|
||||
public override bool CanWrite { get { return false; } }
|
||||
|
||||
public override ArcFile TryOpen (ArcView file)
|
||||
{
|
||||
if (!file.View.AsciiEqual (4, "-PAC"))
|
||||
return null;
|
||||
var hed_name = Path.ChangeExtension (file.Name, "hed");
|
||||
if (!VFS.FileExists (hed_name))
|
||||
return null;
|
||||
using (var hed = VFS.OpenView (hed_name))
|
||||
{
|
||||
if (!hed.View.AsciiEqual (0, "PPAC-HED"))
|
||||
return null;
|
||||
uint index_offset = 0x10;
|
||||
const uint data_offset = 0x10;
|
||||
int count = (int)(hed.MaxOffset - index_offset) / 0x20;
|
||||
if (!IsSaneCount (count))
|
||||
return null;
|
||||
var dir = new List<Entry> (count);
|
||||
while (index_offset+0x20 <= hed.MaxOffset)
|
||||
{
|
||||
var name = hed.View.ReadString (index_offset, 0x10);
|
||||
var entry = FormatCatalog.Instance.Create<PackedEntry> (name);
|
||||
entry.Offset = hed.View.ReadUInt32 (index_offset+0x10) + data_offset;
|
||||
entry.Size = hed.View.ReadUInt32 (index_offset+0x14);
|
||||
if (!entry.CheckPlacement (file.MaxOffset))
|
||||
return null;
|
||||
dir.Add (entry);
|
||||
index_offset += 0x20;
|
||||
}
|
||||
return new ArcFile (file, this, dir);
|
||||
}
|
||||
}
|
||||
|
||||
public override Stream OpenEntry (ArcFile arc, Entry entry)
|
||||
{
|
||||
var pent = entry as PackedEntry;
|
||||
if (null == pent)
|
||||
return base.OpenEntry (arc, entry);
|
||||
if (!pent.IsPacked)
|
||||
{
|
||||
if (!arc.File.View.AsciiEqual (entry.Offset, "LZS\0"))
|
||||
return base.OpenEntry (arc, entry);
|
||||
pent.IsPacked = true;
|
||||
pent.UnpackedSize = arc.File.View.ReadUInt32 (entry.Offset+4);
|
||||
}
|
||||
var input = arc.File.CreateStream (entry.Offset+8, entry.Size-8);
|
||||
bool embedded_lzs = (input.Signature & ~0xF0u) == 0x535A4C0F; // 'LZS'
|
||||
var lzs = new LzssStream (input);
|
||||
if (embedded_lzs)
|
||||
{
|
||||
var header = new byte[8];
|
||||
lzs.Read (header, 0, 8);
|
||||
pent.UnpackedSize = header.ToUInt32 (4);
|
||||
lzs = new LzssStream (lzs);
|
||||
}
|
||||
return lzs;
|
||||
}
|
||||
}
|
||||
}
|
||||
152
ArcFormats/DigitalWorks/ImageTM2.cs
Normal file
152
ArcFormats/DigitalWorks/ImageTM2.cs
Normal file
@@ -0,0 +1,152 @@
|
||||
//! \file ImageTM2.cs
|
||||
//! \date 2018 Sep 18
|
||||
//! \brief PlayStation2 image format.
|
||||
//
|
||||
// Copyright (C) 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.IO;
|
||||
using System.Windows.Media;
|
||||
using System.Windows.Media.Imaging;
|
||||
|
||||
namespace GameRes.Formats.DigitalWorks
|
||||
{
|
||||
internal class Tim2MetaData : ImageMetaData
|
||||
{
|
||||
public int PaletteSize;
|
||||
public int HeaderSize;
|
||||
public int Colors;
|
||||
}
|
||||
|
||||
[Export(typeof(ImageFormat))]
|
||||
public class Tim2Format : ImageFormat
|
||||
{
|
||||
public override string Tag { get { return "TIM2"; } }
|
||||
public override string Description { get { return "PlayStation/2 image format"; } }
|
||||
public override uint Signature { get { return 0x324D4954; } } // 'TIM2'
|
||||
|
||||
public Tim2Format ()
|
||||
{
|
||||
Extensions = new string[] { "tm2", "ext" };
|
||||
}
|
||||
|
||||
public override ImageMetaData ReadMetaData (IBinaryStream file)
|
||||
{
|
||||
var header = file.ReadHeader (0x40);
|
||||
byte bpp = header[0x23];
|
||||
switch (bpp)
|
||||
{
|
||||
case 1: bpp = 16; break;
|
||||
case 2: bpp = 24; break;
|
||||
case 3: bpp = 32; break;
|
||||
case 5: bpp = 8; break;
|
||||
default: return null;
|
||||
}
|
||||
return new Tim2MetaData {
|
||||
Width = header.ToUInt16 (0x24),
|
||||
Height = header.ToUInt16 (0x26),
|
||||
BPP = bpp,
|
||||
PaletteSize = header.ToInt32 (0x14),
|
||||
HeaderSize = header.ToUInt16 (0x1C),
|
||||
Colors = header.ToUInt16 (0x1E),
|
||||
};
|
||||
}
|
||||
|
||||
public override ImageData Read (IBinaryStream file, ImageMetaData info)
|
||||
{
|
||||
var reader = new Tim2Reader (file, (Tim2MetaData)info);
|
||||
var pixels = reader.Unpack();
|
||||
return ImageData.Create (info, reader.Format, reader.Palette, pixels);
|
||||
}
|
||||
|
||||
public override void Write (Stream file, ImageData image)
|
||||
{
|
||||
throw new System.NotImplementedException ("Tim2Format.Write not implemented");
|
||||
}
|
||||
}
|
||||
|
||||
internal class Tim2Reader
|
||||
{
|
||||
IBinaryStream m_input;
|
||||
Tim2MetaData m_info;
|
||||
|
||||
public PixelFormat Format { get; private set; }
|
||||
public BitmapPalette Palette { get; private set; }
|
||||
|
||||
public Tim2Reader (IBinaryStream input, Tim2MetaData info)
|
||||
{
|
||||
m_input = input;
|
||||
m_info = info;
|
||||
switch (info.BPP)
|
||||
{
|
||||
case 8: Format = PixelFormats.Indexed8; break;
|
||||
case 16: Format = PixelFormats.Bgr555; break;
|
||||
case 24: Format = PixelFormats.Bgr24; break;
|
||||
case 32: Format = PixelFormats.Bgra32; break;
|
||||
}
|
||||
}
|
||||
|
||||
public byte[] Unpack ()
|
||||
{
|
||||
m_input.Position = 0x10 + m_info.HeaderSize;
|
||||
int pixel_size = m_info.BPP / 8;
|
||||
int image_size = (int)m_info.Width * (int)m_info.Height * pixel_size;
|
||||
var output = m_input.ReadBytes (image_size);
|
||||
if (pixel_size <= 8 && m_info.Colors > 0)
|
||||
Palette = ReadPalette (m_info.Colors);
|
||||
|
||||
if (pixel_size >= 3)
|
||||
{
|
||||
for (int i = 0; i < image_size; i += pixel_size)
|
||||
{
|
||||
byte r = output[i];
|
||||
output[i] = output[i+2];
|
||||
output[i+2] = r;
|
||||
}
|
||||
}
|
||||
return output;
|
||||
}
|
||||
|
||||
BitmapPalette ReadPalette (int color_num)
|
||||
{
|
||||
var source = ImageFormat.ReadColorMap (m_input.AsStream, color_num, PaletteFormat.RgbA);
|
||||
var color_map = new Color[color_num];
|
||||
|
||||
int parts = color_num / 32;
|
||||
const int blocks = 2;
|
||||
const int rows = 2;
|
||||
const int colors = 8;
|
||||
|
||||
int dst = 0;
|
||||
for (int part = 0; part < parts; part++)
|
||||
for (int block = 0; block < blocks; block++)
|
||||
for (int row = 0; row < rows; row++)
|
||||
{
|
||||
int src = (part * rows * blocks + row * rows + block) * colors;
|
||||
Array.Copy (source, src, color_map, dst, colors);
|
||||
dst += colors;
|
||||
}
|
||||
return new BitmapPalette (color_map);
|
||||
}
|
||||
}
|
||||
}
|
||||
102
ArcFormats/DigitalWorks/ImageTX.cs
Normal file
102
ArcFormats/DigitalWorks/ImageTX.cs
Normal file
@@ -0,0 +1,102 @@
|
||||
//! \file ImageTX.cs
|
||||
//! \date 2018 Sep 19
|
||||
//! \brief Digital Works texture format.
|
||||
//
|
||||
// Copyright (C) 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.ComponentModel.Composition;
|
||||
using System.IO;
|
||||
using System.Windows.Media;
|
||||
using System.Windows.Media.Imaging;
|
||||
|
||||
namespace GameRes.Formats.DigitalWorks
|
||||
{
|
||||
[Export(typeof(ImageFormat))]
|
||||
public class TxFormat : ImageFormat
|
||||
{
|
||||
public override string Tag { get { return "TX"; } }
|
||||
public override string Description { get { return "Digital Works texture format"; } }
|
||||
public override uint Signature { get { return 0x00035854; } } // 'TX'
|
||||
|
||||
const int BlockSize = 256;
|
||||
|
||||
public TxFormat ()
|
||||
{
|
||||
Extensions = new string[] { "tmx", "tx" };
|
||||
Signatures = new uint[] { 0x00035854, 0x00025854, 0 };
|
||||
}
|
||||
|
||||
public override ImageMetaData ReadMetaData (IBinaryStream file)
|
||||
{
|
||||
var header = file.ReadHeader (0x10);
|
||||
if (!header.AsciiEqual ("TX"))
|
||||
return null;
|
||||
byte bpp = header[6];
|
||||
if (bpp < 1 || bpp > 4)
|
||||
return null;
|
||||
uint w_blocks = header.ToUInt16 (2);
|
||||
uint h_blocks = header.ToUInt16 (4);
|
||||
return new ImageMetaData {
|
||||
Width = w_blocks * BlockSize,
|
||||
Height = h_blocks * BlockSize,
|
||||
BPP = bpp * 8,
|
||||
};
|
||||
}
|
||||
|
||||
public override ImageData Read (IBinaryStream file, ImageMetaData info)
|
||||
{
|
||||
file.Position = 0x10;
|
||||
int block_stride = BlockSize * (info.BPP / 8);
|
||||
int stride = (int)info.Width * (info.BPP / 8);
|
||||
var pixels = new byte[stride * (int)info.Height];
|
||||
int x_blocks = (int)info.Width / BlockSize;
|
||||
int y_blocks = (int)info.Height / BlockSize;
|
||||
int dst_row = 0;
|
||||
for (int y = 0; y < y_blocks; ++y)
|
||||
{
|
||||
int dst_col = dst_row;
|
||||
for (int x = 0; x < x_blocks; ++x)
|
||||
{
|
||||
int dst = dst_col;
|
||||
for (int i = 0; i < BlockSize; ++i)
|
||||
{
|
||||
file.Read (pixels, dst, block_stride);
|
||||
dst += stride;
|
||||
}
|
||||
dst_col += block_stride;
|
||||
}
|
||||
dst_row += stride * BlockSize;
|
||||
}
|
||||
BitmapPalette palette = null;
|
||||
if (8 == info.BPP)
|
||||
palette = ReadPalette (file.AsStream, 0x100, PaletteFormat.BgrA);
|
||||
PixelFormat format = 8 == info.BPP ? PixelFormats.Indexed8
|
||||
: 24 == info.BPP ? PixelFormats.Bgr24 : PixelFormats.Bgra32;
|
||||
return ImageData.Create (info, format, palette, pixels, stride);
|
||||
}
|
||||
|
||||
public override void Write (Stream file, ImageData image)
|
||||
{
|
||||
throw new System.NotImplementedException ("TxFormat.Write not implemented");
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user