536cd135cc
Former-commit-id: 5624ac747d633e885131e8349322922b6a59baaa
184 lines
6.7 KiB
C#
184 lines
6.7 KiB
C#
//------------------------------------------------------------------------------
|
|
// <copyright file="ApplyTemplatesAction.cs" company="Microsoft">
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// </copyright>
|
|
// <owner current="true" primary="true">Microsoft</owner>
|
|
//------------------------------------------------------------------------------
|
|
|
|
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;
|
|
|
|
//
|
|
// <xsl:template match="*|/" [mode="?"]>
|
|
// <xsl:apply-templates [mode="?"]/>
|
|
// </xsl:template>
|
|
//
|
|
|
|
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;
|
|
}
|
|
}
|
|
}
|
|
}
|