1
1
mirror of https://github.com/OpenDiablo2/OpenDiablo2 synced 2024-06-30 02:55:23 +00:00
OpenDiablo2/OpenDiablo2.Core/TextDictionary.cs
Mike Bundy b4d362fc0d Change TextDictionary Population + Read char desc from there (#19)
* Change MPQ text reading

Previously checked for there to be exactly 3 commas, changed to check that line doesn't start with hash or slash (includes/comments) + that it has 3 or more commas.

This might cause issues later, but it's needed for now otherwise lines get skipped.

* Read in Not Xpac Char desc from dictionary

Also added a string utils class. How do I test with .Net?
2018-11-25 19:06:22 -05:00

110 lines
3.7 KiB
C#

using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Text;
using System.Threading.Tasks;
using OpenDiablo2.Common;
using OpenDiablo2.Common.Interfaces;
namespace OpenDiablo2.Core
{
struct HashTableEntry
{
public bool IsActive { get; set; }
public UInt16 Index { get; set; }
public UInt32 HashValue { get; set; }
public UInt32 IndexString { get; set; }
public UInt32 NameString { get; set; }
public UInt16 NameLength { get; set; }
}
public sealed class TextDictionary : ITextDictionary
{
private readonly IMPQProvider mpqProvider;
private Dictionary<string, string> lookupTable = new Dictionary<string, string>();
public TextDictionary(IMPQProvider mpqProvider)
{
this.mpqProvider = mpqProvider;
LoadDictionary();
LoadExpansionTable();
}
private void LoadExpansionTable()
{
using (var stream = mpqProvider.GetStream(ResourcePaths.ExpansionStringTable))
using (var br = new BinaryReader(stream))
{
br.ReadBytes(2); // CRC
var numberOfElements = br.ReadUInt16();
var hashTableSize = br.ReadUInt32();
br.ReadByte(); // Version (always 0)
var stringOffset = br.ReadUInt32();
var numberOfLoopsOffset = br.ReadUInt32();
var fileSize = br.ReadUInt32();
var elementIndexes = new List<UInt16>();
for (var elementIndex = 0; elementIndex < numberOfElements; elementIndex++)
{
elementIndexes.Add(br.ReadUInt16());
}
var hashEntries = new List<HashTableEntry>();
for (var hashEntryIndex = 0; hashEntryIndex < hashTableSize; hashEntryIndex++)
{
hashEntries.Add(new HashTableEntry
{
IsActive = br.ReadByte() == 1,
Index = br.ReadUInt16(),
HashValue = br.ReadUInt32(),
IndexString = br.ReadUInt32(),
NameString = br.ReadUInt32(),
NameLength = br.ReadUInt16()
});
}
foreach (var hashEntry in hashEntries.Where(x => x.IsActive))
{
stream.Seek(hashEntry.NameString, SeekOrigin.Begin);
var value = Encoding.ASCII.GetString(br.ReadBytes(hashEntry.NameLength - 1));
stream.Seek(hashEntry.IndexString, SeekOrigin.Begin);
var key = "";
while(true)
{
var b = br.ReadByte();
if (b == 0)
break;
key += (char)b;
}
lookupTable[key] = value;
}
}
}
private void LoadDictionary()
{
var text = mpqProvider.GetTextFile(ResourcePaths.EnglishTable);
var rowsToLoad = text.Where(
x => x.Length > 0 &&
!x.StartsWith("//") &&
!x.StartsWith("#") &&
x.Split(',').Length >= 3
).Select(x => x.Split(new[] {','}, 3).Select(z => z.Trim()).ToArray());
foreach (var row in rowsToLoad)
lookupTable[row[1]] = !(row[2].StartsWith("\"") && row[2].EndsWith("\""))
? row[2]
: row[2].Substring(1, row[2].Length - 2);
}
public string Translate(string key) => lookupTable[key];
}
}