#region ================== Namespaces using System; using System.Collections.Generic; using System.IO; using CodeImp.DoomBuilder.Geometry; using CodeImp.DoomBuilder.GZBuilder.Data; using SlimDX; #endregion namespace CodeImp.DoomBuilder.ZDoom { public sealed class VoxeldefParser : ZDTextParser { private Dictionary entries; //sprite name, entry internal Dictionary Entries { get { return entries; } } public override bool Parse(Stream stream, string sourcefilename) { base.Parse(stream, sourcefilename); entries = new Dictionary(StringComparer.Ordinal); string prevToken = string.Empty; List spriteNames = new List(); string modelName = string.Empty; // Continue until at the end of the stream while(SkipWhitespace(true)) { string token = ReadToken(); if(token != null) { token = StripTokenQuotes(token).ToLowerInvariant(); if(token == ",") //previous token was a sprite name { if(!string.IsNullOrEmpty(prevToken) && !spriteNames.Contains(prevToken)) spriteNames.Add(prevToken); prevToken = token.ToUpperInvariant(); } else if(token == "=") //next token should be a voxel model name { if(!string.IsNullOrEmpty(prevToken) && !spriteNames.Contains(prevToken)) spriteNames.Add(prevToken); SkipWhitespace(true); token = ReadToken(); if(string.IsNullOrEmpty(token)) { General.ErrorLogger.Add(ErrorType.Error, "Unable to get voxel model name from '" + sourcefilename + "', line " + GetCurrentLineNumber()); spriteNames.Clear(); continue; } modelName = StripTokenQuotes(token).ToLowerInvariant(); } else if(token == "{") //read the settings { ModelData mde = new ModelData(); mde.IsVoxel = true; float scale = 1.0f; float angleoffset = 0; while(SkipWhitespace(true)) { token = ReadToken(); if(!string.IsNullOrEmpty(token)) { token = StripTokenQuotes(token).ToLowerInvariant(); if(token == "}") //store data { if(!string.IsNullOrEmpty(modelName) && spriteNames.Count > 0) { mde.ModelNames.Add(modelName); mde.SetTransform(Matrix.RotationZ(Angle2D.DegToRad(angleoffset)), Matrix.Identity, new Vector3(scale)); foreach(string s in spriteNames) { if(entries.ContainsKey(s)) //TODO: is this a proper behaviour? { entries[s] = mde; } else { entries.Add(s, mde); } } //reset local data modelName = string.Empty; prevToken = string.Empty; spriteNames.Clear(); } break; } else if(token == "overridepalette") { mde.OverridePalette = true; } else if(token == "angleoffset") { SkipWhitespace(true); token = StripTokenQuotes(ReadToken()); if(token != "=") { General.ErrorLogger.Add(ErrorType.Error, "Error in " + sourcefilename + " at line " + GetCurrentLineNumber() + ": expected '=', but got '" + token + "'"); break; } token = StripTokenQuotes(ReadToken()); if(!ReadSignedFloat(token, ref angleoffset)) { // Not numeric! General.ErrorLogger.Add(ErrorType.Error, "Error in " + sourcefilename + " at line " + GetCurrentLineNumber() + ": expected AngleOffset value, but got '" + token + "'"); } } else if(token == "scale") { SkipWhitespace(true); token = StripTokenQuotes(ReadToken()); if(token != "=") { General.ErrorLogger.Add(ErrorType.Error, "Error in " + sourcefilename + " at line " + GetCurrentLineNumber() + ": expected '=', but got '" + token + "'"); break; } token = StripTokenQuotes(ReadToken()); if(!ReadSignedFloat(token, ref scale)) { // Not numeric! General.ErrorLogger.Add(ErrorType.Error, "Error in " + sourcefilename + " at line " + GetCurrentLineNumber() + ": expected Scale value, but got '" + token + "'"); } } prevToken = token.ToUpperInvariant(); } } } else { prevToken = token.ToUpperInvariant(); } } } return entries.Count > 0; } } }