//------------------------------------------------------------------------------
//
// Copyright (c) Microsoft Corporation. All rights reserved.
//
// [....]
//------------------------------------------------------------------------------
using System.Diagnostics;
using System.IO;
namespace System.Xml.Xsl {
///
/// Cardinality of part of XmlQueryType
/// struct is being used because enum doesn't allow members
///
internal struct XmlQueryCardinality {
private int value;
#region ctor
///
/// Private constructor
///
private XmlQueryCardinality(int value) {
Debug.Assert(0x00 <= value && value <= 0x07);
this.value = value;
}
#endregion
#region enum
///
/// exactly zero (empty)
///
public static XmlQueryCardinality None {
get { return new XmlQueryCardinality(0x00); }
}
///
/// exactly zero (empty)
///
public static XmlQueryCardinality Zero {
get { return new XmlQueryCardinality(0x01); }
}
///
/// exactly one
///
public static XmlQueryCardinality One {
get { return new XmlQueryCardinality(0x02); }
}
///
/// zero or one (not more)
///
public static XmlQueryCardinality ZeroOrOne {
get { return new XmlQueryCardinality(0x03); }
}
///
/// strictly more than one
///
public static XmlQueryCardinality More {
get { return new XmlQueryCardinality(0x04); }
}
///
/// not one (strictly zero or strictly more)
///
public static XmlQueryCardinality NotOne {
get { return new XmlQueryCardinality(0x05); }
}
///
/// one or more (not empty)
///
public static XmlQueryCardinality OneOrMore {
get { return new XmlQueryCardinality(0x06); }
}
///
/// zero or more (any cardinality)
///
public static XmlQueryCardinality ZeroOrMore {
get { return new XmlQueryCardinality(0x07); }
}
#endregion
#region ==
///
/// Strongly-typed Equals that returns true if this type and "other" type are equivalent.
///
public bool Equals(XmlQueryCardinality other) {
return this.value == other.value;
}
///
/// Overload == operator to call Equals rather than do reference equality.
///
public static bool operator ==(XmlQueryCardinality left, XmlQueryCardinality right) {
return left.value == right.value;
}
///
/// Overload != operator to call Equals rather than do reference inequality.
///
public static bool operator !=(XmlQueryCardinality left, XmlQueryCardinality right) {
return left.value != right.value;
}
///
/// True if "other" is an XmlQueryCardinality, and this type is the exact same static type.
///
public override bool Equals(object other) {
if (other is XmlQueryCardinality) {
return Equals((XmlQueryCardinality)other);
}
return false;
}
///
/// Return hash code of this instance.
///
public override int GetHashCode() {
return value;
}
#endregion
#region algebra
///
/// Return union with other
///
public static XmlQueryCardinality operator |(XmlQueryCardinality left, XmlQueryCardinality right) {
return new XmlQueryCardinality(left.value | right.value);
}
///
/// Return intersection with other
///
public static XmlQueryCardinality operator &(XmlQueryCardinality left, XmlQueryCardinality right) {
return new XmlQueryCardinality(left.value & right.value);
}
///
/// Return this product other
///
public static XmlQueryCardinality operator *(XmlQueryCardinality left, XmlQueryCardinality right) {
return cardinalityProduct[left.value, right.value];
}
///
/// Return sum with other
///
public static XmlQueryCardinality operator +(XmlQueryCardinality left, XmlQueryCardinality right) {
return cardinalitySum[left.value, right.value];
}
#if NEVER
///
/// Returns true if this cardinality is guaranteed to be a subset of "other".
///
private bool IsSubset(XmlQueryCardinality other) {
return (this.value & ~other.value) == 0;
}
#endif
///
/// Returns true is left is subset of right.
///
public static bool operator <=(XmlQueryCardinality left, XmlQueryCardinality right) {
return (left.value & ~right.value) == 0;
}
///
/// Returns true is right is subset of left.
///
public static bool operator >=(XmlQueryCardinality left, XmlQueryCardinality right) {
return (right.value & ~left.value) == 0;
}
///
/// Compute the cardinality of a subset of a set of the given cardinality.
///
/// the cardinality of a set
/// the cardinality of a subset
public XmlQueryCardinality AtMost() {
// Fill downward to zero
return new XmlQueryCardinality(this.value | (this.value >> 1) | (this.value >> 2));
}
///
/// Returns true if every non-None subset of this cardinality is disjoint with "other" cardinality.
/// Here is the behavior for None, which is the inverse of the None behavior for IsSubset:
/// None op None = false
/// None op ~None = false
/// ~None op None = true
///
public bool NeverSubset(XmlQueryCardinality other) {
return this.value != 0 && (this.value & other.value) == 0;
}
///
/// Table of cardinality products.
///
private static readonly XmlQueryCardinality[,] cardinalityProduct = {
// None Zero One ZeroOrOne More NotOne OneOrMore ZeroOrMore
/* None */ { None, Zero, None , Zero , None , Zero , None , Zero },
/* Zero */ { Zero, Zero, Zero , Zero , Zero , Zero , Zero , Zero },
/* One */ { None, Zero, One , ZeroOrOne , More , NotOne, OneOrMore , ZeroOrMore },
/* ZeroOrOne */ { Zero, Zero, ZeroOrOne , ZeroOrOne , NotOne, NotOne, ZeroOrMore, ZeroOrMore },
/* More */ { None, Zero, More , NotOne , More , NotOne, More , NotOne },
/* NotOne */ { Zero, Zero, NotOne , NotOne , NotOne, NotOne, NotOne , NotOne },
/* OneOrMore */ { None, Zero, OneOrMore , ZeroOrMore, More , NotOne, OneOrMore , ZeroOrMore },
/* ZeroOrMore */ { Zero, Zero, ZeroOrMore, ZeroOrMore, NotOne, NotOne, ZeroOrMore, ZeroOrMore }
};
///
/// Table of cardinality sums.
///
private static readonly XmlQueryCardinality[,] cardinalitySum = {
// None Zero One ZeroOrOne More NotOne OneOrMore ZeroOrMore
/* None */ { None , Zero , One , ZeroOrOne , More, NotOne , OneOrMore, ZeroOrMore},
/* Zero */ { Zero , Zero , One , ZeroOrOne , More, NotOne , OneOrMore, ZeroOrMore},
/* One */ { One , One , More , OneOrMore , More, OneOrMore , More , OneOrMore },
/* ZeroOrOne */ { ZeroOrOne , ZeroOrOne , OneOrMore, ZeroOrMore, More, ZeroOrMore, OneOrMore, ZeroOrMore},
/* More */ { More , More , More , More , More, More , More , More },
/* NotOne */ { NotOne , NotOne , OneOrMore, ZeroOrMore, More, NotOne , OneOrMore, ZeroOrMore},
/* OneOrMore */ { OneOrMore , OneOrMore , More , OneOrMore , More, OneOrMore , More , OneOrMore },
/* ZeroOrMore */ { ZeroOrMore, ZeroOrMore, OneOrMore, ZeroOrMore, More, ZeroOrMore, OneOrMore, ZeroOrMore}
};
#endregion
#region Serialization
///
/// String representation.
///
private static readonly string[] toString = {
/* None */ "" ,
/* Zero */ "?" ,
/* One */ "" ,
/* ZeroOrOne */ "?" ,
/* More */ "+" ,
/* NotOne */ "*" ,
/* OneOrMore */ "+" ,
/* ZeroOrMore */ "*"
};
///
/// Serialization
///
private static readonly string[] serialized = {
/* None */ "None",
/* Zero */ "Zero",
/* One */ "One",
/* ZeroOrOne */ "ZeroOrOne",
/* More */ "More",
/* NotOne */ "NotOne",
/* OneOrMore */ "OneOrMore",
/* ZeroOrMore */ "ZeroOrMore"
};
///
/// Return the string representation of a cardinality, normalized to either ?, +, *, or "" (card 1).
///
public string ToString(string format) {
if (format == "S") {
return serialized[this.value];
}
else {
return ToString();
}
}
///
/// Return the string representation of a cardinality, normalized to either ?, +, *, or "" (card 1).
///
public override string ToString() {
return toString[this.value];
}
///
/// Deserialization
///
public XmlQueryCardinality(string s) {
this.value = 0x00;
for (int i = 0; i < serialized.Length; i++) {
if (s == serialized[i]) {
this.value = i;
break;
}
}
}
///
/// Serialize the object to BinaryWriter.
///
public void GetObjectData(BinaryWriter writer) {
writer.Write((byte)value);
}
///
/// Deserialize the object from BinaryReader.
///
public XmlQueryCardinality(BinaryReader reader) : this(reader.ReadByte()) {
}
#endregion
}
}