using System; using System.Collections.Generic; using System.Text; using Substrate.Nbt; namespace NBTExplorer.Model.Search { public enum NumericOperator { Equals, NotEquals, GreaterThan, LessThan, Any, } public enum StringOperator { Equals, NotEquals, Contains, NotContains, StartsWith, EndsWith, Any, } public enum WildcardOperator { Equals, NotEquals, Any, } public abstract class SearchRule { public static readonly Dictionary NumericOpStrings = new Dictionary() { { NumericOperator.Equals, "=" }, { NumericOperator.NotEquals, "!=" }, { NumericOperator.GreaterThan, ">" }, { NumericOperator.LessThan, "<" }, { NumericOperator.Any, "ANY" }, }; public static readonly Dictionary StringOpStrings = new Dictionary() { { StringOperator.Equals, "=" }, { StringOperator.NotEquals, "!=" }, { StringOperator.Contains, "Contains" }, { StringOperator.NotContains, "Does not Contain" }, { StringOperator.StartsWith, "Begins With" }, { StringOperator.EndsWith, "Ends With" }, { StringOperator.Any, "ANY" }, }; public static readonly Dictionary WildcardOpStrings = new Dictionary() { { WildcardOperator.Equals, "=" }, { WildcardOperator.NotEquals, "!=" }, { WildcardOperator.Any, "ANY" }, }; public abstract string NodeDisplay { get; } public virtual bool Matches (TagCompoundDataNode container, List matchedNodes) { return false; } public virtual bool CanAddRules { get { return false; } } protected static TagDataNode GetChild (TagCompoundDataNode container, string name) { foreach (var child in container.Nodes) { TagDataNode tagChild = child as TagDataNode; if (tagChild != null && tagChild.NodeName == name) return tagChild; } return null; } } public abstract class GroupRule : SearchRule { protected GroupRule () { Rules = new List(); } public List Rules { get; set; } public override bool CanAddRules { get { return true; } } } public class UnionRule : GroupRule { public override string NodeDisplay { get { return "Match Any"; } } public override bool Matches (TagCompoundDataNode container, List matchedNodes) { foreach (var rule in Rules) { if (rule.Matches(container, matchedNodes)) return true; } return false; } } public class IntersectRule : GroupRule { public override string NodeDisplay { get { return "Match All"; } } public override bool Matches (TagCompoundDataNode container, List matchedNodes) { foreach (var rule in Rules) { if (!rule.Matches(container, matchedNodes)) return false; } return true; } } public class RootRule : IntersectRule { public override string NodeDisplay { get { return "Search Rules"; } } } public abstract class TagRule : SearchRule { public TagType TagType { get; set; } public string Name { get; set; } protected T LookupTag (TagCompoundDataNode container, string name) where T : TagNode { return container.NamedTagContainer.GetTagNode(name) as T; } } public abstract class IntegralTagRule : TagRule where T : TagNode { public long Value { get; set; } public NumericOperator Operator { get; set; } public override string NodeDisplay { get { return string.Format("{0} {1} {2}", Name, NumericOpStrings[Operator], Operator != NumericOperator.Any ? Value.ToString() : ""); } } public override bool Matches (TagCompoundDataNode container, List matchedNodes) { TagDataNode childNode = GetChild(container, Name); T data = LookupTag(container, Name); if (data != null) { switch (Operator) { case NumericOperator.Equals: if (data.ToTagLong() != Value) return false; break; case NumericOperator.NotEquals: if (data.ToTagLong() == Value) return false; break; case NumericOperator.GreaterThan: if (data.ToTagLong() <= Value) return false; break; case NumericOperator.LessThan: if (data.ToTagLong() >= Value) return false; break; } } if (!matchedNodes.Contains(childNode)) matchedNodes.Add(childNode); return true; } } public class ByteTagRule : IntegralTagRule { } public class ShortTagRule : IntegralTagRule { } public class IntTagRule : IntegralTagRule { } public class LongTagRule : IntegralTagRule { } public abstract class FloatTagRule : TagRule where T : TagNode { public double Value { get; set; } public NumericOperator Operator { get; set; } public override string NodeDisplay { get { return string.Format("{0} {1} {2}", Name, NumericOpStrings[Operator], Operator != NumericOperator.Any ? Value.ToString() : ""); } } public override bool Matches (TagCompoundDataNode container, List matchedNodes) { TagDataNode childNode = GetChild(container, Name); T data = LookupTag(container, Name); if (data != null) { switch (Operator) { case NumericOperator.Equals: if (data.ToTagDouble() != Value) return false; break; case NumericOperator.NotEquals: if (data.ToTagDouble() == Value) return false; break; case NumericOperator.GreaterThan: if (data.ToTagDouble() <= Value) return false; break; case NumericOperator.LessThan: if (data.ToTagDouble() >= Value) return false; break; } } if (!matchedNodes.Contains(childNode)) matchedNodes.Add(childNode); return true; } } public class FloatTagRule : FloatTagRule { } public class DoubleTagRule : FloatTagRule { } public class StringTagRule : TagRule { public string Value { get; set; } public StringOperator Operator { get; set; } public override string NodeDisplay { get { return string.Format("{0} {1} {2}", Name, StringOpStrings[Operator], Operator != StringOperator.Any ? '"' + Value + '"' : ""); } } public override bool Matches (TagCompoundDataNode container, List matchedNodes) { TagDataNode childNode = GetChild(container, Name); TagNodeString data = LookupTag(container, Name); if (data != null) { switch (Operator) { case StringOperator.Equals: if (data.ToTagString().Data != Value) return false; break; case StringOperator.NotEquals: if (data.ToTagString().Data == Value) return false; break; case StringOperator.Contains: if (!data.ToTagString().Data.Contains(Value)) return false; break; case StringOperator.NotContains: if (data.ToTagString().Data.Contains(Value)) return false; break; case StringOperator.StartsWith: if (!data.ToTagString().Data.StartsWith(Value)) return false; break; case StringOperator.EndsWith: if (!data.ToTagString().Data.EndsWith(Value)) return false; break; } } if (!matchedNodes.Contains(childNode)) matchedNodes.Add(childNode); return true; } } public class WildcardRule : SearchRule { public string Name { get; set; } public string Value { get; set; } public WildcardOperator Operator { get; set; } public override string NodeDisplay { get { return string.Format("{0} {1} {2}", Name, WildcardOpStrings[Operator], Operator != WildcardOperator.Any ? Value : ""); } } public override bool Matches (TagCompoundDataNode container, List matchedNodes) { TagDataNode childNode = GetChild(container, Name); TagNode tag = container.NamedTagContainer.GetTagNode(Name); if (tag == null) return false; try { switch (tag.GetTagType()) { case TagType.TAG_BYTE: case TagType.TAG_INT: case TagType.TAG_LONG: case TagType.TAG_SHORT: switch (Operator) { case WildcardOperator.Equals: if (long.Parse(Value) != tag.ToTagLong()) return false; break; case WildcardOperator.NotEquals: if (long.Parse(Value) == tag.ToTagLong()) return false; break; } if (!matchedNodes.Contains(childNode)) matchedNodes.Add(childNode); return true; case TagType.TAG_FLOAT: case TagType.TAG_DOUBLE: switch (Operator) { case WildcardOperator.Equals: if (double.Parse(Value) != tag.ToTagDouble()) return false; break; case WildcardOperator.NotEquals: if (double.Parse(Value) == tag.ToTagDouble()) return false; break; } if (!matchedNodes.Contains(childNode)) matchedNodes.Add(childNode); return true; case TagType.TAG_STRING: switch (Operator) { case WildcardOperator.Equals: if (Value != tag.ToTagString().Data) return false; break; case WildcardOperator.NotEquals: if (Value == tag.ToTagString().Data) return false; break; } if (!matchedNodes.Contains(childNode)) matchedNodes.Add(childNode); return true; } } catch { } return false; } } }