| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171 |
- //
- // Mono.Xml.XPath.LocationPathPattern
- //
- // Author:
- // Ben Maurer ([email protected])
- //
- // (C) 2003 Ben Maurer
- //
- //
- // Permission is hereby granted, free of charge, to any person obtaining
- // a copy of this software and associated documentation files (the
- // "Software"), to deal in the Software without restriction, including
- // without limitation the rights to use, copy, modify, merge, publish,
- // distribute, sublicense, and/or sell copies of the Software, and to
- // permit persons to whom the Software is furnished to do so, subject to
- // the following conditions:
- //
- // The above copyright notice and this permission notice shall be
- // included in all copies or substantial portions of the Software.
- //
- // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
- // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
- // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
- // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
- // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
- // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
- // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
- //
- using System;
- using System.Collections;
- using System.IO;
- using System.Xml;
- using System.Xml.Schema;
- using System.Xml.XPath;
- using System.Xml.Xsl;
- namespace Mono.Xml.XPath {
- internal class LocationPathPattern : Pattern {
-
- LocationPathPattern patternPrevious;
- bool isAncestor;
- NodeTest nodeTest;
- ExprFilter filter;
- XPathNavigator previousNavigator;
-
- public LocationPathPattern (NodeTest nodeTest)
- {
- this.nodeTest = nodeTest;
- }
-
- public LocationPathPattern (ExprFilter filter)
- {
- this.filter = filter;
-
- while (! (filter.expr is NodeTest))
- filter = (ExprFilter)filter.expr;
-
- this.nodeTest = (NodeTest)filter.expr;
- }
- internal void SetPreviousPattern (Pattern prev, bool isAncestor)
- {
- LocationPathPattern toSet = LastPathPattern;
- toSet.patternPrevious = (LocationPathPattern)prev;
- toSet.isAncestor = isAncestor;
- }
-
- public override double DefaultPriority {
- get {
- if (patternPrevious == null && filter == null) {
- NodeNameTest t = nodeTest as NodeNameTest;
- if (t != null) {
- if (t.Name.Name == "*")
- return -.25;
- return 0;
- }
- return -.5;
- }
- return .5;
- }
- }
-
- public override bool Matches (XPathNavigator node, XsltContext ctx)
- {
- if (! nodeTest.Match (ctx, node))
- return false;
-
- if (nodeTest is NodeTypeTest) {
- // node () is different in xslt patterns
- if (((NodeTypeTest)nodeTest).type == XPathNodeType.All &&
- (node.NodeType == XPathNodeType.Root ||
- node.NodeType == XPathNodeType.Attribute)
- )
- return false;
- }
-
- if (filter == null && patternPrevious == null)
- return true;
-
- if (patternPrevious != null) {
- if (!isAncestor) {
- XPathNavigator parent = node.Clone ();
- parent.MoveToParent ();
- if (!patternPrevious.Matches (parent, ctx))
- return false;
- } else {
- XPathNavigator anc = node.Clone ();
- while (true) {
- if (!anc.MoveToParent ())
- return false;
-
- if (patternPrevious.Matches (anc, ctx))
- break;
- }
- }
- }
-
- if (filter == null)
- return true;
- // Optimization for non-positional predicate
- if (!filter.IsPositional && !(filter.expr is ExprFilter)) {
- return filter.pred.EvaluateBoolean (new NullIterator (node, ctx));
- }
- XPathNavigator p = null;
- if (previousNavigator == node) {
- p = previousNavigator;
- p.MoveTo (node);
- } else {
- p = node.Clone ();
- previousNavigator = p;
- }
- p.MoveToParent ();
- BaseIterator matches = filter.EvaluateNodeSet (new NullIterator (p, ctx));
-
- while (matches.MoveNext ()) {
- if (node.IsSamePosition (matches.Current))
- return true;
- }
-
- return false;
- }
-
- public override string ToString ()
- {
- string ret = "";
- if (patternPrevious != null) ret = patternPrevious.ToString () + (isAncestor ? "//" : "/");
- if (filter != null) ret += filter.ToString ();
- else ret += nodeTest.ToString ();
-
- return ret;
- }
-
- public LocationPathPattern LastPathPattern {
- get {
- LocationPathPattern ret = this;
-
- while (ret.patternPrevious != null)
- ret = ret.patternPrevious;
-
- return ret;
- }
- }
- }
- }
|