//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // // Microsoft //------------------------------------------------------------------------------ namespace System.Xml.Xsl.XsltOld { using Res = System.Xml.Utils.Res; using System; using System.Diagnostics; using System.Collections; using System.Xml; using System.Xml.XPath; internal class ApplyTemplatesAction : ContainerAction { private const int ProcessedChildren = 2; private const int ProcessNextNode = 3; private const int PositionAdvanced = 4; private const int TemplateProcessed = 5; private int selectKey = Compiler.InvalidQueryKey; private XmlQualifiedName mode; // // // // // private static ApplyTemplatesAction s_BuiltInRule = new ApplyTemplatesAction(); internal static ApplyTemplatesAction BuiltInRule() { Debug.Assert(s_BuiltInRule != null); return s_BuiltInRule; } internal static ApplyTemplatesAction BuiltInRule(XmlQualifiedName mode) { return(mode == null || mode.IsEmpty) ? BuiltInRule() : new ApplyTemplatesAction(mode); } internal ApplyTemplatesAction() {} private ApplyTemplatesAction(XmlQualifiedName mode) { Debug.Assert(mode != null); this.mode = mode; } internal override void Compile(Compiler compiler) { CompileAttributes(compiler); CompileContent(compiler); } internal override bool CompileAttribute(Compiler compiler) { string name = compiler.Input.LocalName; string value = compiler.Input.Value; if (Ref.Equal(name, compiler.Atoms.Select )) { this.selectKey = compiler.AddQuery(value); } else if (Ref.Equal(name, compiler.Atoms.Mode )) { Debug.Assert(this.mode == null); if (compiler.AllowBuiltInMode && value == "*") { this.mode = Compiler.BuiltInMode; } else { this.mode = compiler.CreateXPathQName(value); } } else { return false; } return true; } private void CompileContent(Compiler compiler) { NavigatorInput input = compiler.Input; if (compiler.Recurse()) { do { switch (input.NodeType) { case XPathNodeType.Element: compiler.PushNamespaceScope(); string nspace = input.NamespaceURI; string name = input.LocalName; if (Ref.Equal(nspace, input.Atoms.UriXsl)) { if (Ref.Equal(name, input.Atoms.Sort)) { AddAction(compiler.CreateSortAction()); } else if (Ref.Equal(name, input.Atoms.WithParam)) { WithParamAction par = compiler.CreateWithParamAction(); CheckDuplicateParams(par.Name); AddAction(par); } else { throw compiler.UnexpectedKeyword(); } } else { throw compiler.UnexpectedKeyword(); } compiler.PopScope(); break; case XPathNodeType.Comment: case XPathNodeType.ProcessingInstruction: case XPathNodeType.Whitespace: case XPathNodeType.SignificantWhitespace: break; default: throw XsltException.Create(Res.Xslt_InvalidContents, "apply-templates"); } } while (compiler.Advance()); compiler.ToParent(); } } internal override void Execute(Processor processor, ActionFrame frame) { Debug.Assert(processor != null && frame != null); switch (frame.State) { case Initialized: processor.ResetParams(); processor.InitSortArray(); if (this.containedActions != null && this.containedActions.Count > 0) { processor.PushActionFrame(frame); frame.State = ProcessedChildren; break; } goto case ProcessedChildren; case ProcessedChildren: if (this.selectKey == Compiler.InvalidQueryKey) { if (! frame.Node.HasChildren) { frame.Finished(); break; } frame.InitNewNodeSet(frame.Node.SelectChildren(XPathNodeType.All)); } else { frame.InitNewNodeSet(processor.StartQuery(frame.NodeSet, this.selectKey)); } if (processor.SortArray.Count != 0) { frame.SortNewNodeSet(processor, processor.SortArray); } frame.State = ProcessNextNode; goto case ProcessNextNode; case ProcessNextNode: Debug.Assert(frame.State == ProcessNextNode); Debug.Assert(frame.NewNodeSet != null); if (frame.NewNextNode(processor)) { frame.State = PositionAdvanced; goto case PositionAdvanced; } else { frame.Finished(); break; } case PositionAdvanced: Debug.Assert(frame.State == PositionAdvanced); processor.PushTemplateLookup(frame.NewNodeSet, this.mode, /*importsOf:*/null); frame.State = TemplateProcessed; break; case TemplateProcessed: frame.State = ProcessNextNode; goto case ProcessNextNode; default: Debug.Fail("Invalid ApplyTemplatesAction execution state"); break; } } } }