diff --git a/WzComparerR2.WzLib/WzComparerR2.WzLib.csproj b/WzComparerR2.WzLib/WzComparerR2.WzLib.csproj new file mode 100644 index 0000000..a723bc2 --- /dev/null +++ b/WzComparerR2.WzLib/WzComparerR2.WzLib.csproj @@ -0,0 +1,70 @@ + + + + Debug + AnyCPU + 8.0.30703 + 2.0 + {0E9801FD-44A2-4AF8-AE91-D6E74BAD56B2} + Library + Properties + WzComparerR2.WzLib + WzComparerR2.WzLib + v4.0 + 512 + + + + true + full + false + bin\Debug\ + TRACE;DEBUG + prompt + 4 + true + + + pdbonly + true + bin\Release\ + TRACE + prompt + 4 + true + + + + + + + + + Properties\CommonAssemblyInfo.cs + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/WzComparerR2.WzLib/Wz_Crypto.cs b/WzComparerR2.WzLib/Wz_Crypto.cs new file mode 100644 index 0000000..f4f0bbc --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Crypto.cs @@ -0,0 +1,329 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.IO; +using System.Security.Cryptography; +using System.Collections.Specialized; +using System.Text.RegularExpressions; + +namespace WzComparerR2.WzLib +{ + public class Wz_Crypto + { + public Wz_Crypto() + { + this.keys_bms = new Wz_CryptoKey(iv_bms); + this.keys_kms = new Wz_CryptoKey(iv_kms); + this.keys_gms = new Wz_CryptoKey(iv_gms); + this.listwz = false; + this.EncType = Wz_CryptoKeyType.Unknown; + this.List = new StringCollection(); + } + + public void Reset() + { + this.encryption_detected = false; + this.listwz = false; + this.EncType = Wz_CryptoKeyType.Unknown; + this.List.Clear(); + } + + public bool list_contains(string name) + { + bool contains = this.List.Contains(name); + if (contains) + this.List.Remove(name); + return contains; + // foreach (string list_entry in this.list) + // { + // // if (list_entry.Contains(Name)) + // if (list_entry == Name) + // { + // this.list.Remove(list_entry); + // return true; + // } + // } + // return false; + } + + public void LoadListWz(string path) + { + path = Path.Combine(path, "List.wz"); + if (File.Exists(path)) + { + this.listwz = true; + using (FileStream list_file = new FileStream(path, FileMode.Open, FileAccess.Read, FileShare.Read)) + { + BinaryReader listwz = new BinaryReader(list_file); + int length = (int)list_file.Length; + int len = 0; + byte b = 0; + string folder = ""; + list_file.Position += 4; + byte check_for_d = listwz.ReadByte(); + + if ((char)(check_for_d ^ this.keys_gms[0]) == 'd') + { + this.EncType = Wz_CryptoKeyType.GMS; + } + else if ((char)(check_for_d ^ this.keys_kms[0]) == 'd') + { + this.EncType = Wz_CryptoKeyType.KMS; + } + + list_file.Position = 0; + while (list_file.Position < length) + { + len = listwz.ReadInt32() * 2; + for (int i = 0; i < len; i += 2) + { + b = (byte)(listwz.ReadByte() ^ this.keys[i]); + folder += (char)(b); + list_file.Position++; + } + list_file.Position += 2; + folder.Replace(".im/", ".img"); + this.List.Add(folder); + folder = ""; + } + this.List.Remove("dummy"); + } + } + } + + public void DetectEncryption(Wz_File f) + { + int old_off = (int)f.FileStream.Position; + f.FileStream.Position = f.Header.DataStartPosition; + if (f.ReadInt32() <= 0) //只有文件头 无法预判 + { + return; + } + f.FileStream.Position++; + int len = (int)(-f.BReader.ReadSByte()); + byte[] bytes = f.BReader.ReadBytes(len); + + for (int i = 0; i < len; i++) + { + bytes[i] ^= (byte)(0xAA + i); + } + + StringBuilder sb = new StringBuilder(); + if (!this.encryption_detected) + { + //测试bms + sb.Clear(); + for (int i = 0; i < len; i++) + { + sb.Append((char)(keys_bms[i] ^ bytes[i])); + } + if (IsLegalNodeName(sb.ToString())) + { + this.EncType = Wz_CryptoKeyType.BMS; + this.encryption_detected = true; + goto lbl_end; + } + + //测试kms + sb.Clear(); + for (int i = 0; i < len; i++) + { + sb.Append((char)(keys_kms[i] ^ bytes[i])); + } + if (IsLegalNodeName(sb.ToString())) + { + this.EncType = Wz_CryptoKeyType.KMS; + this.encryption_detected = true; + goto lbl_end; + } + + //测试gms + sb.Clear(); + for (int i = 0; i < len; i++) + { + sb.Append((char)(keys_gms[i] ^ bytes[i])); + } + if (IsLegalNodeName(sb.ToString())) + { + this.EncType = Wz_CryptoKeyType.GMS; + this.encryption_detected = true; + goto lbl_end; + } + } + + lbl_end: + f.FileStream.Position = old_off; + } + + public bool IsLegalNodeName(string nodeName) + { + return nodeName.EndsWith(".img") || nodeName.EndsWith(".lua") || Regex.IsMatch(nodeName, @"^[A-Za-z-9_]+$"); + } + + static readonly byte[] iv_gms = { 0x4d, 0x23, 0xc7, 0x2b }; + static readonly byte[] iv_kms = { 0xb9, 0x7d, 0x63, 0xe9 }; + static readonly byte[] iv_bms = { 0x00, 0x00, 0x00, 0x00 }; + + private Wz_CryptoKey keys_bms, keys_gms, keys_kms; + private Wz_CryptoKeyType enc_type; + + public bool encryption_detected = false; + public bool listwz = false; + + public Wz_CryptoKey keys { get; private set; } + public StringCollection List { get; private set; } + + public Wz_CryptoKeyType EncType + { + get { return enc_type; } + set + { + enc_type = value; + switch (enc_type) + { + case Wz_CryptoKeyType.Unknown: + this.keys = null; + break; + + case Wz_CryptoKeyType.BMS: + this.keys = keys_bms; + break; + + case Wz_CryptoKeyType.KMS: + this.keys = keys_kms; + break; + + case Wz_CryptoKeyType.GMS: + this.keys = keys_gms; + break; + } + } + } + + public enum Wz_CryptoKeyType + { + Unknown = 0, + BMS = 1, + KMS = 2, + GMS = 3 + } + + public class Wz_CryptoKey + { + public Wz_CryptoKey(byte[] iv) + { + this.iv = iv; + if (iv == null || BitConverter.ToInt32(iv, 0) == 0) + { + this.isEmptyIV = true; + } + } + + private byte[] keys; + private byte[] iv; + private bool isEmptyIV; + + public byte this[int index] + { + get + { + if (isEmptyIV) + { + return 0; + } + if (keys == null || keys.Length <= index) + { + EnsureKeySize(index + 1); + } + return this.keys[index]; + } + } + + public void EnsureKeySize(int size) + { + if (isEmptyIV) + { + return; + } + if (this.keys != null && this.keys.Length >= size) + { + return; + } + + size = (int)Math.Ceiling(1.0 * size / 4096) * 4096; + int startIndex = 0; + + if (this.keys == null) + { + keys = new byte[size]; + } + else + { + startIndex = this.keys.Length; + Array.Resize(ref this.keys, size); + } + + Rijndael aes = Rijndael.Create(); + aes.KeySize = 256; + aes.BlockSize = 128; + aes.Key = aesKey; + aes.Mode = CipherMode.ECB; + MemoryStream ms = new MemoryStream(keys, startIndex, keys.Length - startIndex, true); + CryptoStream s = new CryptoStream(ms, aes.CreateEncryptor(), CryptoStreamMode.Write); + + for (int i = startIndex; i < size; i += 16) + { + if (i == 0) + { + byte[] block = new byte[16]; + for (int j = 0; j < block.Length; j++) + { + block[j] = iv[j % 4]; + } + s.Write(block, 0, block.Length); + } + else + { + s.Write(keys, i - 16, 16); + } + } + + s.Flush(); + ms.Close(); + } + + public unsafe void Decrypt(byte[] buffer, int startIndex, int length) + { + if (isEmptyIV) + return; + + this.EnsureKeySize(length); + + fixed (byte* pBuffer = buffer, pKeys = this.keys) + { + int i = 0; + byte* pData = pBuffer + startIndex; + + for (int i1 = length / 4 * 4; i < i1; i += 4, pData += 4) + { + *((int*)pData) ^= *(int*)(pKeys + i); + } + + for (; i < length; i++, pData++) + { + *pData ^= *(pKeys + i); + } + } + } + + static readonly byte[] aesKey = {0x13, 0x00, 0x00, 0x00, + 0x08, 0x00, 0x00, 0x00, + 0x06, 0x00, 0x00, 0x00, + 0xB4, 0x00, 0x00, 0x00, + 0x1B, 0x00, 0x00, 0x00, + 0x0F, 0x00, 0x00, 0x00, + 0x33, 0x00, 0x00, 0x00, + 0x52, 0x00, 0x00, 0x00 }; + } + } +} diff --git a/WzComparerR2.WzLib/Wz_Directory.cs b/WzComparerR2.WzLib/Wz_Directory.cs new file mode 100644 index 0000000..1354ace --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Directory.cs @@ -0,0 +1,28 @@ +using System; +using System.Collections.Generic; +using System.Linq; +using System.Text; + +namespace WzComparerR2.WzLib +{ + public class Wz_Directory + { + public Wz_Directory(string name, int size, int cs32, uint hashOff, uint hashPos, Wz_File wz_f) + { + this.Name = name; + this.WzFile = wz_f; + this.Size = size; + this.Checksum = cs32; + this.HashedOffset = hashOff; + this.HashedOffsetPosition = hashPos; + } + + public string Name { get; set; } + public Wz_File WzFile { get; set; } + public int Size { get; set; } + public int Checksum { get; set; } + public uint HashedOffset { get; set; } + public uint HashedOffsetPosition { get; set; } + public long Offset { get; set; } + } +} diff --git a/WzComparerR2.WzLib/Wz_File.cs b/WzComparerR2.WzLib/Wz_File.cs new file mode 100644 index 0000000..42b313f --- /dev/null +++ b/WzComparerR2.WzLib/Wz_File.cs @@ -0,0 +1,762 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.IO; +using System.Linq; +using System.Text.RegularExpressions; + +namespace WzComparerR2.WzLib +{ + public class Wz_File : IDisposable + { + public Wz_File(string fileName, Wz_Structure wz) + { + this.imageCount = 0; + this.wzStructure = wz; + this.fileStream = new FileStream(fileName, FileMode.Open, FileAccess.Read, FileShare.ReadWrite); + this.bReader = new BinaryReader(this.FileStream); + this.loaded = this.GetHeader(fileName); + this.stringTable = new Dictionary(); + this.directories = new List(); + } + + private FileStream fileStream; + private BinaryReader bReader; + private Wz_Structure wzStructure; + private Wz_Header header; + private Wz_Node node; + private int imageCount; + private bool loaded; + private bool isSubDir; + private Wz_Type type; + private List mergedWzFiles; + private Wz_File ownerWzFile; + private readonly List directories; + + public Encoding TextEncoding { get; set; } + + public readonly object ReadLock = new object(); + + internal Dictionary stringTable; + internal byte[] tempBuffer; + + public FileStream FileStream + { + get { return fileStream; } + } + + public BinaryReader BReader + { + get { return bReader; } + } + + public Wz_Structure WzStructure + { + get { return wzStructure; } + set { wzStructure = value; } + } + + public Wz_Header Header + { + get { return header; } + private set { header = value; } + } + + public Wz_Node Node + { + get { return node; } + set { node = value; } + } + + public int ImageCount + { + get { return imageCount; } + } + + public bool Loaded + { + get { return loaded; } + } + + public bool IsSubDir + { + get { return this.isSubDir; } + } + + public Wz_Type Type + { + get { return type; } + set { type = value; } + } + + public IEnumerable MergedWzFiles + { + get { return this.mergedWzFiles ?? Enumerable.Empty(); } + } + + public Wz_File OwnerWzFile + { + get { return this.ownerWzFile; } + } + + public void Close() + { + if (this.bReader != null) + this.bReader.Close(); + if (this.fileStream != null) + this.fileStream.Close(); + } + + void IDisposable.Dispose() + { + this.Close(); + } + + private bool GetHeader(string fileName) + { + this.fileStream.Position = 0; + long filesize = this.FileStream.Length; + if (filesize < 4) { goto __failed; } + + string signature = new string(this.BReader.ReadChars(4)); + if (signature != "PKG1") { goto __failed; } + + long dataSize = this.BReader.ReadInt64(); + int headerSize = this.BReader.ReadInt32(); + string copyright = new string(this.BReader.ReadChars(headerSize - (int)this.FileStream.Position)); + + // encver detecting: + // Since KMST1132, wz removed the 2 bytes encver, and use a fixed wzver '777'. + // Here we try to read the first 2 bytes from data part and guess if it looks like an encver. + bool encverMissing = false; + int encver = -1; + if (dataSize >= 2) + { + this.fileStream.Position = headerSize; + encver = this.BReader.ReadUInt16(); + // encver always less than 256 + if (encver > 0xff) + { + encverMissing = true; + } + else if (encver == 0x80) + { + // there's an exceptional case that the first field of data part is a compressed int which determined property count, + // if the value greater than 127 and also to be a multiple of 256, the first 5 bytes will become to + // 80 00 xx xx xx + // so we additional check the int value, at most time the child node count in a wz won't greater than 65536. + if (dataSize >= 5) + { + this.fileStream.Position = headerSize; + int propCount = this.ReadInt32(); + if (propCount > 0 && (propCount & 0xff) == 0 && propCount <= 0xffff) + { + encverMissing = true; + } + } + } + } + else + { + // Obviously, if data part have only 1 byte, encver must be deleted. + encverMissing = true; + } + + int dataStartPos = headerSize + (encverMissing ? 0 : 2); + this.Header = new Wz_Header(signature, copyright, fileName, headerSize, dataSize, filesize, dataStartPos); + + if (encverMissing) + { + // not sure if nexon will change this magic version, just hard coded. + this.Header.SetWzVersion(777); + this.Header.VersionChecked = true; + } + else + { + this.Header.SetOrdinalVersionDetector(encver); + } + + return true; + + __failed: + this.header = new Wz_Header(null, null, fileName, 0, 0, filesize, 0); + return false; + } + + public int ReadInt32() + { + int s = this.BReader.ReadSByte(); + return (s == -128) ? this.BReader.ReadInt32() : s; + } + + public long ReadInt64() + { + int s = this.BReader.ReadSByte(); + return (s == -128) ? this.BReader.ReadInt64() : s; + } + + public float ReadSingle() + { + float fl = this.BReader.ReadSByte(); + return (fl == -128) ? this.BReader.ReadSingle() : fl; + } + + public string ReadString(long offset) + { + byte b = this.BReader.ReadByte(); + switch (b) + { + case 0x00: + case 0x73: + return ReadString(); + + case 0x01: + case 0x1B: + return ReadStringAt(offset + this.BReader.ReadInt32()); + + case 0x04: + this.FileStream.Position += 8; + break; + + default: + throw new Exception("문자열을 읽는 중 오류: " + this.FileStream.Name + " " + this.FileStream.Position); + } + return string.Empty; + } + + public string ReadStringAt(long offset) + { + long oldoffset = this.FileStream.Position; + string str; + if (!stringTable.TryGetValue(offset, out str)) + { + this.FileStream.Position = offset; + str = ReadString(); + stringTable[offset] = str; + this.FileStream.Position = oldoffset; + } + return str; + } + + public unsafe string ReadString() + { + int size = this.BReader.ReadSByte(); + string result = null; + if (size < 0) + { + byte mask = 0xAA; + size = (size == -128) ? this.BReader.ReadInt32() : -size; + + var buffer = GetStringBuffer(size); + this.fileStream.Read(buffer, 0, size); + this.WzStructure.encryption.keys.Decrypt(buffer, 0, size); + + fixed (byte* pData = buffer) + { + for (int i = 0; i < size; i++) + { + pData[i] ^= mask; + unchecked { mask++; } + } + + var enc = this.TextEncoding ?? Encoding.Default; + result = enc.GetString(buffer, 0, size); + } + } + else if (size > 0) + { + ushort mask = 0xAAAA; + if (size == 127) + { + size = this.BReader.ReadInt32(); + } + + var buffer = GetStringBuffer(size * 2); + this.fileStream.Read(buffer, 0, size * 2); + this.WzStructure.encryption.keys.Decrypt(buffer, 0, size * 2); + + fixed (byte* pData = buffer) + { + ushort* pChar = (ushort*)pData; + for (int i = 0; i < size; i++) + { + pChar[i] ^= mask; + unchecked { mask++; } + } + + result = new string((char*)pChar, 0, size); + } + } + else + { + return string.Empty; + } + + //memory optimize + if (result.Length <= 4) + { + for (int i = 0; i < result.Length; i++) + { + if (result[i] >= 0x80) + { + return result; + } + } + return string.Intern(result); + } + else + { + return result; + } + } + + /// + /// 为字符串解密提供缓冲区。 + /// + /// + /// + private byte[] GetStringBuffer(int size) + { + if (size <= 4096) + { + if (tempBuffer == null || tempBuffer.Length < size) + { + Array.Resize(ref tempBuffer, size); + } + return tempBuffer; + } + else + { + return new byte[size]; + } + } + + public uint CalcOffset(uint filePos, uint hashedOffset) + { + uint offset = (uint)(filePos - 0x3C) ^ 0xFFFFFFFF; + int distance; + + offset *= this.Header.HashVersion; + offset -= 0x581C3F6D; + distance = (int)offset & 0x1F; + offset = (offset << distance) | (offset >> (32 - distance)); + offset ^= hashedOffset; + offset += 0x78; + + return offset; + } + + public void GetDirTree(Wz_Node parent, bool useBaseWz = false, bool loadWzAsFolder = false) + { + List dirs = new List(); + string name = null; + int size = 0; + int cs32 = 0; + uint pos = 0, hashOffset = 0; + //int offs = 0; + + int count = ReadInt32(); + + for (int i = 0; i < count; i++) + { + switch ((int)this.BReader.ReadByte()) + { + case 0x02: + foreach (Wz_Crypto.Wz_CryptoKeyType encType in new[] { Wz_Crypto.Wz_CryptoKeyType.BMS, Wz_Crypto.Wz_CryptoKeyType.KMS, Wz_Crypto.Wz_CryptoKeyType.GMS, this.WzStructure.encryption.EncType }) + { + long oldoffset = this.FileStream.Position; + int stroffset = this.Header.HeaderSize + 1 + this.BReader.ReadInt32(); + name = this.ReadStringAt(stroffset); + if (this.WzStructure.encryption.IsLegalNodeName(name)) + { + break; + } + this.FileStream.Position = oldoffset; + stringTable.Remove(stroffset); + this.WzStructure.encryption.EncType = encType; + } + goto case 0xffff; + case 0x04: + foreach (Wz_Crypto.Wz_CryptoKeyType encType in new[] { Wz_Crypto.Wz_CryptoKeyType.BMS, Wz_Crypto.Wz_CryptoKeyType.KMS, Wz_Crypto.Wz_CryptoKeyType.GMS, this.WzStructure.encryption.EncType }) + { + long oldoffset = this.FileStream.Position; + name = this.ReadString(); + if (this.WzStructure.encryption.IsLegalNodeName(name)) + { + break; + } + this.FileStream.Position = oldoffset; + this.WzStructure.encryption.EncType = encType; + } + goto case 0xffff; + + case 0xffff: + size = this.ReadInt32(); + cs32 = this.ReadInt32(); + pos = (uint)this.bReader.BaseStream.Position; + hashOffset = this.bReader.ReadUInt32(); + + Wz_Image img = new Wz_Image(name, size, cs32, hashOffset, pos, this); + Wz_Node childNode = parent.Nodes.Add(name); + childNode.Value = img; + img.OwnerNode = childNode; + + this.imageCount++; + break; + + case 0x03: + name = this.ReadString(); + size = this.ReadInt32(); + cs32 = this.ReadInt32(); + pos = (uint)this.bReader.BaseStream.Position; + hashOffset = this.bReader.ReadUInt32(); + this.directories.Add(new Wz_Directory(name, size, cs32, hashOffset, pos, this)); + dirs.Add(name); + break; + } + } + + int dirCount = dirs.Count; + bool willLoadBaseWz = useBaseWz ? parent.Text.Equals("base.wz", StringComparison.OrdinalIgnoreCase) : false; + + var baseFolder = Path.GetDirectoryName(this.header.FileName); + + if (willLoadBaseWz && this.WzStructure.AutoDetectExtFiles) + { + for (int i = 0; i < dirCount; i++) + { + //检测文件名 + var m = Regex.Match(dirs[i], @"^([A-Za-z]+)$"); + if (m.Success) + { + string wzTypeName = m.Result("$1"); + + //检测扩展wz文件 + for (int fileID = 2; ; fileID++) + { + string extDirName = wzTypeName + fileID; + string extWzFile = Path.Combine(baseFolder, extDirName + ".wz"); + if (File.Exists(extWzFile)) + { + if (!dirs.Take(dirCount).Any(dir => extDirName.Equals(dir, StringComparison.OrdinalIgnoreCase))) + { + dirs.Add(extDirName); + } + } + else + { + break; + } + } + //检测KMST1058的wz文件 + for (int fileID = 1; ; fileID++) + { + string extDirName = wzTypeName + fileID.ToString("D3"); + string extWzFile = Path.Combine(baseFolder, extDirName + ".wz"); + if (File.Exists(extWzFile)) + { + if (!dirs.Take(dirCount).Any(dir => extDirName.Equals(dir, StringComparison.OrdinalIgnoreCase))) + { + dirs.Add(extDirName); + } + } + else + { + break; + } + } + } + } + } + + for (int i = 0; i < dirs.Count; i++) + { + string dir = dirs[i]; + Wz_Node t = parent.Nodes.Add(dir); + if (i < dirCount) + { + GetDirTree(t, false); + } + + if (t.Nodes.Count == 0) + { + this.WzStructure.has_basewz |= willLoadBaseWz; + + try + { + if (loadWzAsFolder) + { + string wzFolder = willLoadBaseWz ? Path.Combine(Path.GetDirectoryName(baseFolder), dir) : Path.Combine(baseFolder, dir); + if (Directory.Exists(wzFolder)) + { + this.wzStructure.LoadWzFolder(wzFolder, ref t, false); + if (!willLoadBaseWz) + { + var dirWzFile = t.GetValue(); + dirWzFile.Type = Wz_Type.Unknown; + dirWzFile.isSubDir = true; + } + } + } + else if (willLoadBaseWz) + { + string filePath = Path.Combine(baseFolder, dir + ".wz"); + if (File.Exists(filePath)) + { + this.WzStructure.LoadFile(filePath, t, false, loadWzAsFolder); + } + } + } + catch (Exception ex) + { + } + } + } + + parent.Nodes.Trim(); + } + + private string getFullPath(Wz_Node parent, string name) + { + List path = new List(5); + path.Add(name.ToLower()); + while (parent != null && !(parent.Value is Wz_File)) + { + path.Insert(0, parent.Text.ToLower()); + parent = parent.ParentNode; + } + if (parent != null) + { + path.Insert(0, parent.Text.ToLower().Replace(".wz", "")); + } + return string.Join("/", path.ToArray()); + } + + public void DetectWzType() + { + this.type = Wz_Type.Unknown; + if (this.node == null) + { + return; + } + + if (this.node.Nodes["smap.img"] != null + || this.node.Nodes["zmap.img"] != null) + { + this.type = Wz_Type.Base; + } + else if (this.node.Nodes["00002000.img"] != null + || this.node.Nodes["Accessory"] != null + || this.node.Nodes["Weapon"] != null) + { + this.type = Wz_Type.Character; + } + else if (this.node.Nodes["BasicEff.img"] != null + || this.node.Nodes["SetItemInfoEff.img"] != null) + { + this.type = Wz_Type.Effect; + } + else if (this.node.Nodes["Commodity.img"] != null + || this.node.Nodes["Curse.img"] != null) + { + this.type = Wz_Type.Etc; + } + else if (this.node.Nodes["Cash"] != null + || this.node.Nodes["Consume"] != null) + { + this.type = Wz_Type.Item; + } + else if (this.node.Nodes["Back"] != null + || this.node.Nodes["Obj"] != null + || this.node.Nodes["Physics.img"] != null) + { + this.type = Wz_Type.Map; + } + else if (this.node.Nodes["PQuest.img"] != null + || this.node.Nodes["QuestData"] != null) + { + this.type = Wz_Type.Quest; + } + else if (this.node.Nodes["Attacktype.img"] != null + || this.node.Nodes["Recipe_9200.img"] != null) + { + this.type = Wz_Type.Skill; + } + else if (this.node.Nodes["Bgm00.img"] != null + || this.node.Nodes["BgmUI.img"] != null) + { + this.type = Wz_Type.Sound; + } + else if (this.node.Nodes["MonsterBook.img"] != null + || this.node.Nodes["EULA.img"] != null) + { + this.type = Wz_Type.String; + } + else if (this.node.Nodes["CashShop.img"] != null + || this.node.Nodes["UIWindow.img"] != null) + { + this.type = Wz_Type.UI; + } + + if (this.type == Wz_Type.Unknown) //用文件名来判断 + { + string wzName = this.node.Text; + + Match m = Regex.Match(wzName, @"^([A-Za-z]+)_?(\d+)?(?:\.wz)?$"); + if (m.Success) + { + wzName = m.Result("$1"); + } + this.type = Enum.TryParse(wzName, true, out var result) ? result : Wz_Type.Unknown; + } + } + + public void DetectWzVersion() + { + bool DetectWithWzImage(Wz_Image testWzImg) + { + while (this.header.TryGetNextVersion()) + { + uint offs = CalcOffset(testWzImg.HashedOffsetPosition, testWzImg.HashedOffset); + + if (offs < this.header.HeaderSize || offs + testWzImg.Size > this.fileStream.Length) //img块越界 + { + continue; + } + + this.fileStream.Position = offs; + switch (this.fileStream.ReadByte()) + { + case 0x73: + case 0x1b: + //试读img第一个string + break; + default: + continue; + } + + testWzImg.Offset = offs; + if (testWzImg.TryExtract()) //试读成功 + { + testWzImg.Unextract(); + this.header.VersionChecked = true; + break; + } + } + return this.header.VersionChecked; + } + + bool DetectWithAllWzDir() + { + while (this.header.TryGetNextVersion()) + { + bool isSuccess = true; + foreach (var testDir in this.directories) + { + uint offs = CalcOffset(testDir.HashedOffsetPosition, testDir.HashedOffset); + + if (offs < this.header.HeaderSize || offs + 1 > this.fileStream.Length) // dir offset out of file size. + { + isSuccess = false; + break; + } + + this.fileStream.Position = offs; + if (this.fileStream.ReadByte() != 0) // dir data only contains one byte: 0x00 + { + isSuccess = false; + break; + } + } + + if (isSuccess) + { + this.header.VersionChecked = true; + break; + } + } + + return this.header.VersionChecked; + } + + List imgList = EnumerableAllWzImage(this.node).Where(_img => _img.WzFile == this).ToList(); + + if (this.header.VersionChecked) + { + foreach (var img in imgList) + { + img.Offset = CalcOffset(img.HashedOffsetPosition, img.HashedOffset); + } + } + else + { + //选择最小的img作为实验品 + Wz_Image minSizeImg = imgList.Where(_img => _img.Size >= 20).DefaultIfEmpty().Aggregate((_img1, _img2) => _img1.Size < _img2.Size ? _img1 : _img2); + + if (minSizeImg == null && imgList.Count > 0) + { + minSizeImg = imgList[0]; + } + + if (minSizeImg != null) + { + DetectWithWzImage(minSizeImg); + } + else if (this.directories.Count > 0) + { + DetectWithAllWzDir(); + } + + if (this.header.VersionChecked) //重新计算全部img + { + foreach (var img in imgList) + { + img.Offset = CalcOffset(img.HashedOffsetPosition, img.HashedOffset); + } + } + else //最终测试失败 那就失败吧.. + { + this.header.VersionChecked = true; + } + } + } + + public void MergeWzFile(Wz_File wz_File) + { + var children = wz_File.node.Nodes.ToList(); + wz_File.node.Nodes.Clear(); + foreach (var child in children) + { + this.node.Nodes.Add(child); + } + + if (this.mergedWzFiles == null) + { + this.mergedWzFiles = new List(); + } + this.mergedWzFiles.Add(wz_File); + + wz_File.ownerWzFile = this; + } + + private IEnumerable EnumerableAllWzImage(Wz_Node parentNode) + { + foreach (var node in parentNode.Nodes) + { + Wz_Image img = node.Value as Wz_Image; + if (img != null) + { + yield return img; + } + + if (!(node.Value is Wz_File) && node.Nodes.Count > 0) + { + foreach (var imgChild in EnumerableAllWzImage(node)) + { + yield return imgChild; + } + } + } + } + } +} diff --git a/WzComparerR2.WzLib/Wz_Header.cs b/WzComparerR2.WzLib/Wz_Header.cs new file mode 100644 index 0000000..8e0ada8 --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Header.cs @@ -0,0 +1,154 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace WzComparerR2.WzLib +{ + public class Wz_Header + { + public Wz_Header(string signature, string copyright, string file_name, int head_size, long data_size, long file_size, long dataStartPosition) + { + this.Signature = signature; + this.Copyright = copyright; + this.FileName = file_name; + this.HeaderSize = head_size; + this.DataSize = data_size; + this.FileSize = file_size; + this.DataStartPosition = dataStartPosition; + this.VersionChecked = false; + } + + public string Signature { get; private set; } + public string Copyright { get; private set; } + public string FileName { get; private set; } + + public int HeaderSize { get; private set; } + public long DataSize { get; private set; } + public long FileSize { get; private set; } + public long DataStartPosition { get; private set; } + + public bool VersionChecked { get; set; } + + public int WzVersion => this.versionDetector?.WzVersion ?? 0; + public uint HashVersion => this.versionDetector?.HashVersion ?? 0; + public bool TryGetNextVersion() => this.versionDetector?.TryGetNextVersion() ?? false; + + private IWzVersionDetector versionDetector; + + public void SetWzVersion(int wzVersion) + { + this.versionDetector = new FixedVersion(wzVersion); + } + + public void SetOrdinalVersionDetector(int encryptedVersion) + { + this.versionDetector = new OrdinalVersionDetector(encryptedVersion); + } + + public static int CalcHashVersion(int wzVersion) + { + int sum = 0; + string versionStr = wzVersion.ToString(System.Globalization.CultureInfo.InvariantCulture); + for (int j = 0; j < versionStr.Length; j++) + { + sum <<= 5; + sum += (int)versionStr[j] + 1; + } + + return sum; + } + + public interface IWzVersionDetector + { + bool TryGetNextVersion(); + int WzVersion { get; } + uint HashVersion { get; } + } + + public class FixedVersion : IWzVersionDetector + { + public FixedVersion(int wzVersion) + { + this.WzVersion = wzVersion; + this.HashVersion = (uint)CalcHashVersion(wzVersion); + } + + private bool hasReturned; + + + public int WzVersion { get; private set; } + + public uint HashVersion { get; private set; } + + public bool TryGetNextVersion() + { + if (!hasReturned) + { + hasReturned = true; + return true; + } + + return false; + } + } + + public class OrdinalVersionDetector : IWzVersionDetector + { + public OrdinalVersionDetector(int encryptVersion) + { + this.EncryptedVersion = encryptVersion; + this.versionTest = new List(); + this.hasVersionTest = new List(); + this.startVersion = -1; + } + + public int EncryptedVersion { get; private set; } + + private int startVersion; + private List versionTest; + private List hasVersionTest; + + public int WzVersion + { + get + { + int idx = this.versionTest.Count - 1; + return idx > -1 ? this.versionTest[idx] : 0; + } + } + + public uint HashVersion + { + get + { + int idx = this.hasVersionTest.Count - 1; + return idx > -1 ? this.hasVersionTest[idx] : 0; + } + } + + public bool TryGetNextVersion() + { + for (int i = startVersion + 1; i < Int16.MaxValue; i++) + { + int sum = CalcHashVersion(i); + int enc = 0xff + ^ ((sum >> 24) & 0xFF) + ^ ((sum >> 16) & 0xFF) + ^ ((sum >> 8) & 0xFF) + ^ ((sum) & 0xFF); + + // if encver does not passed, try every version one by one + if (enc == this.EncryptedVersion) + { + this.versionTest.Add(i); + this.hasVersionTest.Add((uint)sum); + startVersion = i; + return true; + } + } + + return false; + } + } + } +} diff --git a/WzComparerR2.WzLib/Wz_Image.cs b/WzComparerR2.WzLib/Wz_Image.cs new file mode 100644 index 0000000..757b16d --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Image.cs @@ -0,0 +1,403 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.Linq; + +namespace WzComparerR2.WzLib +{ + public class Wz_Image + { + public Wz_Image(string name, int size, int cs32, uint hashOff, uint hashPos, Wz_File wz_f) + { + this.Name = name; + this.WzFile = wz_f; + this.Size = size; + this.Checksum = cs32; + this.HashedOffset = hashOff; + this.HashedOffsetPosition = hashPos; + this.Node = new Wz_ImageNode(name, this); + this.Encryption = null; + + this.extr = false; + this.chec = false; + } + + private bool extr; + private bool chec; + + public string Name { get; set; } + public Wz_File WzFile { get; set; } + public int Size { get; set; } + public int Checksum { get; set; } + public uint HashedOffset { get; set; } + public uint HashedOffsetPosition { get; set; } + public long Offset { get; set; } + + public Wz_Node Node { get; private set; } + + public Wz_Node OwnerNode { get; set; } + + public Wz_Crypto Encryption { get; private set; } + + public bool IsChecksumChecked + { + get { return this.chec; } + internal set { this.chec = value; } + } + public bool IsLuaImage + { + get { return this.Name.EndsWith(".lua"); } + } + + public bool TryExtract() + { + Exception ex; + return TryExtract(out ex); + } + + public bool TryExtract(out Exception e) + { + if (!this.extr) + { + bool disabledChec = this.WzFile?.WzStructure?.ImgCheckDisabled ?? false; + if (!disabledChec && !this.chec) + { + if (this.Checksum != CalcCheckSum()) + { + e = new ArgumentException("checksum error"); + return false; + } + this.chec = true; + } + if (this.Encryption == null) + { + if (!this.IsLuaImage) + { + try + { + this.TryDetectEnc(); + if (this.Encryption == null) + { + e = null; + return false; + } + } + catch (Exception ex) + { + e = ex; + this.Unextract(); + return false; + } + } + } + + try + { + lock (this.WzFile.ReadLock) + { + this.WzFile.FileStream.Position = this.Offset; + if (!this.IsLuaImage) + { + ExtractImg(this.Offset, this.Node, 0); + this.WzFile.stringTable.Clear(); + } + else + { + ExtractLua(); + } + this.extr = true; + } + } + catch (Exception ex) + { + e = ex; + this.Unextract(); + return false; + } + } + e = null; + return true; + } + + public void Unextract() + { + this.extr = false; + this.Node.Nodes.Clear(); + } + + public unsafe int CalcCheckSum() + { + lock (this.WzFile.ReadLock) + { + this.WzFile.FileStream.Position = this.Offset; + int cs = 0; + byte[] buffer = new byte[4096]; + int count; + int size = this.Size; + while ((count = this.WzFile.FileStream.Read(buffer, 0, Math.Min(size, buffer.Length))) > 0) + { + fixed (byte* pBuffer = buffer) + { + int* p = (int*)pBuffer; + int i, j = count / 4; + for (i = 0; i < j; i++) + { + int data = *(p + i); + cs += (data & 0xff) + (data >> 8 & 0xff) + (data >> 16 & 0xff) + (data >> 24 & 0xff); + } + for (i = i * 4; i < count; i++) + { + cs += buffer[i]; + } + } + + size -= count; + } + return cs; + } + } + + private void ExtractImg(long offset, Wz_Node parent, long eob) + { + int entries = 0; + string tag = this.WzFile.ReadString(offset); + switch (tag) + { + case "Property": + this.WzFile.FileStream.Position += 2; + entries = this.WzFile.ReadInt32(); + for (int i = 0; i < entries; i++) + { + ExtractValue(offset, parent); + } + break; + + case "Shape2D#Vector2D": + parent.Value = new Wz_Vector(this.WzFile.ReadInt32(), this.WzFile.ReadInt32()); + break; + + case "Canvas": + this.WzFile.FileStream.Position++; + if (this.WzFile.BReader.ReadByte() == 0x01) + { + this.WzFile.FileStream.Position += 2; + entries = this.WzFile.ReadInt32(); + for (int i = 0; i < entries; i++) + { + ExtractValue(offset, parent); + } + } + int w = this.WzFile.ReadInt32(); + int h = this.WzFile.ReadInt32(); + int form = this.WzFile.ReadInt32() + this.WzFile.BReader.ReadByte(); + this.WzFile.FileStream.Position += 4; + int bufsize = this.WzFile.BReader.ReadInt32(); + parent.Value = new Wz_Png(w, h, bufsize - 1, form, (uint)this.WzFile.FileStream.Position + 1, this); + this.WzFile.FileStream.Position += bufsize; + break; + + case "Shape2D#Convex2D": + entries = this.WzFile.ReadInt32(); + for (int i = 0; i < entries; i++) + { + ExtractImg(offset, parent, 0); + } + break; + + case "Sound_DX8": + this.WzFile.FileStream.Position++; + int len = this.WzFile.ReadInt32(); + int ms = this.WzFile.ReadInt32(); + int headerLen = (int)(eob - len - this.WzFile.FileStream.Position); + byte[] header = this.WzFile.BReader.ReadBytes(headerLen); + parent.Value = new Wz_Sound((uint)(eob - len), len, header, ms, this); + this.WzFile.FileStream.Position = eob; + break; + + case "UOL": + this.WzFile.FileStream.Position++; + parent.Value = new Wz_Uol(this.WzFile.ReadString(offset)); + break; + + default: + throw new Exception("unknown wz tag: " + tag); + } + } + + private void TryDetectEnc() + { + Wz_Crypto crypto = this.WzFile.WzStructure.encryption; + + if (crypto.EncType != Wz_Crypto.Wz_CryptoKeyType.Unknown) + { + if (IsIllegalTag()) + { + this.Encryption = new Wz_Crypto() { EncType = crypto.EncType }; + return; + } + } + var oldenc = crypto.EncType; + crypto.EncType = Wz_Crypto.Wz_CryptoKeyType.KMS; + if (IsIllegalTag()) + { + this.Encryption = new Wz_Crypto() { EncType = crypto.EncType }; + return; + } + + crypto.EncType = Wz_Crypto.Wz_CryptoKeyType.GMS; + if (IsIllegalTag()) + { + this.Encryption = new Wz_Crypto() { EncType = crypto.EncType }; + return; + } + + crypto.EncType = Wz_Crypto.Wz_CryptoKeyType.BMS; + if (IsIllegalTag()) + { + this.Encryption = new Wz_Crypto() { EncType = crypto.EncType }; + return; + } + + crypto.EncType = oldenc; + } + + private bool IsIllegalTag() + { + this.WzFile.FileStream.Position = this.Offset; + this.WzFile.stringTable.Remove(Offset); + switch (this.WzFile.ReadString(Offset)) + { + case "Property": + case "Shape2D#Vector2D": + case "Canvas": + case "Shape2D#Convex2D": + case "Sound_DX8": + case "UOL": + return true; + default: + return false; + } + } + + + private void ExtractValue(long offset, Wz_Node parent) + { + parent = parent.Nodes.Add(this.WzFile.ReadString(offset)); + byte flag = this.WzFile.BReader.ReadByte(); + switch (flag) + { + case 0x00: + parent.Value = null; + break; + + case 0x02: + case 0x0B: + parent.Value = this.WzFile.BReader.ReadInt16(); + break; + + case 0x03: + case 0x13: + // case 0x14: + parent.Value = this.WzFile.ReadInt32(); + break; + + case 0x14: + parent.Value = this.WzFile.ReadInt64(); + break; + + case 0x04: + parent.Value = this.WzFile.ReadSingle(); + break; + + case 0x05: + parent.Value = this.WzFile.BReader.ReadDouble(); + break; + + case 0x08: + parent.Value = this.WzFile.ReadString(offset); + break; + + case 0x09: + ExtractImg(offset, parent, this.WzFile.BReader.ReadInt32() + this.WzFile.FileStream.Position); + break; + + default: + throw new Exception("读取值错误." + flag + " at Offset: " + this.WzFile.FileStream.Position); + } + } + + private void ExtractLua() + { + while(this.WzFile.FileStream.Position < this.Offset + this.Size) + { + var flag = this.WzFile.BReader.ReadByte(); + + switch (flag) + { + case 0x01: + ExtractLuaValue(this.Node); + break; + + default: + throw new Exception("读取Lua错误." + flag + " at Offset: " + this.WzFile.FileStream.Position); + } + } + } + + private void ExtractLuaValue(Wz_Node parent) + { + int len = this.WzFile.ReadInt32(); + byte[] data = this.WzFile.BReader.ReadBytes(len); + if (this.Encryption == null) + { + TryDetectLuaEnc(data); + } + this.Encryption.keys.Decrypt(data, 0, data.Length); + string luaCode = Encoding.UTF8.GetString(data); + parent.Value = luaCode; + } + + private void TryDetectLuaEnc(byte[] luaBinary) + { + Wz_Crypto crypto = this.WzFile.WzStructure.encryption; + byte[] tempBuffer = new byte[Math.Min(luaBinary.Length, 64)]; + char[] tempStr = new char[tempBuffer.Length]; + + //测试各种加密方式 判断符合度最高的 + int maxCharCount = 0; + var maxCharEnc = Wz_Crypto.Wz_CryptoKeyType.Unknown; + + foreach (var enc in new[] { + Wz_Crypto.Wz_CryptoKeyType.GMS, + Wz_Crypto.Wz_CryptoKeyType.KMS, + Wz_Crypto.Wz_CryptoKeyType.BMS + }) + { + Buffer.BlockCopy(luaBinary, 0, tempBuffer, 0, tempBuffer.Length); + crypto.EncType = enc; + crypto.keys.Decrypt(tempBuffer, 0, tempBuffer.Length); + int count = Encoding.UTF8.GetChars(tempBuffer, 0, tempBuffer.Length, tempStr, 0); + int asciiCount = tempStr.Take(count).Count(chr => 32 <= chr && chr <= 127); + + if (maxCharCount < asciiCount) + { + maxCharEnc = enc; + maxCharCount = asciiCount; + } + } + + crypto.EncType = maxCharEnc; + this.Encryption = new Wz_Crypto() { EncType = crypto.EncType }; + } + + internal class Wz_ImageNode : Wz_Node + { + public Wz_ImageNode(string nodeText, Wz_Image image) : base(nodeText) + { + this.Image = image; + } + + public Wz_Image Image { get; private set; } + } + } +} diff --git a/WzComparerR2.WzLib/Wz_Node.cs b/WzComparerR2.WzLib/Wz_Node.cs new file mode 100644 index 0000000..b3170d4 --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Node.cs @@ -0,0 +1,770 @@ +using System; +using System.Collections.Generic; +using System.Collections.ObjectModel; +using System.IO; +using System.Runtime.InteropServices; +using System.Xml; +using System.Reflection; +using System.Text.RegularExpressions; + +namespace WzComparerR2.WzLib +{ + public class Wz_Node : ICloneable, IComparable, IComparable + { + public Wz_Node() + { + this.nodes = new WzNodeCollection(this); + } + + public Wz_Node(string nodeText) + : this() + { + this.text = nodeText; + } + + //fields + private object value; + private string text; + private WzNodeCollection nodes; + private Wz_Node parentNode; + + //properties + public object Value + { + get { return value; } + set { this.value = value; } + } + + public string Text + { + get { return this.text; } + set { this.text = value; } + } + + public string FullPath + { + get + { + Stack path = new Stack(); + Wz_Node node = this; + do + { + path.Push(node.text); + node = node.parentNode; + } while (node != null); + return string.Join("\\", path.ToArray()); + } + } + + public string FullPathToFile + { + get + { + Stack path = new Stack(); + Wz_Node node = this; + do + { + if (node.value is Wz_File wzf && !wzf.IsSubDir) + { + if (node.text.EndsWith(".wz", StringComparison.OrdinalIgnoreCase)) + { + path.Push(node.text.Substring(0, node.text.Length - 3)); + } + else + { + path.Push(node.text); + } + break; + } + + path.Push(node.text); + + var img = node.GetValue(); + if (img != null) + { + node = img.OwnerNode; + } + + if (node != null) + { + node = node.parentNode; + } + } while (node != null); + return string.Join("\\", path.ToArray()); + } + } + + public WzNodeCollection Nodes + { + get { return this.nodes; } + } + + public Wz_Node ParentNode + { + get { return parentNode; } + private set { parentNode = value; } + } + + //methods + public override string ToString() + { + return this.Text + " " + (this.value != null ? this.value.ToString() : "-") + " " + this.nodes.Count; + } + + public Wz_Node FindNodeByPath(string fullPath) + { + return FindNodeByPath(fullPath, false); + } + + public Wz_Node FindNodeByPath(string fullPath, bool extractImage) + { + string[] patten = fullPath.Split('\\'); + return FindNodeByPath(extractImage, patten); + } + + public Wz_Node FindNodeByPath(bool extractImage, params string[] fullPath) + { + return FindNodeByPath(extractImage, false, fullPath); + } + + public Wz_Node FindNodeByPath(bool extractImage, bool ignoreCase, params string[] fullPath) + { + Wz_Node node = this; + + Wz_Image img; + + //首次解压 + if (extractImage && (img = this.GetValue()) != null) + { + if (img.TryExtract()) + { + node = img.Node; + } + } + + foreach (string txt in fullPath) + { + if (ignoreCase) + { + bool find = false; + + foreach (Wz_Node subNode in node.nodes) + { + if (string.Equals(subNode.text, txt, StringComparison.OrdinalIgnoreCase)) + { + find = true; + node = subNode; + } + } + if (!find) + node = null; + } + else + { + node = node.nodes[txt]; + } + + if (node == null) + return null; + + if (extractImage) + { + img = node.GetValue(); + if (img != null && img.TryExtract()) //判断是否是img + { + node = img.Node; + } + } + } + return node; + } + + public T GetValue(T defaultValue) + { + var typeT = typeof(T); + if (typeof(Wz_Image) == typeT) + { + if (this is Wz_Image.Wz_ImageNode) + { + return (T)(object)(((Wz_Image.Wz_ImageNode)this).Image); + } + else + { + return (this.value is T) ? (T)this.value : default(T); + } + } + if (this.value == null) + return defaultValue; + if (this.value is T) + return (T)this.value; + + + if (this.value is string s) + { + if (ObjectConverter.TryParse(s, out T result, out bool hasTryParse)) + { + return result; + } + if (hasTryParse) + { + return defaultValue; + } + } + + if (this.value is IConvertible iconvertible) + { + if (typeT.IsGenericType && typeT.GetGenericTypeDefinition() == typeof(Nullable<>)) + { + typeT = typeT.GetGenericArguments()[0]; + } + + try + { + T result = (T)iconvertible.ToType(typeT, null); + return result; + } + catch + { + } + } + return defaultValue; + } + + public T GetValue() + { + return GetValue(default(T)); + } + + //innerClass + public class WzNodeCollection : IEnumerable + { + public WzNodeCollection(Wz_Node owner) + + { + this.owner = owner; + this.innerCollection = null; + } + + private readonly Wz_Node owner; + private InnerCollection innerCollection; + + public Wz_Node this[int index] + { + get { return this.innerCollection?[index]; } + } + + public Wz_Node this[string key] + { + get { return this.innerCollection?[key]; } + } + + public int Count + { + get { return this.innerCollection?.Count ?? 0; } + } + + public Wz_Node Add(string nodeText) + { + this.EnsureInnerCollection(); + return this.innerCollection.Add(nodeText); + } + + public void Add(Wz_Node item) + { + this.EnsureInnerCollection(); + this.innerCollection.Add(item); + } + + public void Sort() + { + this.innerCollection?.Sort(); + } + + public void Sort(Func getKeyFunc) where T : IComparable + { + if (getKeyFunc == null) + { + this.Sort(); + } + else if (this.innerCollection != null) + { + this.innerCollection.Sort(getKeyFunc); + } + } + + public void Trim() + { + this.innerCollection?.Trim(); + } + + public void Clear() + { + this.innerCollection?.Clear(); + } + + public IEnumerator GetEnumerator() + { + return this.innerCollection?.GetEnumerator() ?? System.Linq.Enumerable.Empty().GetEnumerator(); + } + + System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator() + { + return this.GetEnumerator(); + } + + private void EnsureInnerCollection() + { + if (this.innerCollection == null) + { + this.innerCollection = new InnerCollection(this.owner); + } + } + + private class InnerCollection : KeyedCollection + { + public InnerCollection(Wz_Node owner) + : base(null, 12) + { + this.parentNode = owner; + } + + public Wz_Node Add(string nodeText) + { + Wz_Node newNode = new Wz_Node(nodeText); + this.Add(newNode); + return newNode; + } + + public new void Add(Wz_Node item) + { + base.Add(item); + if (item.parentNode != null) + { + int index = item.parentNode.nodes.innerCollection.Items.IndexOf(item); + if (index > -1) + { + item.parentNode.nodes.innerCollection.RemoveItem(index); + } + } + item.parentNode = this.parentNode; + } + + protected override void RemoveItem(int index) + { + var item = this[index]; + if (item != null) + { + item.parentNode = null; + } + base.RemoveItem(index); + } + + private readonly Wz_Node parentNode; + + public void Sort() + { + (base.Items as List)?.Sort(); + } + + public void Sort(Func getKeyFunc) where T : IComparable + { + ListSorter.Sort(base.Items as List, getKeyFunc); + } + + public void Trim() + { + (base.Items as List)?.TrimExcess(); + } + + public new Wz_Node this[string key] + { + get + { + if (key == null) + { + return null; + } + if (this.Dictionary != null) + { + Wz_Node node; + this.Dictionary.TryGetValue(key, out node); + return node; + } + else + { + List list = this.Items as List; + foreach (var node in list) + { + if (this.Comparer.Equals(this.GetKeyForItem(node), key)) + { + return node; + } + } + return null; + } + } + } + + protected override string GetKeyForItem(Wz_Node item) + { + return item.text; + } + } + + internal static class ListSorter + { + public static void Sort(List list, Func getKeyFunc) + { + T[] innerArray = list.GetType() + .GetField("_items", BindingFlags.Instance | BindingFlags.NonPublic) + .GetValue(list) as T[]; + + TKey[] keys = new TKey[list.Count]; + + for (int i = 0; i < keys.Length; i++) + { + keys[i] = getKeyFunc(innerArray[i]); + } + + Array.Sort(keys, innerArray, 0, keys.Length); + } + } + } + + public object Clone() + { + Wz_Node newNode = new Wz_Node(this.text); + newNode.value = this.value; + foreach (Wz_Node node in this.nodes) + { + Wz_Node newChild = node.Clone() as Wz_Node; + newNode.nodes.Add(newChild); + } + return newNode; + } + + int IComparable.CompareTo(object obj) + { + return ((IComparable)this).CompareTo(obj as Wz_Node); + } + + int IComparable.CompareTo(Wz_Node other) + { + if (other != null) + { + //return string.Compare(this.Text, other.Text, StringComparison.Ordinal); + return StrCmpLogicalW(this.Text, other.Text); + } + else + { + return 1; + } + } + + [DllImport("shlwapi.dll", CharSet = CharSet.Unicode)] + static extern int StrCmpLogicalW(string psz1, string psz2); + } + + public static class Wz_NodeExtension + { + public static T GetValueEx(this Wz_Node node, T defaultValue) + { + if (node == null) + return defaultValue; + return node.GetValue(defaultValue); + } + + public static T? GetValueEx(this Wz_Node node) where T : struct + { + if (node == null) + return null; + return node.GetValue(); + } + + public static Wz_Node ResolveUol(this Wz_Node node) + { + if (node == null) + return null; + Wz_Uol uol; + while ((uol = node?.GetValueEx(null)) != null) + { + node = uol.HandleUol(node); + } + return node; + } + + /// + /// 搜索node所属的wz_file,若搜索不到则返回null。 + /// + /// 要搜索的wznode。 + /// + public static Wz_File GetNodeWzFile(this Wz_Node node, bool returnClosestWzFile = false) + { + Wz_File wzfile = null; + while (node != null) + { + if ((wzfile = node.Value as Wz_File) != null) + { + if (wzfile.OwnerWzFile != null) + { + wzfile = wzfile.OwnerWzFile; + node = wzfile.Node; + } + if (!wzfile.IsSubDir || returnClosestWzFile) + { + break; + } + } + else if (node.Value is Wz_Image wzImg + || (wzImg = (node as Wz_Image.Wz_ImageNode)?.Image) != null) + { + wzfile = GetImageWzFile(wzImg, returnClosestWzFile); + break; + } + node = node.ParentNode; + } + return wzfile; + } + + public static Wz_File GetImageWzFile(this Wz_Image wzImg, bool returnClosestWzFile = false) + { + if (!returnClosestWzFile && wzImg.WzFile != null) + { + return GetNodeWzFile(wzImg.WzFile.Node, returnClosestWzFile); + } + + return wzImg.WzFile; + } + + public static int GetMergedVersion(this Wz_File wzFile) + { + if (wzFile.Header.WzVersion != 0) + { + return wzFile.Header.WzVersion; + } + foreach (var subFile in wzFile.MergedWzFiles) + { + if (subFile.Header.WzVersion != 0) + { + return subFile.Header.WzVersion; + } + } + return 0; + } + + public static Wz_Image GetNodeWzImage(this Wz_Node node) + { + Wz_Image wzImg = null; + while (node != null) + { + if ((wzImg = node.Value as Wz_Image) != null + || (wzImg = (node as Wz_Image.Wz_ImageNode)?.Image) != null) + { + break; + } + node = node.ParentNode; + } + return wzImg; + } + + public static void DumpAsXml(this Wz_Node node, XmlWriter writer) + { + object value = node.Value; + + if (value == null || value is Wz_Image) + { + writer.WriteStartElement("dir"); + writer.WriteAttributeString("name", node.Text); + } + else if (value is Wz_Png) + { + var png = (Wz_Png)value; + writer.WriteStartElement("png"); + writer.WriteAttributeString("name", node.Text); + using (var bmp = png.ExtractPng()) + { + using (var ms = new MemoryStream()) + { + bmp.Save(ms, System.Drawing.Imaging.ImageFormat.Png); + byte[] data = ms.ToArray(); + writer.WriteAttributeString("value", Convert.ToBase64String(data)); + } + } + } + else if (value is Wz_Uol) + { + var uol = (Wz_Uol)value; + writer.WriteStartElement("uol"); + writer.WriteAttributeString("name", node.Text); + writer.WriteAttributeString("value", uol.Uol); + } + else if (value is Wz_Vector) + { + var vector = (Wz_Vector)value; + writer.WriteStartElement("vector"); + writer.WriteAttributeString("name", node.Text); + writer.WriteAttributeString("value", $"{vector.X}, {vector.Y}"); + } + else if (value is Wz_Sound) + { + var sound = (Wz_Sound)value; + writer.WriteStartElement("sound"); + writer.WriteAttributeString("name", node.Text); + byte[] data = sound.ExtractSound(); + if (data == null) + { + data = new byte[sound.DataLength]; + sound.WzFile.FileStream.Seek(sound.Offset, SeekOrigin.Begin); + sound.WzFile.FileStream.Read(data, 0, sound.DataLength); + } + writer.WriteAttributeString("value", Convert.ToBase64String(data)); + } + else + { + var tag = value.GetType().Name.ToLower(); + writer.WriteStartElement(tag); + writer.WriteAttributeString("name", node.Text); + writer.WriteAttributeString("value", value.ToString()); + } + + //输出子节点 + foreach (var child in node.Nodes) + { + DumpAsXml(child, writer); + } + + //结束标识 + writer.WriteEndElement(); + } + + public static void SortByImgID(this Wz_Node.WzNodeCollection nodes) + { + if (regexImgID == null) + { + regexImgID = new Regex(@"^(\d+)\.img$", RegexOptions.Compiled); + } + + nodes.Sort(GetKey); + } + + private static Regex regexImgID; + + private static SortKey GetKey(Wz_Node node) + { + var key = new SortKey(); + var m = regexImgID.Match(node.Text); + if (m.Success) + { + key.HasID = Int32.TryParse(m.Result("$1"), out key.ImgID); + } + key.Text = node.Text; + return key; + } + + private struct SortKey : IComparable + { + public bool HasID; + public int ImgID; + public string Text; + + public int CompareTo(SortKey other) + { + if (this.HasID && other.HasID) return this.ImgID.CompareTo(other.ImgID); + return StringComparer.Ordinal.Compare(this.Text, other.Text); + } + } + } + + public static class ObjectConverter + { + private static readonly Dictionary cache = new Dictionary(); + private delegate bool TryParseFunc(string s, out T value); + + public static bool TryParse(string s, out T value, out bool hasTryParse) + { + var typeT = typeof(T); + + TryParseFunc tryParseFunc = null; + if (!cache.TryGetValue(typeT, out var dele)) + { + bool isNullable = false; + Type innerType; + if (typeT.IsGenericType && typeT.GetGenericTypeDefinition() == typeof(Nullable<>)) + { + isNullable = true; + innerType = typeT.GetGenericArguments()[0]; + } + else + { + innerType = typeT; + } + + var methodInfo = innerType.GetMethod("TryParse", + BindingFlags.Static | BindingFlags.Public, + null, + new[] { typeof(string), innerType.MakeByRefType() }, + null); + + if (methodInfo != null && methodInfo.ReturnType == typeof(bool)) + { + if (isNullable) + { + dele = Delegate.CreateDelegate(typeof(TryParseFunc<>).MakeGenericType(innerType), methodInfo); + var proxyType = typeof(NullableTryParse<>).MakeGenericType(innerType); + var proxyInstance = Activator.CreateInstance(proxyType, dele); + var proxyParseFunc = proxyType.GetMethod("TryParse", BindingFlags.Public | BindingFlags.Instance); + cache[typeT] = tryParseFunc = (TryParseFunc)Delegate.CreateDelegate(typeof(TryParseFunc), proxyInstance, proxyParseFunc); + } + else + { + cache[typeT] = tryParseFunc = (TryParseFunc)Delegate.CreateDelegate(typeof(TryParseFunc), methodInfo); + } + } + else + { + cache[typeT] = null; + } + } + else + { + tryParseFunc = dele as TryParseFunc; + } + + if (tryParseFunc != null) + { + hasTryParse = true; + return tryParseFunc(s, out value); + } + else + { + hasTryParse = false; + value = default(T); + return false; + } + } + + private class NullableTryParse where T : struct + { + public NullableTryParse(TryParseFunc func) + { + this.func = func; + } + + private readonly TryParseFunc func; + + public bool TryParse(string s, out T? value) + { + if (this.func(s, out var v)) + { + value = v; + return true; + } + else + { + value = default(T?); + return false; + } + } + } + } +} \ No newline at end of file diff --git a/WzComparerR2.WzLib/Wz_Png.cs b/WzComparerR2.WzLib/Wz_Png.cs new file mode 100644 index 0000000..44f0cad --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Png.cs @@ -0,0 +1,562 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.Drawing; +using System.IO; +using System.IO.Compression; +using System.Drawing.Imaging; +using System.Runtime.InteropServices; + +namespace WzComparerR2.WzLib +{ + public class Wz_Png + { + public Wz_Png(int w, int h, int data_length, int form, uint offs, Wz_Image wz_i) + { + this.w = w; + this.h = h; + this.data_length = data_length; + this.form = form; + this.offs = offs; + this.wz_i = wz_i; + } + + private int w; + private int h; + private int data_length; + private int form; + private uint offs; + private Wz_Image wz_i; + + /// + /// 获取或设置图片的宽度。 + /// + public int Width + { + get { return w; } + set { w = value; } + } + + /// + /// 获取或设置图片的高度。 + /// + public int Height + { + get { return h; } + set { h = value; } + } + + /// + /// 获取或设置数据块的长度。 + /// + public int DataLength + { + get { return data_length; } + set { data_length = value; } + } + + /// + /// 获取或设置数据块对于文件的偏移。 + /// + public uint Offset + { + get { return offs; } + set { offs = value; } + } + + /// + /// 获取或设置图片的数据压缩方式。 + /// + public int Form + { + get { return form; } + set { form = value; } + } + + /// + /// 获取或设置图片所属的WzFile + /// + public Wz_File WzFile + { + get { return wz_i.WzFile; } + set { wz_i.WzFile = value; } + } + + /// + /// 获取或设置图片所属的WzImage + /// + public Wz_Image WzImage + { + get { return wz_i; } + set { wz_i = value; } + } + + public byte[] GetRawData() + { + lock (this.WzFile.ReadLock) + { + DeflateStream zlib; + byte[] plainData = null; + + this.WzFile.FileStream.Position = this.Offset; + + if (this.WzFile.BReader.ReadUInt16() == 0x9C78) + { + byte[] buffer = this.WzFile.BReader.ReadBytes(this.data_length - 2); + MemoryStream dataStream = new MemoryStream(buffer); + + zlib = new DeflateStream(dataStream, CompressionMode.Decompress); + } + else + { + this.WzFile.FileStream.Position -= 2; + MemoryStream dataStream = new MemoryStream(this.DataLength); + int blocksize = 0; + int endPosition = (int)(this.DataLength + this.WzFile.FileStream.Position); + + var encKeys = this.WzImage.Encryption.keys; + + while (this.WzFile.FileStream.Position < endPosition) + { + blocksize = this.WzFile.BReader.ReadInt32(); + byte[] dataBlock = this.WzFile.BReader.ReadBytes(blocksize); + encKeys.Decrypt(dataBlock, 0, dataBlock.Length); + + dataStream.Write(dataBlock, 0, dataBlock.Length); + } + dataStream.Position = 2; + zlib = new DeflateStream(dataStream, CompressionMode.Decompress); + } + + switch (this.Form) + { + case 1: + case 257: + case 513: + plainData = new byte[this.w * this.h * 2]; + zlib.Read(plainData, 0, plainData.Length); + break; + + case 2: + plainData = new byte[this.w * this.h * 4]; + zlib.Read(plainData, 0, plainData.Length); + break; + + case 3: + plainData = new byte[((int)Math.Ceiling(this.w / 4.0)) * 4 * ((int)Math.Ceiling(this.h / 4.0)) * 4 / 8]; + zlib.Read(plainData, 0, plainData.Length); + break; + + case 517: + plainData = new byte[this.w * this.h / 128]; + zlib.Read(plainData, 0, plainData.Length); + break; + + case 1026: + case 2050: + plainData = new byte[this.w * this.h]; + zlib.Read(plainData, 0, plainData.Length); + break; + + default: + var msOut = new MemoryStream(); + zlib.CopyTo(msOut); + plainData = msOut.ToArray(); + break; + } + if (zlib != null) + { + zlib.Close(); + } + return plainData; + } + } + + public Bitmap ExtractPng() + { + byte[] pixel = this.GetRawData(); + if (pixel == null) + { + return null; + } + Bitmap pngDecoded = null; + BitmapData bmpdata; + byte[] argb; + + switch (this.form) + { + case 1: //16位argba4444 + argb = GetPixelDataBgra4444(pixel, this.w, this.h); + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format32bppArgb); + bmpdata = pngDecoded.LockBits(new Rectangle(Point.Empty, pngDecoded.Size), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb); + Marshal.Copy(argb, 0, bmpdata.Scan0, argb.Length); + pngDecoded.UnlockBits(bmpdata); + break; + + case 2: //32位argb8888 + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format32bppArgb); + bmpdata = pngDecoded.LockBits(new Rectangle(Point.Empty, pngDecoded.Size), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb); + Marshal.Copy(pixel, 0, bmpdata.Scan0, pixel.Length); + pngDecoded.UnlockBits(bmpdata); + break; + + case 3: //黑白缩略图 + argb = GetPixelDataForm3(pixel, this.w, this.h); + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format32bppArgb); + bmpdata = pngDecoded.LockBits(new Rectangle(Point.Empty, pngDecoded.Size), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb); + Marshal.Copy(argb, 0, bmpdata.Scan0, argb.Length); + pngDecoded.UnlockBits(bmpdata); + break; + + case 257: //16位argb1555 + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format16bppArgb1555); + bmpdata = pngDecoded.LockBits(new Rectangle(Point.Empty, pngDecoded.Size), ImageLockMode.WriteOnly, PixelFormat.Format16bppArgb1555); + CopyBmpDataWithStride(pixel, pngDecoded.Width * 2, bmpdata); + pngDecoded.UnlockBits(bmpdata); + break; + + case 513: //16位rgb565 + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format16bppRgb565); + bmpdata = pngDecoded.LockBits(new Rectangle(new Point(), pngDecoded.Size), ImageLockMode.WriteOnly, PixelFormat.Format16bppRgb565); + CopyBmpDataWithStride(pixel, pngDecoded.Width * 2, bmpdata); + pngDecoded.UnlockBits(bmpdata); + break; + + case 517: //16位rgb565缩略图 + argb = GetPixelDataForm517(pixel, this.w, this.h); + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format16bppRgb565); + bmpdata = pngDecoded.LockBits(new Rectangle(0, 0, this.w, this.h), ImageLockMode.WriteOnly, PixelFormat.Format16bppRgb565); + CopyBmpDataWithStride(pixel, pngDecoded.Width * 2, bmpdata); + pngDecoded.UnlockBits(bmpdata); + break; + /* pngDecoded = new Bitmap(this.w, this.h); + pngSize = this.w * this.h / 128; + plainData = new byte[pngSize]; + zlib.Read(plainData, 0, pngSize); + byte iB = 0; + for (int i = 0; i < pngSize; i++) + { + for (byte j = 0; j < 8; j++) + { + iB = Convert.ToByte(((plainData[i] & (0x01 << (7 - j))) >> (7 - j)) * 0xFF); + for (int k = 0; k < 16; k++) + { + if (x == this.w) { x = 0; y++; } + pngDecoded.SetPixel(x, y, Color.FromArgb(0xFF, iB, iB, iB)); + x++; + } + } + } + break;*/ + + case 1026: //dxt3 + argb = GetPixelDataDXT3(pixel, this.w, this.h); + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format32bppArgb); + bmpdata = pngDecoded.LockBits(new Rectangle(new Point(), pngDecoded.Size), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb); + Marshal.Copy(argb, 0, bmpdata.Scan0, argb.Length); + pngDecoded.UnlockBits(bmpdata); + break; + + case 2050: //dxt5 + argb = GetPixelDataDXT5(pixel, this.w, this.h); + pngDecoded = new Bitmap(this.w, this.h, PixelFormat.Format32bppArgb); + bmpdata = pngDecoded.LockBits(new Rectangle(new Point(), pngDecoded.Size), ImageLockMode.WriteOnly, PixelFormat.Format32bppArgb); + Marshal.Copy(argb, 0, bmpdata.Scan0, argb.Length); + pngDecoded.UnlockBits(bmpdata); + break; + } + + return pngDecoded; + } + + public static byte[] GetPixelDataBgra4444(byte[] rawData, int width, int height) + { + byte[] argb = new byte[width * height * 4]; + { + int p; + for (int i = 0; i < rawData.Length; i++) + { + p = rawData[i] & 0x0F; p |= (p << 4); argb[i * 2] = (byte)p; + p = rawData[i] & 0xF0; p |= (p >> 4); argb[i * 2 + 1] = (byte)p; + } + } + return argb; + } + + public static byte[] GetPixelDataDXT3(byte[] rawData, int width, int height) + { + byte[] pixel = new byte[width * height * 4]; + + Color[] colorTable = new Color[4]; + int[] colorIdxTable = new int[16]; + byte[] alphaTable = new byte[16]; + for (int y = 0; y < height; y += 4) + { + for (int x = 0; x < width; x += 4) + { + int off = x * 4 + y * width; + ExpandAlphaTableDXT3(alphaTable, rawData, off); + ushort u0 = BitConverter.ToUInt16(rawData, off + 8); + ushort u1 = BitConverter.ToUInt16(rawData, off + 10); + ExpandColorTable(colorTable, u0, u1); + ExpandColorIndexTable(colorIdxTable, rawData, off + 12); + + for (int j = 0; j < 4; j++) + { + for (int i = 0; i < 4; i++) + { + SetPixel(pixel, + x + i, + y + j, + width, + colorTable[colorIdxTable[j * 4 + i]], + alphaTable[j * 4 + i]); + } + } + } + } + + return pixel; + } + + public static byte[] GetPixelDataDXT5(byte[] rawData, int width, int height) + { + byte[] pixel = new byte[width * height * 4]; + + Color[] colorTable = new Color[4]; + int[] colorIdxTable = new int[16]; + byte[] alphaTable = new byte[8]; + int[] alphaIdxTable = new int[16]; + for (int y = 0; y < height; y += 4) + { + for (int x = 0; x < width; x += 4) + { + int off = x * 4 + y * width; + ExpandAlphaTableDXT5(alphaTable, rawData[off + 0], rawData[off + 1]); + ExpandAlphaIndexTableDXT5(alphaIdxTable, rawData, off + 2); + ushort u0 = BitConverter.ToUInt16(rawData, off + 8); + ushort u1 = BitConverter.ToUInt16(rawData, off + 10); + ExpandColorTable(colorTable, u0, u1); + ExpandColorIndexTable(colorIdxTable, rawData, off + 12); + + for (int j = 0; j < 4; j++) + { + for (int i = 0; i < 4; i++) + { + SetPixel(pixel, + x + i, + y + j, + width, + colorTable[colorIdxTable[j * 4 + i]], + alphaTable[alphaIdxTable[j * 4 + i]]); + } + } + } + } + + return pixel; + } + + public static unsafe byte[] GetPixelDataForm3(byte[] rawData, int width, int height) + { + byte[] pixel = new byte[width * height * 4]; + fixed (byte* pArray = pixel) + { + int* argb2 = (int*)pArray; + int w = ((int)Math.Ceiling(width / 4.0)); + int h = ((int)Math.Ceiling(height / 4.0)); + for (int y = 0; y < h; y++) + { + for (int x = 0; x < w; x++) + { + var index = (x + y * w) * 2; //原像素索引 + var index2 = x * 4 + y * width * 4; //目标像素索引 + var p = (rawData[index] & 0x0F) | ((rawData[index] & 0x0F) << 4) + | ((rawData[index] & 0xF0) | ((rawData[index] & 0xF0) >> 4)) << 8 + | ((rawData[index + 1] & 0x0F) | ((rawData[index + 1] & 0x0F) << 4)) << 16 + | ((rawData[index + 1] & 0xF0) | ((rawData[index + 1] & 0xF0) >> 4)) << 24; + + for (int i = 0; i < 4; i++) + { + if (x * 4 + i < width) + { + argb2[index2 + i] = p; + } + else + { + break; + } + } + } + //复制行 + var srcIndex = y * width * 4 * 4; + var dstIndex = srcIndex + width * 4; + for (int j = 1; j < 4; j++) + { + if (y * 4 + j < height) + { + Array.Copy(pixel, srcIndex, pixel, dstIndex, width * 4); + dstIndex += width * 4; + } + else + { + break; + } + } + } + } + return pixel; + } + + public static byte[] GetPixelDataForm517(byte[] rawData, int width, int height) + { + byte[] pixel = new byte[width * height * 2]; + int lineIndex = 0; + for (int j0 = 0, j1 = height / 16; j0 < j1; j0++) + { + var dstIndex = lineIndex; + for (int i0 = 0, i1 = width / 16; i0 < i1; i0++) + { + int idx = (i0 + j0 * i1) * 2; + byte b0 = rawData[idx]; + byte b1 = rawData[idx + 1]; + for (int k = 0; k < 16; k++) + { + pixel[dstIndex++] = b0; + pixel[dstIndex++] = b1; + } + } + + for (int k = 1; k < 16; k++) + { + Array.Copy(pixel, lineIndex, pixel, dstIndex, width * 2); + dstIndex += width * 2; + } + + lineIndex += width * 32; + } + return pixel; + } + + private static void SetPixel(byte[] pixelData, int x, int y, int width, Color color, byte alpha) + { + int offset = (y * width + x) * 4; + pixelData[offset + 0] = color.B; + pixelData[offset + 1] = color.G; + pixelData[offset + 2] = color.R; + pixelData[offset + 3] = alpha; + } + + #region DXT1 Color + private static void ExpandColorTable(Color[] color, ushort c0, ushort c1) + { + color[0] = RGB565ToColor(c0); + color[1] = RGB565ToColor(c1); + if (c0 > c1) + { + color[2] = Color.FromArgb(0xff, (color[0].R * 2 + color[1].R + 1) / 3, (color[0].G * 2 + color[1].G + 1) / 3, (color[0].B * 2 + color[1].B + 1) / 3); + color[3] = Color.FromArgb(0xff, (color[0].R + color[1].R * 2 + 1) / 3, (color[0].G + color[1].G * 2 + 1) / 3, (color[0].B + color[1].B * 2 + 1) / 3); + } + else + { + color[2] = Color.FromArgb(0xff, (color[0].R + color[1].R) / 2, (color[0].G + color[1].G) / 2, (color[0].B + color[1].B) / 2); + color[3] = Color.FromArgb(0xff, Color.Black); + } + } + + private static void ExpandColorIndexTable(int[] colorIndex, byte[] rawData, int offset) + { + for (int i = 0; i < 16; i += 4, offset++) + { + colorIndex[i + 0] = (rawData[offset] & 0x03); + colorIndex[i + 1] = (rawData[offset] & 0x0c) >> 2; + colorIndex[i + 2] = (rawData[offset] & 0x30) >> 4; + colorIndex[i + 3] = (rawData[offset] & 0xc0) >> 6; + } + } + #endregion + + #region DXT3/DXT5 Alpha + private static void ExpandAlphaTableDXT3(byte[] alpha, byte[] rawData, int offset) + { + for (int i = 0; i < 16; i += 2, offset++) + { + alpha[i + 0] = (byte)(rawData[offset] & 0x0f); + alpha[i + 1] = (byte)((rawData[offset] & 0xf0) >> 4); + } + for (int i = 0; i < 16; i++) + { + alpha[i] = (byte)(alpha[i] | (alpha[i] << 4)); + } + } + + private static void ExpandAlphaTableDXT5(byte[] alpha, byte a0, byte a1) + { + alpha[0] = a0; + alpha[1] = a1; + if (a0 > a1) + { + for(int i = 2; i < 8; i++) + { + alpha[i] = (byte)(((8 - i) * a0 + (i - 1) * a1 + 3) / 7); + } + } + else + { + for(int i = 2; i < 6; i++) + { + alpha[i] = (byte)(((6 - i) * a0 + (i - 1) * a1 + 2) / 5); + } + alpha[6] = 0; + alpha[7] = 255; + } + } + + private static void ExpandAlphaIndexTableDXT5(int[] alphaIndex, byte[] rawData, int offset) + { + for (int i = 0; i < 16; i += 8, offset += 3) + { + int flags = rawData[offset] + | (rawData[offset + 1] << 8) + | (rawData[offset + 2] << 16); + for(int j = 0; j < 8; j++) + { + int mask = 0x07 << (3 * j); + alphaIndex[i + j] = (flags & mask) >> (3 * j); + } + } + } + #endregion + + public static Color RGB565ToColor(ushort val) + { + const int rgb565_mask_r = 0xf800; + const int rgb565_mask_g = 0x07e0; + const int rgb565_mask_b = 0x001f; + int r = (val & rgb565_mask_r) >> 11; + int g = (val & rgb565_mask_g) >> 5; + int b = (val & rgb565_mask_b); + var c = Color.FromArgb( + (r << 3) | (r >> 2), + (g << 2) | (g >> 4), + (b << 3) | (b >> 2)); + return c; + } + + public static void CopyBmpDataWithStride(byte[] source, int stride, BitmapData bmpData) + { + if (bmpData.Stride == stride) + { + Marshal.Copy(source, 0, bmpData.Scan0, source.Length); + } + else + { + for (int y = 0; y < bmpData.Height; y++) + { + Marshal.Copy(source, stride * y, bmpData.Scan0 + bmpData.Stride * y, stride); + } + } + + } + } +} diff --git a/WzComparerR2.WzLib/Wz_Sound.cs b/WzComparerR2.WzLib/Wz_Sound.cs new file mode 100644 index 0000000..7a00405 --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Sound.cs @@ -0,0 +1,196 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace WzComparerR2.WzLib +{ + public class Wz_Sound + { + public Wz_Sound(uint offset, int length, byte[] header, int ms, Wz_Image wz_i) + { + this.offset = offset; + this.dataLength = length; + this.header = header; + this.ms = ms; + this.wz_i = wz_i; + TryDecryptHeader(); + } + + private uint offset; + private byte[] header; + private int dataLength; + private int ms; + + private Wz_Image wz_i; + + /// + /// 获取或设置数据块对于文件的偏移。 + /// + public uint Offset + { + get { return offset; } + set { offset = value; } + } + + /// + /// 获取或设置头部字节段。 + /// + public byte[] Header + { + get { return header; } + set { header = value; } + } + + public int Frequency + { + get + { + if (header == null || header.Length < 0x3c) + { + return 0; + } + return BitConverter.ToInt32(header, 0x38); + } + } + + /// + /// 获取或设置数据块的长度。 + /// + public int DataLength + { + get { return dataLength; } + set { dataLength = value; } + } + + /// + /// 获取或设置Mp3的声音毫秒数。 + /// + public int Ms + { + get { return ms; } + set { ms = value; } + } + + /// + /// 获取或设置图片所属的WzFile。 + /// + public Wz_File WzFile + { + get { return wz_i.WzFile; } + set { wz_i.WzFile = value; } + } + + /// + /// 获取或设置图片所属的WzImage。 + /// + public Wz_Image WzImage + { + get { return wz_i; } + set { wz_i = value; } + } + + public Wz_SoundType SoundType + { + get + { + Wz_SoundType soundType; + if (this.header == null) + { + soundType = Wz_SoundType.Mp3; + } + else + { + switch (this.header.Length) + { + default: + case 0x52: + soundType = Wz_SoundType.Mp3; + break; + + case 0x46: + { + if (this.Frequency == this.dataLength) + { + soundType = Wz_SoundType.Binary; + } + else + { + soundType = Wz_SoundType.WavRaw; + } + } + break; + } + } + + return soundType; + } + } + + public byte[] ExtractSound() + { + switch (this.SoundType) + { + case Wz_SoundType.Mp3: + { + byte[] data = new byte[this.dataLength]; + this.WzFile.FileStream.Seek(this.offset, System.IO.SeekOrigin.Begin); + this.WzFile.FileStream.Read(data, 0, this.dataLength); + return data; + } + case Wz_SoundType.WavRaw: + { + byte[] data = new byte[this.dataLength + 44]; + this.WzFile.FileStream.Seek(this.offset, System.IO.SeekOrigin.Begin); + this.WzFile.FileStream.Read(data, 44, this.dataLength); + byte[] wavHeader = new byte[44]{ + 0x52,0x49,0x46,0x46, //"RIFF" + 0,0,0,0, //ChunkSize + 0x57,0x41,0x56,0x45, //"WAVE" + + 0x66,0x6d,0x74,0x20, //"fmt " + 0x10,0,0,0, //chunk1Size + 0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0, // copy16字节 + + 0x64,0x61,0x74,0x61, //"data" + 0,0,0,0 //chunk2Size + }; + Array.Copy(BitConverter.GetBytes(this.dataLength + 36), 0, wavHeader, 4, 4); + Array.Copy(this.header, 0x34, wavHeader, 20, 16); + Array.Copy(BitConverter.GetBytes(this.dataLength), 0, wavHeader, 40, 4); + Array.Copy(wavHeader, data, wavHeader.Length); + return data; + } + } + return null; + } + + private void TryDecryptHeader() + { + if (this.header == null) + { + return; + } + if (this.header.Length > 51) + { + byte waveFormatLen = this.header[51]; + if (this.header.Length != 52 + waveFormatLen) //长度错误 + { + return; + } + int cbSize = BitConverter.ToUInt16(this.header, 52 + 16); + if (cbSize + 18 != waveFormatLen) + { + byte[] tempHeader = new byte[waveFormatLen]; + Buffer.BlockCopy(this.header, 52, tempHeader, 0, tempHeader.Length); + var enc = this.WzImage.Encryption; + enc.keys.Decrypt(tempHeader, 0, tempHeader.Length); //解密 + cbSize = BitConverter.ToUInt16(tempHeader, 16); //重新验证 + if (cbSize + 18 == waveFormatLen) + { + Buffer.BlockCopy(tempHeader, 0, this.header, 52, tempHeader.Length); + } + } + } + } + } +} diff --git a/WzComparerR2.WzLib/Wz_SoundType.cs b/WzComparerR2.WzLib/Wz_SoundType.cs new file mode 100644 index 0000000..6de7f7a --- /dev/null +++ b/WzComparerR2.WzLib/Wz_SoundType.cs @@ -0,0 +1,13 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace WzComparerR2.WzLib +{ + public enum Wz_SoundType + { + Mp3 = 0, + WavRaw = 1, + Binary = 2, + } +} diff --git a/WzComparerR2.WzLib/Wz_Structure.cs b/WzComparerR2.WzLib/Wz_Structure.cs new file mode 100644 index 0000000..521296e --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Structure.cs @@ -0,0 +1,269 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.IO; +using System.Linq; + +namespace WzComparerR2.WzLib +{ + public class Wz_Structure + { + public Wz_Structure() + { + this.wz_files = new List(); + this.encryption = new Wz_Crypto(); + this.img_number = 0; + this.has_basewz = false; + this.TextEncoding = Wz_Structure.DefaultEncoding; + this.AutoDetectExtFiles = Wz_Structure.DefaultAutoDetectExtFiles; + this.ImgCheckDisabled = Wz_Structure.DefaultImgCheckDisabled; + } + + public List wz_files; + public Wz_Crypto encryption; + public Wz_Node WzNode; + public int img_number; + public bool has_basewz; + public bool sorted; //暂时弃用 + + public Encoding TextEncoding { get; set; } + public bool AutoDetectExtFiles { get; set; } + public bool ImgCheckDisabled { get; set; } + + public void Clear() + { + foreach (Wz_File f in this.wz_files) + { + f.Close(); + } + this.wz_files.Clear(); + this.encryption.Reset(); + this.img_number = 0; + this.has_basewz = false; + this.WzNode = null; + this.sorted = false; + } + + public void calculate_img_count() + { + foreach (Wz_File f in this.wz_files) + { + this.img_number += f.ImageCount; + } + } + + public void Load(string fileName, bool useBaseWz = false) + { + //现在我们已经不需要list了 + this.WzNode = new Wz_Node(Path.GetFileName(fileName)); + if (Path.GetFileName(fileName).ToLower() == "list.wz") + { + this.encryption.LoadListWz(Path.GetDirectoryName(fileName)); + foreach (string list in this.encryption.List) + { + WzNode.Nodes.Add(list); + } + } + else + { + LoadFile(fileName, WzNode, useBaseWz); + } + calculate_img_count(); + } + + public Wz_File LoadFile(string fileName, Wz_Node node, bool useBaseWz = false, bool loadWzAsFolder = false) + { + Wz_File file = null; + + try + { + file = new Wz_File(fileName, this); + if (!file.Loaded) + { + throw new Exception("The file is not a valid wz file."); + } + this.wz_files.Add(file); + file.TextEncoding = this.TextEncoding; + if (!this.encryption.encryption_detected) + { + this.encryption.DetectEncryption(file); + } + node.Value = file; + file.Node = node; + file.FileStream.Position = file.Header.DataStartPosition; + file.GetDirTree(node, useBaseWz, loadWzAsFolder); + file.DetectWzType(); + file.DetectWzVersion(); + return file; + } + catch + { + if (file != null) + { + file.Close(); + this.wz_files.Remove(file); + } + throw; + } + } + + public void LoadImg(string fileName) + { + this.WzNode = new Wz_Node(Path.GetFileName(fileName)); + this.LoadImg(fileName, WzNode); + } + + public void LoadImg(string fileName, Wz_Node node) + { + Wz_File file = null; + + try + { + file = new Wz_File(fileName, this); + file.TextEncoding = this.TextEncoding; + var imgNode = new Wz_Node(node.Text); + //跳过checksum检测 + var img = new Wz_Image(node.Text, (int)file.FileStream.Length, 0, 0, 0, file) + { + OwnerNode = imgNode, + Offset = 0, + IsChecksumChecked = true + }; + imgNode.Value = img; + node.Nodes.Add(imgNode); + this.wz_files.Add(file); + } + catch + { + file?.Close(); + throw; + } + } + + public void LoadKMST1125DataWz(string fileName) + { + LoadWzFolder(Path.GetDirectoryName(fileName), ref this.WzNode, true); + calculate_img_count(); + } + + public bool IsKMST1125WzFormat(string fileName) + { + if (!string.Equals(Path.GetExtension(fileName), ".wz", StringComparison.OrdinalIgnoreCase)) + { + return false; + } + + string iniFile = Path.ChangeExtension(fileName, ".ini"); + if (!File.Exists(iniFile)) + { + return false; + } + + // check if the file is an empty wzfile + using (var file = new Wz_File(fileName, this)) + { + if (!file.Loaded) + { + return false; + } + var tempNode = new Wz_Node(); + if (!this.encryption.encryption_detected) + { + this.encryption.DetectEncryption(file); + } + file.FileStream.Position = file.Header.DataStartPosition; + file.GetDirTree(tempNode); + return file.ImageCount == 0; + } + } + + public void LoadWzFolder(string folder, ref Wz_Node node, bool useBaseWz = false) + { + string baseName = Path.Combine(folder, Path.GetFileName(folder)); + string entryWzFileName = Path.ChangeExtension(baseName, ".wz"); + string iniFileName = Path.ChangeExtension(baseName, ".ini"); + Func extraWzFileName = _index => Path.ChangeExtension($"{baseName}_{_index:D3}", ".wz"); + + // load iniFile + int? lastWzIndex = null; + if (File.Exists(iniFileName)) + { + var iniConf = File.ReadAllLines(iniFileName).Select(row => + { + string[] columns = row.Split('|'); + string key = columns.Length > 0 ? columns[0] : null; + string value = columns.Length > 1 ? columns[1] : null; + return new KeyValuePair(key, value); + }); + if (int.TryParse(iniConf.FirstOrDefault(kv => kv.Key == "LastWzIndex").Value, out var indexFromIni)) + { + lastWzIndex = indexFromIni; + } + } + + // ini file missing or unexpected format + if (lastWzIndex == null) + { + for (int i = 0; ; i++) + { + string extraFile = extraWzFileName(i); + if (!File.Exists(extraFile)) + { + break; + } + lastWzIndex = i; + } + } + + // load entry file + if (node == null) + { + node = new Wz_Node(Path.GetFileName(entryWzFileName)); + } + var entryWzf = this.LoadFile(entryWzFileName, node, useBaseWz, true); + + // load extra file + if (lastWzIndex != null) + { + for (int i = 0, j = lastWzIndex.Value; i <= j; i++) + { + string extraFile = extraWzFileName(i); + var tempNode = new Wz_Node(Path.GetFileName(extraFile)); + var extraWzf = this.LoadFile(extraFile, tempNode, false, true); + + /* + * there is a little hack here, we'll move all img to the entry file, and each img still refers to the original wzfile. + * before: + * base.wz (Wz_File) + * |- a.img (Wz_Image) + * base_000.wz (Wz_File) + * |- b.img (Wz_Image) { wz_f = base_000.wz } + * + * after: + * base.wz (Wz_File) { mergedFiles = [base_000.wz] } + * |- a.img (Wz_Image) { wz_f = base.wz } + * |- b.img (Wz_Image) { wz_f = base_000.wz } + * + * this.wz_files references all opened files so they can be closed correctly. + */ + + entryWzf.MergeWzFile(extraWzf); + } + } + } + + #region Global Settings + public static Encoding DefaultEncoding + { + get { return _defaultEncoding ?? Encoding.Default; } + set { _defaultEncoding = value; } + } + + private static Encoding _defaultEncoding; + + public static bool DefaultAutoDetectExtFiles { get; set; } + + public static bool DefaultImgCheckDisabled { get; set; } + #endregion + } +} diff --git a/WzComparerR2.WzLib/Wz_Type.cs b/WzComparerR2.WzLib/Wz_Type.cs new file mode 100644 index 0000000..826190e --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Type.cs @@ -0,0 +1,30 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace WzComparerR2.WzLib +{ + /// + /// 标识Wz_File的内部类型。 + /// + public enum Wz_Type + { + Unknown = 0, + Base, + Character, + Effect, + Etc, + Item, + Map, + Mob, + Morph, + Npc, + Quest, + Reactor, + Skill, + Sound, + String, + TamingMob, + UI, + } +} diff --git a/WzComparerR2.WzLib/Wz_Uol.cs b/WzComparerR2.WzLib/Wz_Uol.cs new file mode 100644 index 0000000..2809799 --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Uol.cs @@ -0,0 +1,78 @@ +using System; +using System.Collections.Generic; +using System.Text; + +namespace WzComparerR2.WzLib +{ + public class Wz_Uol + { + public Wz_Uol(string uol) + { + this.uol = uol; + } + + private string uol; + + /// + /// 获取或设置连接路径字符串。 + /// + public string Uol + { + get { return uol; } + set { uol = value; } + } + + public Wz_Node HandleUol(Wz_Node currentNode) + { + if (currentNode == null || currentNode.ParentNode == null || string.IsNullOrEmpty(uol)) + return null; + string[] dirs = this.uol.Split('/'); + currentNode = currentNode.ParentNode; + + bool outImg = false; + + for (int i = 0; i < dirs.Length; i++) + { + string dir = dirs[i]; + if (dir == "..") + { + if (currentNode.ParentNode == null) + { + Wz_Image img = currentNode.GetValueEx(null); + if (img != null) + { + currentNode = img.OwnerNode.ParentNode; + outImg = true; + } + else + { + currentNode = null; + } + } + else + { + currentNode = currentNode.ParentNode; + } + } + else + { + var dirNode = currentNode.FindNodeByPath(dir); + + if (dirNode == null && outImg) + { + dirNode = currentNode.FindNodeByPath(true, dir + ".img"); + if (dirNode.GetValueEx(null) != null) + { + outImg = false; + } + } + + currentNode = dirNode; + } + if (currentNode == null) + return null; + } + return currentNode; + } + } +} diff --git a/WzComparerR2.WzLib/Wz_Vector.cs b/WzComparerR2.WzLib/Wz_Vector.cs new file mode 100644 index 0000000..873eab1 --- /dev/null +++ b/WzComparerR2.WzLib/Wz_Vector.cs @@ -0,0 +1,42 @@ +using System; +using System.Collections.Generic; +using System.Text; +using System.Drawing; + +namespace WzComparerR2.WzLib +{ + public class Wz_Vector + { + public Wz_Vector(int x, int y) + { + this.x = x; + this.y = y; + } + + private int x; + private int y; + + /// + /// 获取或设置向量的X值。 + /// + public int X + { + get { return x; } + set { x = value; } + } + + /// + /// 获取或设置向量的Y值。 + /// + public int Y + { + get { return y; } + set { y = value; } + } + + public static implicit operator Point(Wz_Vector vector) + { + return vector == null ? new Point() : new Point(vector.x, vector.y); + } + } +}