Jo Shields 8b9b85e7f5 Imported Upstream version 3.10.0
Former-commit-id: 172c8e3c300b39d5785c7a3e8dfb08ebdbc1a99b
2014-10-04 11:27:48 +01:00

218 lines
5.7 KiB
C#

//
// XslTemplateContent.cs
//
// Authors:
// Ben Maurer (bmaurer@users.sourceforge.net)
// Atsushi Enomoto (ginga@kit.hi-ho.ne.jp)
//
// (C) 2003 Ben Maurer
// (C) 2003 Atsushi Enomoto
//
//
// 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.Xml;
using System.Xml.XPath;
using System.Xml.Xsl;
namespace Mono.Xml.Xsl.Operations
{
internal class XslTemplateContent : XslCompiledElementBase
{
ArrayList content = new ArrayList ();
bool hasStack;
int stackSize;
XPathNodeType parentType;
bool xslForEach;
bool isEmptyElement;
public XslTemplateContent (Compiler c,
XPathNodeType parentType, bool xslForEach)
: base (c)
{
this.parentType = parentType;
this.xslForEach = xslForEach;
Compile (c);
}
public XPathNodeType ParentType {
get { return parentType; }
}
public bool IsEmptyElement {
get { return isEmptyElement; }
}
protected override void Compile (Compiler c)
{
if (c.Debugger != null)
c.Debugger.DebugCompile (this.DebugInput);
hasStack = (c.CurrentVariableScope == null);
c.PushScope ();
do {
Debug.EnterNavigator (c);
XPathNavigator n = c.Input;
switch (n.NodeType) {
case XPathNodeType.Element:
switch (n.NamespaceURI) {
case XsltNamespace:
switch (n.LocalName) {
case "apply-imports":
content.Add (new XslApplyImports (c));
break;
case "apply-templates":
content.Add (new XslApplyTemplates (c));
break;
case "attribute":
if (ParentType == XPathNodeType.All
|| ParentType == XPathNodeType.Element)
content.Add (new XslAttribute (c));
break;
case "call-template":
content.Add (new XslCallTemplate (c));
break;
case "choose":
content.Add (new XslChoose (c));
break;
case "comment":
if (ParentType == XPathNodeType.All
|| ParentType == XPathNodeType.Element)
content.Add (new XslComment (c));
break;
case "copy":
content.Add (new XslCopy (c));
break;
case "copy-of":
content.Add (new XslCopyOf (c));
break;
case "element":
if (ParentType == XPathNodeType.All
|| ParentType == XPathNodeType.Element)
content.Add (new XslElement (c));
break;
case "fallback":
break;
case "for-each":
content.Add (new XslForEach (c));
break;
case "if":
content.Add (new XslIf (c));
break;
case "message":
content.Add (new XslMessage(c));
break;
case "number":
content.Add (new XslNumber(c));
break;
case "processing-instruction":
if (ParentType == XPathNodeType.All
|| ParentType == XPathNodeType.Element)
content.Add (new XslProcessingInstruction(c));
break;
case "text":
content.Add (new XslText(c, false));
break;
case "value-of":
content.Add (new XslValueOf(c));
break;
case "variable":
content.Add (new XslLocalVariable (c));
break;
case "sort":
if (xslForEach)
break;
throw new XsltCompileException ("'sort' element is not allowed here as a templete content", null, n);
default:
// TODO: handle fallback, like we should
// throw new XsltCompileException ("Did not recognize element " + n.Name, null, n);
content.Add (new XslNotSupportedOperation (c));
break;
}
break;
default:
if (!c.IsExtensionNamespace (n.NamespaceURI))
content.Add (new XslLiteralElement(c));
else {
if (n.MoveToFirstChild ()) {
do {
if (n.NamespaceURI == XsltNamespace && n.LocalName == "fallback")
content.Add (new XslFallback (c));
} while (n.MoveToNext ());
n.MoveToParent ();
}
}
break;
}
break;
case XPathNodeType.SignificantWhitespace:
content.Add (new XslText(c, true));
break;
case XPathNodeType.Text:
content.Add (new XslText(c, false));
break;
default:
break;
}
Debug.ExitNavigator (c);
} while (c.Input.MoveToNext ());
isEmptyElement = true;
foreach (var n in content) {
if (n is XslAttribute)
continue;
isEmptyElement = false;
break;
}
if (hasStack) {
stackSize = c.PopScope ().VariableHighTide;
hasStack = stackSize > 0;
} else
c.PopScope ();
}
public override void Evaluate (XslTransformProcessor p)
{
if (p.Debugger != null)
p.Debugger.DebugExecute (p, this.DebugInput);
if (hasStack)
p.PushStack (stackSize);
int len = content.Count;
for (int i = 0; i < len; i++)
((XslOperation) content [i]).Evaluate (p);
if (hasStack)
p.PopStack ();
}
}
}