File: System\Xml\XPath\Internal\FunctionQuery.cs
Project: ndp\fx\src\Xml\System.Xml.csproj (System.Xml)
//------------------------------------------------------------------------------
// <copyright file="FunctionQuery.cs" company="Microsoft">
//     Copyright (c) Microsoft Corporation.  All rights reserved.
// </copyright>
// <owner current="true" primary="true">Microsoft</owner>
//------------------------------------------------------------------------------
 
namespace MS.Internal.Xml.XPath {
    using System;
    using System.Xml;
    using System.Xml.Xsl;
    using System.Xml.XPath;
    using System.Diagnostics;
    using System.Globalization;
    using System.Collections.Generic;
 
    internal sealed class FunctionQuery : ExtensionQuery {
        private IList<Query> args;
        private IXsltContextFunction function;
 
        public FunctionQuery(string prefix, string name, List<Query> args) : base(prefix, name) {
            this.args = args;
        }
        private FunctionQuery(FunctionQuery other) : base(other) {
            this.function = other.function;
            Query[] tmp = new Query[other.args.Count]; {                
                for (int i = 0; i < tmp.Length; i ++) {
                    tmp[i] = Clone(other.args[i]);
                }
                args = tmp;
            }
            this.args = tmp;
        }
 
        public override void SetXsltContext(XsltContext context) {
            if (context == null) {
                throw XPathException.Create(Res.Xp_NoContext);
            }
			if (this.xsltContext != context) {
                xsltContext = context;
                foreach (Query argument in args) {
                    argument.SetXsltContext(context);
                }
                XPathResultType[] argTypes = new XPathResultType[args.Count];
                for(int i = 0; i < args.Count; i ++) {
                    argTypes[i] = args[i].StaticType;
                }
                function = xsltContext.ResolveFunction(prefix, name, argTypes);
                // KB article allows to return null, see http://support.microsoft.com/?kbid=324462#6
                if (function == null) {
                    throw XPathException.Create(Res.Xp_UndefFunc, QName);
                }
            }
        }
 
        public override object Evaluate(XPathNodeIterator nodeIterator) {
			if (xsltContext == null) {
				throw XPathException.Create(Res.Xp_NoContext);
			}
 
            // calculate arguments:
            object[] argVals = new object[args.Count];
            for (int i = 0; i < args.Count; i ++) {
                argVals[i] = args[i].Evaluate(nodeIterator);
                if (argVals[i] is XPathNodeIterator) {// ForBack Compat. To protect our queries from users. bug#372077 & 20006123 
                    argVals[i] = new XPathSelectionIterator(nodeIterator.Current, args[i]);
                }
            }
            try {
                return ProcessResult(function.Invoke(xsltContext, argVals, nodeIterator.Current));
            } catch(Exception ex) {
                throw XPathException.Create(Res.Xp_FunctionFailed, QName, ex);
            }
        }
 
        public override XPathNavigator MatchNode(XPathNavigator navigator) {
            if (name != "key" && prefix.Length != 0) {
                throw XPathException.Create(Res.Xp_InvalidPattern);
            }
            this.Evaluate(new XPathSingletonIterator(navigator, /*moved:*/true));
            XPathNavigator nav = null;
            while ((nav = this.Advance()) != null) {
                if (nav.IsSamePosition(navigator)) {
                    return nav;
                }
            }
            return nav;
        }
 
        public override XPathResultType StaticType { get {
            XPathResultType result = function != null ? function.ReturnType : XPathResultType.Any;
			if (result == XPathResultType.Error) {
				// In v.1 we confused Error & Any so now for backward compatibility we should allow users to return any of them.
				result = XPathResultType.Any;
			}
            return result;
        } }
 
        public override XPathNodeIterator Clone() { return new FunctionQuery(this); }
 
        public override void PrintQuery(XmlWriter w) {
            w.WriteStartElement(this.GetType().Name);
            w.WriteAttributeString("name", prefix.Length != 0 ? prefix + ':' + name : name);
            foreach(Query arg in this.args) {
                arg.PrintQuery(w);
            }
            w.WriteEndElement();
        }
    }
}