using System; using System.Collections.Generic; using System.Globalization; using System.Xml.Linq; namespace Barotrauma { class PropertyConditional { public enum ConditionType { PropertyValue, Name, SpeciesName, HasTag, HasStatusTag } public enum OperatorType { Equals, NotEquals, LessThan, LessThanEquals, GreaterThan, GreaterThanEquals } public readonly ConditionType Type; public readonly string PropertyName; public readonly OperatorType Operator; public readonly string Value; private readonly int cancelStatusEffect; public PropertyConditional(XAttribute attribute) { string attributeString = attribute.Value.ToString(); string atStr = attributeString; string[] splitString = atStr.Split(' '); if (splitString.Length > 0) { for (int i = 1; i < splitString.Length; i++) { atStr = splitString[i] + (i > 1 && i < splitString.Length ? " " : ""); } } //thanks xml for not letting me use < or > in attributes :( string op = splitString[0]; switch (op) { case "e": case "eq": case "equals": Operator = OperatorType.Equals; break; case "ne": case "neq": case "notequals": case "!": case "!e": case "!eq": case "!equals": Operator = OperatorType.NotEquals; break; case "gt": case "greaterthan": Operator = OperatorType.GreaterThan; break; case "lt": case "lessthan": Operator = OperatorType.LessThan; break; case "gte": case "gteq": case "greaterthanequals": Operator = OperatorType.GreaterThanEquals; break; case "lte": case "lteq": case "lessthanequals": Operator = OperatorType.LessThanEquals; break; default: if (op != "==" && op != "!=" && op != ">" && op != "<" && op != ">=" && op != "<=") //Didn't use escape strings or anything { atStr = attributeString; //We probably don't even have an operator } break; } foreach (XElement subElement in attribute.Parent.Elements()) { switch (subElement.Name.ToString().ToLowerInvariant()) { case "cancel": case "canceleffect": case "cancelstatuseffect": //This only works if there's a conditional checking for status effect tags. There is no way to cancel *all* status effects atm. cancelStatusEffect = 1; if (subElement.GetAttributeBool("all", false)) cancelStatusEffect = 2; break; } } if (!Enum.TryParse(attribute.Name.ToString(), true, out Type)) { PropertyName = attribute.Name.ToString(); Type = ConditionType.PropertyValue; } Value = atStr; } public bool Matches(ISerializableEntity target) { string valStr = Value.ToString(); switch (Type) { case ConditionType.PropertyValue: SerializableProperty property; if (target.SerializableProperties.TryGetValue(PropertyName.ToLowerInvariant(), out property)) { return Matches(property); } return false; case ConditionType.Name: return (Operator == OperatorType.Equals) == (target.Name == valStr); case ConditionType.HasTag: { string[] readTags = valStr.Split(','); int matches = 0; foreach (string tag in readTags) if (((Item)target).HasTag(tag)) matches++; //If operator is == then it needs to match everything, otherwise if its != there must be zero matches. return Operator == OperatorType.Equals ? matches >= readTags.Length : matches <= 0; } case ConditionType.HasStatusTag: List durations = StatusEffect.DurationList.FindAll(d => d.Targets.Contains(target)); List delays = DelayedEffect.DelayList.FindAll(d => d.Targets.Contains(target)); bool success = false; if (durations.Count > 0 || delays.Count > 0) { string[] readTags = valStr.Split(','); foreach (DurationListElement duration in durations) { int matches = 0; foreach (string tag in readTags) if (duration.Parent.HasTag(tag)) matches++; success = Operator == OperatorType.Equals ? matches >= readTags.Length : matches <= 0; if (cancelStatusEffect > 0 && success) StatusEffect.DurationList.Remove(duration); if (cancelStatusEffect != 2) //cancelStatusEffect 1 = only cancel once, cancelStatusEffect 2 = cancel all of matching tags return success; } foreach (DelayedListElement delay in delays) { int matches = 0; foreach (string tag in readTags) if (delay.Parent.HasTag(tag)) matches++; success = Operator == OperatorType.Equals ? matches >= readTags.Length : matches <= 0; if (cancelStatusEffect > 0 && success) DelayedEffect.DelayList.Remove(delay); if (cancelStatusEffect != 2) //ditto return success; } } return success; case ConditionType.SpeciesName: Character targetCharacter = target as Character; if (targetCharacter == null) return false; return (Operator == OperatorType.Equals) == (targetCharacter.SpeciesName == valStr); default: return false; } } private bool Matches(SerializableProperty property) { object propertyValue = property.GetValue(); if (propertyValue == null) { DebugConsole.ThrowError("Couldn't compare " + Value.ToString() + " (" + Value.GetType() + ") to property \"" + property.Name + "\" - property.GetValue() returns null!"); return false; } Type type = propertyValue.GetType(); float? floatValue = null; float? floatProperty = null; if (type == typeof(float) || type == typeof(int)) { float parsedFloat; if (Single.TryParse(Value, NumberStyles.Float, CultureInfo.InvariantCulture, out parsedFloat)) { floatValue = parsedFloat; } floatProperty = (float)propertyValue; } switch (Operator) { case OperatorType.Equals: if (type == typeof(bool)) { return ((bool)propertyValue) == (Value.ToLowerInvariant() == "true"); } else if (floatValue == null) { return propertyValue.ToString().Equals(Value); } else { return propertyValue.Equals(floatValue); } case OperatorType.NotEquals: if (type == typeof(bool)) { return ((bool)propertyValue) != (Value.ToLowerInvariant() == "true"); } else if (floatValue == null) { return !propertyValue.ToString().Equals(Value); } else { return !propertyValue.Equals(floatValue); } case OperatorType.GreaterThan: if (floatValue == null) { DebugConsole.ThrowError("Couldn't compare " + Value.ToString() + " (" + Value.GetType() + ") to property \"" + property.Name + "\" (" + type + ")! " + "Make sure the type of the value set in the config files matches the type of the property."); } else if (floatProperty > floatValue) return true; break; case OperatorType.LessThan: if (floatValue == null) { DebugConsole.ThrowError("Couldn't compare " + Value.ToString() + " (" + Value.GetType() + ") to property \"" + property.Name + "\" (" + type + ")! " + "Make sure the type of the value set in the config files matches the type of the property."); } else if (floatProperty < floatValue) return true; break; case OperatorType.GreaterThanEquals: if (floatValue == null) { DebugConsole.ThrowError("Couldn't compare " + Value.ToString() + " (" + Value.GetType() + ") to property \"" + property.Name + "\" (" + type + ")! " + "Make sure the type of the value set in the config files matches the type of the property."); } else if (floatProperty >= floatValue) return true; break; case OperatorType.LessThanEquals: if (floatValue == null) { DebugConsole.ThrowError("Couldn't compare " + Value.ToString() + " (" + Value.GetType() + ") to property \"" + property.Name + "\" (" + type + ")! " + "Make sure the type of the value set in the config files matches the type of the property."); } else if (floatProperty <= floatValue) return true; break; } return false; } } }