mirror of
https://git.do.srb2.org/STJr/UltimateZoneBuilder.git
synced 2024-11-24 12:51:30 +00:00
15b2adfe30
Fixed, Texture Browser Form: well, I broke "Tab" key functionality again (in previous commit)... Maintenance: changed curly braces style to match DB2 one (hopefully not breaking anything in the process...). Maintenance: changed private method names casing to match DB2 one.
308 lines
9.5 KiB
C#
308 lines
9.5 KiB
C#
using System.Drawing;
|
|
using System.Globalization;
|
|
using System.IO;
|
|
using SlimDX;
|
|
using CodeImp.DoomBuilder.ZDoom;
|
|
using CodeImp.DoomBuilder.GZBuilder.Data;
|
|
|
|
namespace CodeImp.DoomBuilder.GZBuilder.GZDoom {
|
|
public sealed class MapinfoParser : ZDTextParser {
|
|
|
|
private MapInfo mapInfo;
|
|
public MapInfo MapInfo { get { return mapInfo; } }
|
|
|
|
public bool Parse(Stream stream, string sourcefilename, string mapName)
|
|
{
|
|
base.Parse(stream, sourcefilename);
|
|
mapName = mapName.ToLowerInvariant();
|
|
mapInfo = new MapInfo();
|
|
|
|
while (SkipWhitespace(true))
|
|
{
|
|
string token = ReadToken();
|
|
if (token != null)
|
|
{
|
|
token = token.ToLowerInvariant();
|
|
if (ParseBlock(token, mapName)) break;
|
|
}
|
|
}
|
|
|
|
//check values
|
|
if (mapInfo.FadeColor.Red > 0 || mapInfo.FadeColor.Green > 0 || mapInfo.FadeColor.Blue > 0)
|
|
mapInfo.HasFadeColor = true;
|
|
|
|
if (mapInfo.OutsideFogColor.Red > 0 || mapInfo.OutsideFogColor.Green > 0 || mapInfo.OutsideFogColor.Blue > 0)
|
|
mapInfo.HasOutsideFogColor = true;
|
|
|
|
//Cannot fail here
|
|
return true;
|
|
}
|
|
|
|
//returns true if parsing is finished
|
|
private bool ParseBlock(string token, string mapName)
|
|
{
|
|
string curBlockName;
|
|
if (token == "map" || token == "defaultmap" || token == "adddefaultmap")
|
|
{
|
|
curBlockName = token;
|
|
if (token == "map") //check map name
|
|
{
|
|
//get map name
|
|
SkipWhitespace(true);
|
|
token = ReadToken().ToLowerInvariant();
|
|
|
|
if (token != mapName) return false; //not finished, search for next "map", "defaultmap" or "adddefaultmap" block
|
|
}
|
|
else if (token == "defaultmap")
|
|
{
|
|
//reset MapInfo
|
|
mapInfo = new MapInfo();
|
|
}
|
|
|
|
//search for required keys
|
|
while (SkipWhitespace(true))
|
|
{
|
|
token = ReadToken().ToLowerInvariant();
|
|
//sky1 or sky2
|
|
if (token == "sky1" || token == "sky2")
|
|
{
|
|
string skyType = token;
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken()).ToLowerInvariant();
|
|
|
|
//new format
|
|
if (token == "=")
|
|
{
|
|
SkipWhitespace(true);
|
|
|
|
//should be sky texture name
|
|
token = StripTokenQuotes(ReadToken());
|
|
bool gotComma = (token.IndexOf(",") != -1);
|
|
if (gotComma) token = token.Replace(",", "");
|
|
string skyTexture = StripTokenQuotes(token).ToLowerInvariant();
|
|
|
|
if (!string.IsNullOrEmpty(skyTexture))
|
|
{
|
|
if (skyType == "sky1")
|
|
mapInfo.Sky1 = skyTexture;
|
|
else
|
|
mapInfo.Sky2 = skyTexture;
|
|
|
|
//check if we have scrollspeed
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken());
|
|
|
|
if (!gotComma && token == ",")
|
|
{
|
|
gotComma = true;
|
|
SkipWhitespace(true);
|
|
token = ReadToken();
|
|
}
|
|
|
|
if (gotComma)
|
|
{
|
|
float scrollSpeed = 0;
|
|
if (!ReadSignedFloat(token, ref scrollSpeed))
|
|
{
|
|
// Not numeric!
|
|
General.ErrorLogger.Add(ErrorType.Warning, "Unexpected token found in '" + sourcename + "' at line " + GetCurrentLineNumber() + ": expected " + skyType + " scroll speed value, but got '" + token + "'");
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
continue;
|
|
}
|
|
|
|
if (skyType == "sky1")
|
|
mapInfo.Sky1ScrollSpeed = scrollSpeed;
|
|
else
|
|
mapInfo.Sky2ScrollSpeed = scrollSpeed;
|
|
}
|
|
else
|
|
{
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
}
|
|
}
|
|
else
|
|
{
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
General.ErrorLogger.Add(ErrorType.Error, "Unexpected token found in '" + sourcename + "' at line " + GetCurrentLineNumber() + ": expected " + skyType + " texture name.");
|
|
}
|
|
}
|
|
//old format
|
|
else
|
|
{
|
|
//token should be sky1/2 name
|
|
if (!string.IsNullOrEmpty(token))
|
|
{
|
|
if (skyType == "sky1")
|
|
mapInfo.Sky1 = token;
|
|
else
|
|
mapInfo.Sky2 = token;
|
|
|
|
//try to read scroll speed
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken());
|
|
|
|
float scrollSpeed = 0;
|
|
if (!ReadSignedFloat(token, ref scrollSpeed))
|
|
{
|
|
// Not numeric!
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
continue;
|
|
}
|
|
|
|
if (skyType == "sky1")
|
|
mapInfo.Sky1ScrollSpeed = scrollSpeed;
|
|
else
|
|
mapInfo.Sky2ScrollSpeed = scrollSpeed;
|
|
|
|
}
|
|
else
|
|
{
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
General.ErrorLogger.Add(ErrorType.Error, "Unexpected token found in '" + sourcename + "' at line " + GetCurrentLineNumber() + ": expected " + skyType + " texture name.");
|
|
}
|
|
}
|
|
}
|
|
//fade or outsidefog
|
|
else if (token == "fade" || token == "outsidefog")
|
|
{
|
|
string fadeType = token;
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken()).ToLowerInvariant();
|
|
|
|
//new format?
|
|
if (token == "=")
|
|
{
|
|
SkipWhitespace(true);
|
|
token = ReadToken();
|
|
}
|
|
|
|
//get the color value
|
|
string colorVal = StripTokenQuotes(token).ToLowerInvariant().Replace(" ", "");
|
|
if(!string.IsNullOrEmpty(colorVal))
|
|
{
|
|
Color4 color = new Color4();
|
|
//try to get the color...
|
|
if(GetColor(colorVal, ref color))
|
|
{
|
|
if(fadeType == "fade")
|
|
mapInfo.FadeColor = color;
|
|
else
|
|
mapInfo.OutsideFogColor = color;
|
|
}
|
|
else //...or not
|
|
{
|
|
General.ErrorLogger.Add(ErrorType.Error, "Failed to parse " + fadeType + " value from string '" + colorVal + "' in '" + sourcename + "' at line " + GetCurrentLineNumber());
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
}
|
|
}
|
|
else
|
|
{
|
|
General.ErrorLogger.Add(ErrorType.Error, "Unexpected token found in '" + sourcename + "' at line " + GetCurrentLineNumber() + ": expected " + fadeType + " color value.");
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
}
|
|
}
|
|
//vertwallshade or horizwallshade
|
|
else if (token == "vertwallshade" || token == "horizwallshade")
|
|
{
|
|
string shadeType = token;
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken());
|
|
|
|
//new format
|
|
if(token == "=")
|
|
{
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken());
|
|
}
|
|
|
|
int val = 0;
|
|
if(!ReadSignedInt(token, ref val))
|
|
{
|
|
// Not numeric!
|
|
General.ErrorLogger.Add(ErrorType.Error, "Unexpected token found in '" + sourcename + "' at line " + GetCurrentLineNumber() + ": expected " + shadeType + " value, but got '" + token + "'");
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
continue;
|
|
}
|
|
|
|
if(shadeType == "vertwallshade")
|
|
mapInfo.VertWallShade = General.Clamp(val, -255, 255);
|
|
else
|
|
mapInfo.HorizWallShade = General.Clamp(val, -255, 255);
|
|
}
|
|
//fogdensity or outsidefogdensity
|
|
else if(token == "fogdensity" || token == "outsidefogdensity")
|
|
{
|
|
string densityType = token;
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken());
|
|
|
|
//new format
|
|
if(token == "=")
|
|
{
|
|
SkipWhitespace(true);
|
|
token = StripTokenQuotes(ReadToken());
|
|
}
|
|
|
|
int val;
|
|
if(!int.TryParse(token, NumberStyles.Integer, CultureInfo.InvariantCulture, out val))
|
|
{
|
|
// Not numeric!
|
|
General.ErrorLogger.Add(ErrorType.Error, "Unexpected token found in '" + sourcename + "' at line " + GetCurrentLineNumber() + ": expected " + densityType + " value, but got '" + token + "'");
|
|
datastream.Seek(-token.Length - 1, SeekOrigin.Current); //step back and try parsing this token again
|
|
continue;
|
|
}
|
|
|
|
if (densityType == "fogdensity")
|
|
mapInfo.FogDensity = (int)(1024 * (256.0f / val));
|
|
else
|
|
mapInfo.OutsideFogDensity = (int)(1024 * (256.0f / val));
|
|
}
|
|
//doublesky
|
|
else if (token == "doublesky")
|
|
{
|
|
mapInfo.DoubleSky = true;
|
|
|
|
}
|
|
//evenlighting
|
|
else if (token == "evenlighting")
|
|
{
|
|
mapInfo.EvenLighting = true;
|
|
}
|
|
//smoothlighting
|
|
else if (token == "smoothlighting")
|
|
{
|
|
mapInfo.SmoothLighting = true;
|
|
}
|
|
//block end
|
|
else if (token == "}")
|
|
{
|
|
return (curBlockName == "map" || ParseBlock(token, mapName));
|
|
}
|
|
}
|
|
}
|
|
return false;
|
|
}
|
|
|
|
private static bool GetColor(string name, ref Color4 color)
|
|
{
|
|
if (name == "black") return true;
|
|
|
|
//probably it's a hex color (like FFCC11)?
|
|
int ci;
|
|
if (int.TryParse(name, NumberStyles.HexNumber, CultureInfo.InvariantCulture, out ci))
|
|
{
|
|
color = new Color4(ci) {Alpha = 1.0f};
|
|
return true;
|
|
}
|
|
|
|
//probably it's a color name?
|
|
Color c = Color.FromName(name); //should be similar to C++ color name detection, I suppose
|
|
if (c.IsKnownColor)
|
|
{
|
|
color = new Color4(c);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
}
|
|
}
|