2005-03-07 Atsushi Enomoto <atsushi@ximian.com>
[mono.git] / mcs / class / System.XML / Mono.Xml.Xsl.Operations / XslLiteralElement.cs
1 //
2 // XslLiteralElement.cs
3 //
4 // Authors:
5 //      Ben Maurer (bmaurer@users.sourceforge.net)
6 //      Atsushi Enomoto (ginga@kit.hi-ho.ne.jp)
7 //      
8 // (C) 2003 Ben Maurer
9 // (C) 2003 Atsushi Enomoto
10 //
11
12 //
13 // Permission is hereby granted, free of charge, to any person obtaining
14 // a copy of this software and associated documentation files (the
15 // "Software"), to deal in the Software without restriction, including
16 // without limitation the rights to use, copy, modify, merge, publish,
17 // distribute, sublicense, and/or sell copies of the Software, and to
18 // permit persons to whom the Software is furnished to do so, subject to
19 // the following conditions:
20 // 
21 // The above copyright notice and this permission notice shall be
22 // included in all copies or substantial portions of the Software.
23 // 
24 // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
25 // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
26 // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
27 // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
28 // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
29 // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
30 // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
31 //
32
33 using System;
34 using System.Collections;
35 using System.Xml;
36 using System.Xml.XPath;
37 using System.Xml.Xsl;
38
39 namespace Mono.Xml.Xsl.Operations {     
40         internal class XslLiteralElement : XslCompiledElement {
41                 XslOperation children;
42                 string localname, prefix, nsUri;
43                 bool isEmptyElement;
44                 ArrayList attrs;
45                 XmlQualifiedName [] useAttributeSets;
46                 Hashtable nsDecls;
47
48                 public XslLiteralElement (Compiler c) : base (c) {}
49                         
50                 class XslLiteralAttribute {
51                         string localname, prefix, nsUri;
52                         XslAvt val;
53                         
54                         public XslLiteralAttribute (Compiler c)
55                         {
56                                 this.prefix = c.Input.Prefix;
57                                 this.nsUri = c.Input.NamespaceURI;
58                                 this.localname = c.Input.LocalName;
59                                 this.val = new XslAvt (c.Input.Value, c);
60                         }
61                         
62                         public void Evaluate (XslTransformProcessor p)
63                         {
64                                 //FIXME: fix attribute prefixes according to aliases
65                                 p.Out.WriteAttributeString (prefix, localname, nsUri, val.Evaluate (p));
66                         }
67                 }
68                 
69                 protected override void Compile (Compiler c)
70                 {
71                         prefix = c.Input.Prefix;
72                         string alias = c.CurrentStylesheet.GetActualPrefix (prefix);
73                         if (alias != prefix) {
74                                 prefix = alias;
75                                 nsUri = c.Input.GetNamespace (alias);
76                         }
77                         else
78                                 nsUri = c.Input.NamespaceURI;
79
80                         this.localname = c.Input.LocalName;
81                         this.useAttributeSets = c.ParseQNameListAttribute ("use-attribute-sets", XsltNamespace);
82                         this.nsDecls = c.GetNamespacesToCopy ();
83                         if (nsDecls.Count == 0) nsDecls = null;
84                         this.isEmptyElement = c.Input.IsEmptyElement;
85
86                         if (c.Input.MoveToFirstAttribute ())
87                         {
88                                 attrs = new ArrayList ();
89                                 do {
90                                         if (c.Input.NamespaceURI == XsltNamespace)
91                                                 continue; //already handled
92                                         attrs.Add (new XslLiteralAttribute (c));
93                                 } while (c.Input.MoveToNextAttribute());
94                                 c.Input.MoveToParent ();
95                         }
96                         
97                         if (!c.Input.MoveToFirstChild ()) return;
98                         children = c.CompileTemplateContent ();
99                         c.Input.MoveToParent ();
100                 }
101
102                 public override void Evaluate (XslTransformProcessor p)
103                 {
104                         // Since namespace-alias might be determined after compilation
105                         // of import-ing stylesheets, this must be determined later.
106                         bool isCData = p.InsideCDataElement;
107                         p.PushElementState (prefix, localname, nsUri, true);
108                         p.Out.WriteStartElement (prefix, localname, nsUri);
109
110                         if (useAttributeSets != null)
111                                 foreach (XmlQualifiedName s in useAttributeSets)
112                                         p.ResolveAttributeSet (s).Evaluate (p);
113                                                 
114                         if (attrs != null) {
115                                 int len = attrs.Count;
116                                 for (int i = 0; i < len; i++)
117                                         ((XslLiteralAttribute)attrs [i]).Evaluate (p);
118                         }
119
120                         p.OutputLiteralNamespaceUriNodes (nsDecls, null, null);
121
122                         if (children != null) children.Evaluate (p);
123
124                         if (isEmptyElement)
125                                 p.Out.WriteEndElement ();
126                         else
127                                 p.Out.WriteFullEndElement ();
128
129                         p.PopCDataState (isCData);
130                 }
131         }
132 }