You've already forked linux-packaging-mono
							
							
		
			
	
	
		
			85 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
		
		
			
		
	
	
			85 lines
		
	
	
		
			3.1 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
|   | //------------------------------------------------------------------------------ | ||
|  | // <copyright file="DescendantOverDescendantQuery.cs" company="Microsoft"> | ||
|  | //     Copyright (c) Microsoft Corporation.  All rights reserved. | ||
|  | // </copyright> | ||
|  | // <owner current="true" primary="true">[....]</owner> | ||
|  | //------------------------------------------------------------------------------ | ||
|  | 
 | ||
|  | namespace MS.Internal.Xml.XPath { | ||
|  |     using System; | ||
|  |     using System.Xml; | ||
|  |     using System.Xml.XPath; | ||
|  |     using System.Diagnostics; | ||
|  | 
 | ||
|  |     // DescendantOverDescendantQuery: for each input it looks for the topmost descendents that matches to ns:name | ||
|  |     // This is posible when query which has this query as its input (child query) is descendent as well. | ||
|  |     // Work of this query doesn't depend on DOD of its input.  | ||
|  |     // It doesn't garate DOD of the output even when input is DOD.  | ||
|  |     internal sealed class DescendantOverDescendantQuery : DescendantBaseQuery { | ||
|  |         private int level = 0; | ||
|  | 
 | ||
|  |         public DescendantOverDescendantQuery(Query  qyParent, bool matchSelf, string name, string prefix, XPathNodeType typeTest, bool abbrAxis) :  | ||
|  |             base(qyParent, name, prefix, typeTest, matchSelf, abbrAxis) {} | ||
|  |         private DescendantOverDescendantQuery(DescendantOverDescendantQuery other) : base(other) { | ||
|  |             this.level = other.level; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override void Reset() { | ||
|  |             level = 0; | ||
|  |             base.Reset(); | ||
|  |         } | ||
|  | 
 | ||
|  |         public override XPathNavigator Advance() { | ||
|  |             while (true) { | ||
|  |                 if (level == 0) { | ||
|  |                     currentNode = qyInput.Advance(); | ||
|  |                     position = 0; | ||
|  |                     if (currentNode == null) { | ||
|  |                         return null; | ||
|  |                     } | ||
|  |                     if (matchSelf && matches(currentNode)) { | ||
|  |                         position = 1; | ||
|  |                         return currentNode; | ||
|  |                     } | ||
|  |                     currentNode = currentNode.Clone(); | ||
|  |                     if (! MoveToFirstChild()) { | ||
|  |                         continue; | ||
|  |                     } | ||
|  |                 } else { | ||
|  |                     if (!MoveUpUntillNext()) { | ||
|  |                         continue; | ||
|  |                     } | ||
|  |                 } | ||
|  |                 do { | ||
|  |                     if (matches(currentNode)) { | ||
|  |                         position++; | ||
|  |                         return currentNode; | ||
|  |                     } | ||
|  |                 } while (MoveToFirstChild()); | ||
|  |             } | ||
|  |         } | ||
|  | 
 | ||
|  |         private bool MoveToFirstChild() { | ||
|  |             if (currentNode.MoveToFirstChild()) { | ||
|  |                 level++; | ||
|  |                 return true; | ||
|  |             } | ||
|  |             return false; | ||
|  |         } | ||
|  | 
 | ||
|  |         private bool MoveUpUntillNext() { // move up untill we can move next | ||
|  |             while (! currentNode.MoveToNext()) { | ||
|  |                 -- level; | ||
|  |                 if (level == 0) { | ||
|  |                     return false; | ||
|  |                 } | ||
|  |                 bool result = currentNode.MoveToParent(); | ||
|  |                 Debug.Assert(result, "Algorithm error, We always should be able to move up if level > 0"); | ||
|  |             } | ||
|  |             return true; | ||
|  |         } | ||
|  | 
 | ||
|  |         public override XPathNodeIterator Clone() { return new DescendantOverDescendantQuery(this); } | ||
|  |     } | ||
|  | } |