2020-01-28 21:00:03 -05:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
|
|
|
|
using System.IO;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
|
|
|
|
using Toolbox;
|
|
|
|
|
using System.Windows.Forms;
|
|
|
|
|
using Toolbox.Library;
|
|
|
|
|
using Toolbox.Library.IO;
|
|
|
|
|
using Toolbox.Library.Forms;
|
|
|
|
|
|
|
|
|
|
namespace FirstPlugin
|
|
|
|
|
{
|
|
|
|
|
public class WTA : IFileFormat, IArchiveFile, ILeaveOpenOnLoad
|
|
|
|
|
{
|
|
|
|
|
public FileType FileType { get; set; } = FileType.Archive;
|
|
|
|
|
|
|
|
|
|
public bool CanSave { get; set; }
|
|
|
|
|
public string[] Description { get; set; } = new string[] { "WT Archive" };
|
|
|
|
|
public string[] Extension { get; set; } = new string[] { "*.wta" };
|
|
|
|
|
public string FileName { get; set; }
|
|
|
|
|
public string FilePath { get; set; }
|
|
|
|
|
public IFileInfo IFileInfo { get; set; }
|
|
|
|
|
|
|
|
|
|
public bool Identify(System.IO.Stream stream)
|
|
|
|
|
{
|
|
|
|
|
using (var reader = new Toolbox.Library.IO.FileReader(stream, true))
|
|
|
|
|
{
|
|
|
|
|
return reader.CheckSignature(3, "WTA");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public Type[] Types
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
|
|
|
|
List<Type> types = new List<Type>();
|
|
|
|
|
return types.ToArray();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public List<FileInfo> files = new List<FileInfo>();
|
|
|
|
|
|
|
|
|
|
public IEnumerable<ArchiveFileInfo> Files => files;
|
|
|
|
|
|
|
|
|
|
public void ClearFiles() { files.Clear(); }
|
|
|
|
|
|
|
|
|
|
public bool CanAddFiles { get; set; } = true;
|
|
|
|
|
public bool CanRenameFiles { get; set; } = true;
|
|
|
|
|
public bool CanReplaceFiles { get; set; } = true;
|
|
|
|
|
public bool CanDeleteFiles { get; set; } = true;
|
|
|
|
|
|
|
|
|
|
public Header ApakHeader;
|
|
|
|
|
|
|
|
|
|
public void Load(System.IO.Stream stream)
|
|
|
|
|
{
|
|
|
|
|
CanSave = true;
|
|
|
|
|
using (var reader = new FileReader(stream, true))
|
|
|
|
|
{
|
|
|
|
|
ApakHeader = new WTA.Header(reader, files);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Save(System.IO.Stream stream)
|
|
|
|
|
{
|
|
|
|
|
using (var writer = new FileWriter(stream))
|
|
|
|
|
{
|
|
|
|
|
ApakHeader.Write(writer, files);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class Header
|
|
|
|
|
{
|
|
|
|
|
public ushort Version { get; set; }
|
|
|
|
|
public bool IsBigEndian { get; set; }
|
|
|
|
|
|
|
|
|
|
public uint Unknown1 { get; set; }
|
|
|
|
|
public uint Unknown2 { get; set; }
|
|
|
|
|
|
|
|
|
|
public Header(FileReader reader, List<FileInfo> files)
|
|
|
|
|
{
|
|
|
|
|
reader.SetByteOrder(false);
|
|
|
|
|
|
|
|
|
|
uint magic = reader.ReadUInt32();
|
|
|
|
|
uint Version = reader.ReadUInt32();
|
|
|
|
|
if (Version != 1)
|
|
|
|
|
reader.SetByteOrder(true);
|
|
|
|
|
reader.ReadUInt32(); //0
|
|
|
|
|
uint alignment = reader.ReadUInt32();
|
|
|
|
|
reader.ReadUInt32(); //slightly larger than file size
|
|
|
|
|
reader.ReadUInt32(); //0
|
|
|
|
|
uint dataOffset = reader.ReadUInt32();
|
|
|
|
|
uint stringTableOffset = reader.ReadUInt32();
|
|
|
|
|
uint stringTableSize = reader.ReadUInt32();
|
|
|
|
|
reader.ReadUInt32(); //64
|
2020-02-12 16:21:42 -05:00
|
|
|
|
uint unkSectionCount = reader.ReadUInt32();
|
2020-01-28 21:00:03 -05:00
|
|
|
|
uint FileCount = reader.ReadUInt32();
|
|
|
|
|
reader.ReadUInt32(); //1
|
2020-02-12 16:21:42 -05:00
|
|
|
|
reader.ReadUInt32(); //0
|
|
|
|
|
reader.ReadUInt32(); //0
|
|
|
|
|
reader.ReadUInt32(); //0
|
2020-01-28 21:00:03 -05:00
|
|
|
|
|
2020-02-12 16:56:16 -05:00
|
|
|
|
//Skip an unknown section that is 32 bytes in size
|
2020-02-12 16:21:42 -05:00
|
|
|
|
reader.Seek(unkSectionCount * 32);
|
2020-01-28 21:00:03 -05:00
|
|
|
|
|
|
|
|
|
for (int i = 0; i < FileCount; i++)
|
|
|
|
|
files.Add(new FileInfo(reader, dataOffset, stringTableOffset));
|
|
|
|
|
|
|
|
|
|
//Now read data and align offsets
|
|
|
|
|
reader.SeekBegin(dataOffset);
|
|
|
|
|
for (int i = 0; i < FileCount; i++)
|
|
|
|
|
{
|
|
|
|
|
files[i].FileName = $"File {i}";
|
|
|
|
|
files[i].DataOffset = reader.Position;
|
2020-02-12 18:24:21 -05:00
|
|
|
|
if (files[i].CompressedSize != 0)
|
|
|
|
|
reader.Seek((int)files[i].CompressedSize);
|
|
|
|
|
else
|
|
|
|
|
reader.Seek((int)files[i].UncompressedSize);
|
|
|
|
|
|
|
|
|
|
// Console.WriteLine($"{i} {files[i].DataOffset} {files[i].CompressedSize} {files[i].Alignment}");
|
2020-02-06 18:20:42 -05:00
|
|
|
|
|
|
|
|
|
if (files[i].Alignment != 0)
|
|
|
|
|
reader.Align((int)files[i].Alignment);
|
2020-01-28 21:00:03 -05:00
|
|
|
|
}
|
2020-01-29 16:08:29 -05:00
|
|
|
|
|
|
|
|
|
//Try to get file names from file formats inside
|
|
|
|
|
//The string table for this file uses a bunch of ids and not very ideal for viewing
|
|
|
|
|
for (int i = 0; i < FileCount; i++)
|
|
|
|
|
{
|
|
|
|
|
if (files[i].CompressedSize == 0)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
reader.SeekBegin(files[i].DataOffset);
|
|
|
|
|
var data = reader.ReadBytes((int)files[i].CompressedSize);
|
2020-02-12 18:24:21 -05:00
|
|
|
|
if (files[i].CompressedSize != 0 && files[i].CompressedSize != files[i].UncompressedSize && data[0] == 0x78 && data[1] == 0x5E)
|
2020-01-29 16:08:29 -05:00
|
|
|
|
data = STLibraryCompression.ZLIB.Decompress(data);
|
|
|
|
|
|
|
|
|
|
using (var dataReader = new FileReader(data))
|
|
|
|
|
{
|
2020-02-12 16:21:42 -05:00
|
|
|
|
if (dataReader.CheckSignature(4, "FRES") || dataReader.CheckSignature(4, "BNTX"))
|
2020-01-29 16:08:29 -05:00
|
|
|
|
{
|
|
|
|
|
dataReader.SetByteOrder(false);
|
2020-02-12 16:21:42 -05:00
|
|
|
|
dataReader.SeekBegin(16);
|
2020-01-29 16:08:29 -05:00
|
|
|
|
uint fileNameOffset = dataReader.ReadUInt32();
|
2020-02-12 16:21:42 -05:00
|
|
|
|
dataReader.SeekBegin(fileNameOffset );
|
2020-01-29 16:08:29 -05:00
|
|
|
|
files[i].FileName = dataReader.ReadZeroTerminatedString();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-01-28 21:00:03 -05:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Write(FileWriter writer, List<FileInfo> files)
|
|
|
|
|
{
|
|
|
|
|
writer.SetByteOrder(IsBigEndian);
|
|
|
|
|
writer.WriteSignature("WTA ");
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class FileInfo : ArchiveFileInfo
|
|
|
|
|
{
|
|
|
|
|
public uint Alignment;
|
|
|
|
|
public uint CompressedSize;
|
|
|
|
|
public uint UncompressedSize;
|
|
|
|
|
|
|
|
|
|
public long DataOffset;
|
|
|
|
|
|
|
|
|
|
public override Stream FileDataStream
|
|
|
|
|
{
|
|
|
|
|
get { return DecompressData(); }
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Stream DecompressData()
|
|
|
|
|
{
|
|
|
|
|
using (var reader = new FileReader(baseStream, true))
|
|
|
|
|
{
|
|
|
|
|
reader.SeekBegin(DataOffset);
|
|
|
|
|
var data = reader.ReadBytes((int)CompressedSize);
|
2020-02-12 16:21:42 -05:00
|
|
|
|
if (CompressedSize != UncompressedSize && data[0] == 0x78 && data[1] == 0x5E)
|
2020-01-28 21:00:03 -05:00
|
|
|
|
data = STLibraryCompression.ZLIB.Decompress(data);
|
|
|
|
|
|
|
|
|
|
return new MemoryStream(data);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private Stream baseStream;
|
|
|
|
|
|
|
|
|
|
public FileInfo(FileReader reader, uint dataOffset, uint stringTableOffset)
|
|
|
|
|
{
|
|
|
|
|
long pos = reader.Position;
|
|
|
|
|
|
|
|
|
|
baseStream = reader.BaseStream;
|
|
|
|
|
uint nameLength = reader.ReadUInt32();
|
|
|
|
|
uint hash = reader.ReadUInt32();
|
|
|
|
|
Alignment = reader.ReadUInt32();
|
2020-01-29 16:08:29 -05:00
|
|
|
|
UncompressedSize = reader.ReadUInt32();
|
2020-01-28 21:00:03 -05:00
|
|
|
|
CompressedSize = reader.ReadUInt32();
|
|
|
|
|
uint nameOffset = reader.ReadUInt32();
|
|
|
|
|
uint unk = reader.ReadUInt32();
|
|
|
|
|
uint unk2 = reader.ReadUInt32();
|
|
|
|
|
|
2020-01-29 16:08:29 -05:00
|
|
|
|
// FileName = reader.ReadString(0x20, true);
|
|
|
|
|
// reader.ReadUInt32();
|
2020-01-28 21:00:03 -05:00
|
|
|
|
|
|
|
|
|
long endpos = reader.Position;
|
|
|
|
|
reader.Seek(endpos, System.IO.SeekOrigin.Begin);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public void Unload()
|
|
|
|
|
{
|
|
|
|
|
|
|
|
|
|
}
|
|
|
|
|
public byte[] Save()
|
|
|
|
|
{
|
|
|
|
|
return null;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
public bool AddFile(ArchiveFileInfo archiveFileInfo)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public bool DeleteFile(ArchiveFileInfo archiveFileInfo)
|
|
|
|
|
{
|
|
|
|
|
return false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|