diff --git a/ArcFormats/ArcFormats.csproj b/ArcFormats/ArcFormats.csproj index 579dea49..6eb91a32 100644 --- a/ArcFormats/ArcFormats.csproj +++ b/ArcFormats/ArcFormats.csproj @@ -516,6 +516,7 @@ + diff --git a/ArcFormats/TanukiSoft/ArcTAC.cs b/ArcFormats/TanukiSoft/ArcTAC.cs index 11800931..8b54d1e6 100644 --- a/ArcFormats/TanukiSoft/ArcTAC.cs +++ b/ArcFormats/TanukiSoft/ArcTAC.cs @@ -129,7 +129,7 @@ namespace GameRes.Formats.Tanuki if (res != null) entry.ChangeType (res); } - if ("image" == entry.Type) + if ("image" == entry.Type && !entry.Name.HasExtension (".af")) entry.EncryptedSize = Math.Min (10240, entry.Size); else entry.EncryptedSize = entry.Size; diff --git a/ArcFormats/TanukiSoft/ImageAF.cs b/ArcFormats/TanukiSoft/ImageAF.cs new file mode 100644 index 00000000..0448044a --- /dev/null +++ b/ArcFormats/TanukiSoft/ImageAF.cs @@ -0,0 +1,111 @@ +//! \file ImageAF.cs +//! \date 2018 May 03 +//! \brief TanukiSoft bitmap. +// +// 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 GameRes.Compression; + +namespace GameRes.Formats.Tanuki +{ + [Export(typeof(ImageFormat))] + public class AmapFormat : ImageFormat + { + public override string Tag { get { return "AMAP"; } } + public override string Description { get { return "TanukiSoft bitmap format"; } } + public override uint Signature { get { return 0x50414D41; } } // 'AMAP' + + public AmapFormat () + { + Extensions = new[] { "af" }; + } + + public override ImageMetaData ReadMetaData (IBinaryStream file) + { + var header = file.ReadHeader (0x14); + return new ImageMetaData { + Width = header.ToUInt16 (4), + Height = header.ToUInt16 (6), + BPP = 8, + }; + } + + public override ImageData Read (IBinaryStream file, ImageMetaData info) + { + file.Position = 0x10; + var pixels = LzssUnpack (file); + return ImageData.Create (info, PixelFormats.Gray8, null, pixels); + } + + public override void Write (Stream file, ImageData image) + { + throw new System.NotImplementedException ("AmapFormat.Write not implemented"); + } + + byte[] LzssUnpack (IBinaryStream input) + { + int unpacked_size = input.ReadInt32(); + var output = new byte[unpacked_size]; + var frame = new byte[0x1000]; + int frame_pos = 0xFEE; + int dst = 0; + int ctl = 0; + while (dst < unpacked_size) + { + ctl >>= 1; + if (0 == (ctl & 0x100)) + { + ctl = input.ReadByte(); + if (-1 == ctl) + break; + ctl |= 0xFF00; + } + if (0 != (ctl & 1)) + { + int b = input.ReadByte(); + if (-1 == b) + break; + output[dst++] = frame[frame_pos++ & 0xFFF] = (byte)b; + } + else + { + int lo = input.ReadByte(); + if (-1 == lo) + break; + int hi = input.ReadByte(); + if (-1 == hi) + break; + int offset = (hi & 0xF0) << 4 | lo; + for (int count = 3 + (~hi & 0xF); count != 0; --count) + { + byte v = frame[offset++ & 0xFFF]; + output[dst++] = frame[frame_pos++ & 0xFFF] = v; + } + } + } + return output; + } + } +}