#region ================== Copyright (c) 2007 Pascal vd Heiden /* * Copyright (c) 2007 Pascal vd Heiden, www.codeimp.com * This program is released under GNU General Public License * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * */ #endregion #region ================== Namespaces using System; using System.Collections; using System.Collections.Generic; using System.IO; using CodeImp.DoomBuilder.IO; #endregion namespace CodeImp.DoomBuilder.Config { //mxd internal enum ScriptType { UNKNOWN = 0, ACS = 1, MODELDEF = 2, DECORATE = 3, } internal class ScriptConfiguration : IComparable { #region ================== Constants #endregion #region ================== Variables // Compiler settings private readonly CompilerInfo compiler; private readonly string parameters; private readonly string resultlump; // Editor settings private readonly string description; private readonly int codepage; private readonly string[] extensions; private readonly bool casesensitive; private readonly int insertcase; private readonly int lexer; private readonly string keywordhelp; private readonly string functionopen; private readonly string functionclose; private readonly string argumentdelimiter; private readonly string terminator; private readonly string functionregex; private readonly ScriptType scripttype; //mxd // Collections private readonly Dictionary keywords; private readonly Dictionary lowerkeywords; private readonly List constants; private readonly Dictionary lowerconstants; private readonly Dictionary snippets; //mxd #endregion #region ================== Properties // Compiler settings public CompilerInfo Compiler { get { return compiler; } } public string Parameters { get { return parameters; } } public string ResultLump { get { return resultlump; } } // Editor settings public string Description { get { return description; } } public int CodePage { get { return codepage; } } public string[] Extensions { get { return extensions; } } public bool CaseSensitive { get { return casesensitive; } } public int InsertCase { get { return insertcase; } } public int Lexer { get { return lexer; } } public string KeywordHelp { get { return keywordhelp; } } public string FunctionOpen { get { return functionopen; } } public string FunctionClose { get { return functionclose; } } public string ArgumentDelimiter { get { return argumentdelimiter; } } public string Terminator { get { return terminator; } } public string FunctionRegEx { get { return functionregex; } } public ScriptType ScriptType { get { return scripttype; } } //mxd public Dictionary Snippets { get { return snippets; } } //mxd // Collections public ICollection Keywords { get { return keywords.Keys; } } public ICollection Constants { get { return constants; } } #endregion #region ================== Constructor / Disposer // This creates the default script configuration // that is used for documents of unknown type internal ScriptConfiguration() { // Initialize this.keywords = new Dictionary(StringComparer.Ordinal); this.constants = new List(); this.lowerkeywords = new Dictionary(StringComparer.Ordinal); this.lowerconstants = new Dictionary(StringComparer.Ordinal); // Settings lexer = 1; casesensitive = false; codepage = 65001; parameters = ""; resultlump = ""; insertcase = 0; keywordhelp = ""; functionopen = ""; functionclose = ""; argumentdelimiter = ""; terminator = ""; functionregex = ""; description = "Plain text"; scripttype = ScriptType.UNKNOWN; //mxd extensions = new[] { "txt" }; snippets = new Dictionary(StringComparer.Ordinal); //mxd } // Constructor internal ScriptConfiguration(Configuration cfg) { // Initialize this.keywords = new Dictionary(StringComparer.Ordinal); this.constants = new List(); this.lowerkeywords = new Dictionary(StringComparer.Ordinal); this.lowerconstants = new Dictionary(StringComparer.Ordinal); this.snippets = new Dictionary(StringComparer.Ordinal); //mxd // Read settings description = cfg.ReadSetting("description", "Untitled script"); codepage = cfg.ReadSetting("codepage", 0); string extensionsstring = cfg.ReadSetting("extensions", ""); string compilername = cfg.ReadSetting("compiler", ""); parameters = cfg.ReadSetting("parameters", ""); resultlump = cfg.ReadSetting("resultlump", ""); casesensitive = cfg.ReadSetting("casesensitive", true); insertcase = cfg.ReadSetting("insertcase", 0); lexer = cfg.ReadSetting("lexer", 0); keywordhelp = cfg.ReadSetting("keywordhelp", ""); functionopen = cfg.ReadSetting("functionopen", ""); functionclose = cfg.ReadSetting("functionclose", ""); argumentdelimiter = cfg.ReadSetting("argumentdelimiter", ""); terminator = cfg.ReadSetting("terminator", ""); functionregex = cfg.ReadSetting("functionregex", ""); scripttype = (ScriptType)cfg.ReadSetting("scripttype", (int)ScriptType.UNKNOWN); //mxd // Make extensions array extensions = extensionsstring.Split(','); for(int i = 0; i < extensions.Length; i++) extensions[i] = extensions[i].Trim(); // Load keywords IDictionary dic = cfg.ReadSetting("keywords", new Hashtable()); foreach(DictionaryEntry de in dic) { keywords.Add(de.Key.ToString(), de.Value.ToString()); lowerkeywords.Add(de.Key.ToString().ToLowerInvariant(), de.Key.ToString()); } // Load constants dic = cfg.ReadSetting("constants", new Hashtable()); foreach(DictionaryEntry de in dic) { constants.Add(de.Key.ToString()); lowerconstants.Add(de.Key.ToString().ToLowerInvariant(), de.Key.ToString()); } // Compiler specified? if(compilername.Length > 0) { // Find compiler foreach(CompilerInfo c in General.Compilers) { // Compiler name matches? if(c.Name == compilername) { // Apply compiler this.compiler = c; break; } } // No compiler found? if(this.compiler == null) throw new Exception("No such compiler defined: '" + compilername + "'"); } //mxd. Load Snippets string snippetsdir = cfg.ReadSetting("snippetsdir", ""); if(!string.IsNullOrEmpty(snippetsdir)) { string snippetspath = Path.Combine(General.SnippetsPath, snippetsdir); if(Directory.Exists(snippetspath)) { string[] files = Directory.GetFiles(snippetspath, "*.txt", SearchOption.TopDirectoryOnly); foreach(string file in files) { string name = Path.GetFileNameWithoutExtension(file); if(name.Contains(" ")) { General.ErrorLogger.Add(ErrorType.Warning, "Failed to load snippet '" + file + "' for '" + description + "' script configuration: snippet file name must not contain spaces!"); } else { string[] lines = File.ReadAllLines(file); if(lines.Length > 0) { snippets.Add(name, lines); } else { General.ErrorLogger.Add(ErrorType.Warning, "Failed to load snippet '" + file + "' for '" + description + "' script configuration: file is empty!"); } } } } } } #endregion #region ================== Methods // This returns the correct case for a keyword // Returns the same keyword as the input when it cannot be found public string GetKeywordCase(string keyword) { if(lowerkeywords.ContainsKey(keyword.ToLowerInvariant())) return lowerkeywords[keyword.ToLowerInvariant()]; else return keyword; } // This returns the correct case for a constant // Returns the same constant as the input when it cannot be found public string GetConstantCase(string constant) { if(lowerconstants.ContainsKey(constant.ToLowerInvariant())) return lowerconstants[constant.ToLowerInvariant()]; else return constant; } // This returns true when the given word is a keyword public bool IsKeyword(string keyword) { return lowerkeywords.ContainsKey(keyword.ToLowerInvariant()); } // This returns true when the given word is a contant public bool IsConstant(string constant) { return lowerconstants.ContainsKey(constant.ToLowerInvariant()); } // This returns the function definition for a keyword // Returns null when no function definition exists // NOTE: The keyword parameter is case-sensitive! public string GetFunctionDefinition(string keyword) { if(keywords.ContainsKey(keyword)) return keywords[keyword]; else return null; } // This sorts by description public int CompareTo(ScriptConfiguration other) { return string.Compare(this.description, other.description, true); } //mxd public override string ToString() { return description; } #endregion } }