2020-03-18 12:30:45 -04:00
|
|
|
|
using System;
|
|
|
|
|
using System.Collections.Generic;
|
2020-10-02 16:51:22 +10:00
|
|
|
|
using System.IO;
|
2020-03-18 12:30:45 -04:00
|
|
|
|
using System.Linq;
|
|
|
|
|
using System.Text;
|
|
|
|
|
using System.Threading.Tasks;
|
2020-10-02 16:51:22 +10:00
|
|
|
|
using Toolbox.Library;
|
2020-03-18 12:30:45 -04:00
|
|
|
|
using Toolbox.Library.IO;
|
2020-10-02 16:51:22 +10:00
|
|
|
|
using Toolbox.Library.Security.Cryptography;
|
2020-03-18 12:30:45 -04:00
|
|
|
|
|
|
|
|
|
namespace FirstPlugin
|
|
|
|
|
{
|
|
|
|
|
public class BCSVParse
|
|
|
|
|
{
|
|
|
|
|
public class Field
|
|
|
|
|
{
|
|
|
|
|
public uint Hash { get; set; }
|
|
|
|
|
public uint Offset { get; set; }
|
|
|
|
|
|
|
|
|
|
public object Value;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public class DataEntry
|
|
|
|
|
{
|
|
|
|
|
public Dictionary<string, object> Fields;
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 16:53:26 -04:00
|
|
|
|
static Dictionary<uint, string> hashes = new Dictionary<uint, string>();
|
|
|
|
|
public static Dictionary<uint, string> Hashes
|
|
|
|
|
{
|
|
|
|
|
get
|
|
|
|
|
{
|
2020-10-02 16:51:22 +10:00
|
|
|
|
if (hashes.Count == 0)
|
|
|
|
|
CalculateHashes();
|
2020-03-18 16:53:26 -04:00
|
|
|
|
return hashes;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 12:30:45 -04:00
|
|
|
|
public List<DataEntry> Entries = new List<DataEntry>();
|
|
|
|
|
|
|
|
|
|
public void Read(FileReader reader)
|
|
|
|
|
{
|
|
|
|
|
uint numEntries = reader.ReadUInt32();
|
|
|
|
|
uint entrySize = reader.ReadUInt32();
|
|
|
|
|
ushort numFields = reader.ReadUInt16();
|
2020-03-18 14:10:37 -04:00
|
|
|
|
byte flag1 = reader.ReadByte();
|
|
|
|
|
byte flag2 = reader.ReadByte();
|
|
|
|
|
if (flag1 == 1)
|
|
|
|
|
{
|
|
|
|
|
uint magic = reader.ReadUInt32();
|
|
|
|
|
uint unk = reader.ReadUInt32(); //Always 100000
|
|
|
|
|
reader.ReadUInt32();//0
|
|
|
|
|
reader.ReadUInt32();//0
|
|
|
|
|
}
|
2020-03-18 12:30:45 -04:00
|
|
|
|
|
|
|
|
|
Field[] fields = new Field[numFields];
|
|
|
|
|
for (int i = 0; i < numFields; i++) {
|
|
|
|
|
fields[i] = new Field()
|
|
|
|
|
{
|
|
|
|
|
Hash = reader.ReadUInt32(),
|
|
|
|
|
Offset = reader.ReadUInt32(),
|
|
|
|
|
};
|
|
|
|
|
}
|
|
|
|
|
for (int i = 0; i < numEntries; i++)
|
|
|
|
|
{
|
|
|
|
|
DataEntry entry = new DataEntry();
|
|
|
|
|
Entries.Add(entry);
|
|
|
|
|
entry.Fields = new Dictionary<string, object>();
|
|
|
|
|
|
|
|
|
|
long pos = reader.Position;
|
|
|
|
|
for (int f = 0; f < fields.Length; f++)
|
|
|
|
|
{
|
|
|
|
|
DataType type = DataType.String;
|
|
|
|
|
uint size = entrySize - fields[f].Offset;
|
|
|
|
|
if (f < fields.Length - 1) {
|
|
|
|
|
size = fields[f + 1].Offset - fields[f].Offset;
|
|
|
|
|
}
|
|
|
|
|
if (size == 1)
|
|
|
|
|
type = DataType.Byte;
|
|
|
|
|
if (size == 2)
|
2020-03-18 17:09:33 -04:00
|
|
|
|
type = DataType.Int16;
|
2020-03-18 12:30:45 -04:00
|
|
|
|
if (size == 4)
|
2020-03-18 17:09:33 -04:00
|
|
|
|
type = DataType.Int32;
|
2020-03-18 12:30:45 -04:00
|
|
|
|
|
|
|
|
|
reader.SeekBegin(pos + fields[f].Offset);
|
|
|
|
|
object value = 0;
|
|
|
|
|
string name = fields[f].Hash.ToString("x");
|
|
|
|
|
switch (type)
|
|
|
|
|
{
|
|
|
|
|
case DataType.Byte:
|
|
|
|
|
value = reader.ReadByte();
|
|
|
|
|
break;
|
|
|
|
|
case DataType.Float:
|
|
|
|
|
value = reader.ReadSingle();
|
|
|
|
|
break;
|
2020-03-18 17:09:33 -04:00
|
|
|
|
case DataType.Int16:
|
|
|
|
|
value = reader.ReadInt16();
|
2020-03-18 12:30:45 -04:00
|
|
|
|
break;
|
2020-03-18 17:09:33 -04:00
|
|
|
|
case DataType.Int32:
|
|
|
|
|
value = reader.ReadInt32();
|
|
|
|
|
if (IsFloatValue((int)value))
|
|
|
|
|
{
|
|
|
|
|
reader.Seek(-4);
|
|
|
|
|
value = reader.ReadSingle();
|
|
|
|
|
}
|
2020-03-18 12:30:45 -04:00
|
|
|
|
break;
|
|
|
|
|
case DataType.String:
|
|
|
|
|
value = reader.ReadZeroTerminatedString(Encoding.UTF8);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 16:53:26 -04:00
|
|
|
|
if (Hashes.ContainsKey(fields[f].Hash))
|
|
|
|
|
name = Hashes[fields[f].Hash];
|
|
|
|
|
|
2020-03-18 12:30:45 -04:00
|
|
|
|
entry.Fields.Add(name, value);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
reader.SeekBegin(pos + entrySize);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 17:09:33 -04:00
|
|
|
|
private bool IsFloatValue(int value) {
|
2020-03-18 16:53:26 -04:00
|
|
|
|
return value.ToString().Length > 6;
|
|
|
|
|
}
|
|
|
|
|
|
2020-03-18 12:30:45 -04:00
|
|
|
|
public enum DataType
|
|
|
|
|
{
|
|
|
|
|
Byte,
|
2020-03-18 17:09:33 -04:00
|
|
|
|
Int16,
|
|
|
|
|
Int32,
|
|
|
|
|
Int64,
|
2020-03-18 12:30:45 -04:00
|
|
|
|
Float,
|
|
|
|
|
String,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void Write(FileWriter writer)
|
|
|
|
|
{
|
2020-03-18 13:42:16 -04:00
|
|
|
|
writer.Write(Entries.FirstOrDefault().Fields.Count);
|
2020-03-18 12:30:45 -04:00
|
|
|
|
}
|
2020-03-18 16:53:26 -04:00
|
|
|
|
|
|
|
|
|
private static void CalculateHashes()
|
|
|
|
|
{
|
2020-10-02 16:51:22 +10:00
|
|
|
|
string dir = Path.Combine(Runtime.ExecutableDir, "Hashes");
|
|
|
|
|
if (!Directory.Exists(dir))
|
|
|
|
|
return;
|
2020-03-18 16:53:26 -04:00
|
|
|
|
|
2020-10-02 16:51:22 +10:00
|
|
|
|
foreach (var file in Directory.GetFiles(dir))
|
|
|
|
|
{
|
|
|
|
|
if (Utils.GetExtension(file) != ".txt")
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
foreach (string hashStr in File.ReadAllLines(file))
|
|
|
|
|
{
|
|
|
|
|
CheckHash(hashStr);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
private static void CheckHash(string hashStr)
|
|
|
|
|
{
|
|
|
|
|
uint hash = Crc32.Compute(hashStr);
|
|
|
|
|
if (!hashes.ContainsKey(hash))
|
|
|
|
|
hashes.Add(hash, hashStr);
|
2020-03-18 16:53:26 -04:00
|
|
|
|
}
|
2020-03-18 12:30:45 -04:00
|
|
|
|
}
|
|
|
|
|
}
|