//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // // [....] //------------------------------------------------------------------------------ namespace System.Xml.Xsl.XsltOld { using Res = System.Xml.Utils.Res; using System; using System.Diagnostics; using System.Xml; using System.Text; using System.Collections; using System.Globalization; internal abstract class SequentialOutput : RecordOutput { private const char s_Colon = ':'; private const char s_GreaterThan = '>'; private const char s_LessThan = '<'; private const char s_Space = ' '; private const char s_Quote = '\"'; private const char s_Semicolon = ';'; private const char s_NewLine = '\n'; private const char s_Return = '\r'; private const char s_Ampersand = '&'; private const string s_LessThanQuestion = ""; private const string s_LessThanSlash = ""; private const string s_EqualQuote = "=\""; private const string s_DocType = ""; private const string s_CDataBegin = ""; private const string s_VersionAll = " version=\"1.0\""; private const string s_Standalone = " standalone=\""; private const string s_EncodingStart = " encoding=\""; private const string s_Public = "PUBLIC "; private const string s_System = "SYSTEM "; private const string s_Html = "html"; private const string s_QuoteSpace = "\" "; private const string s_CDataSplit = "]]]]>"; private const string s_EnLessThan = "<"; private const string s_EnGreaterThan = ">"; private const string s_EnAmpersand = "&"; private const string s_EnQuote = """; private const string s_EnNewLine = " "; private const string s_EnReturn = " "; private const string s_EndOfLine = "\r\n"; static char[] s_TextValueFind = new char[] {s_Ampersand, s_GreaterThan, s_LessThan}; static string[] s_TextValueReplace = new string[] {s_EnAmpersand , s_EnGreaterThan , s_EnLessThan }; static char[] s_XmlAttributeValueFind = new char[] {s_Ampersand, s_GreaterThan, s_LessThan, s_Quote, s_NewLine, s_Return}; static string[] s_XmlAttributeValueReplace = new string[] {s_EnAmpersand , s_EnGreaterThan , s_EnLessThan , s_EnQuote , s_EnNewLine , s_EnReturn }; // Instance members private Processor processor; protected Encoding encoding; private ArrayList outputCache; private bool firstLine = true; private bool secondRoot; // Cached Output propertes: private XsltOutput output; private bool isHtmlOutput; private bool isXmlOutput; private Hashtable cdataElements; private bool indentOutput; private bool outputDoctype; private bool outputXmlDecl; private bool omitXmlDeclCalled; // Uri Escaping: private byte[] byteBuffer; private Encoding utf8Encoding; XmlCharType xmlCharType = XmlCharType.Instance; private void CacheOuptutProps(XsltOutput output) { this.output = output; this.isXmlOutput = this.output.Method == XsltOutput.OutputMethod.Xml; this.isHtmlOutput = this.output.Method == XsltOutput.OutputMethod.Html; this.cdataElements = this.output.CDataElements; this.indentOutput = this.output.Indent; this.outputDoctype = this.output.DoctypeSystem != null || (this.isHtmlOutput && this.output.DoctypePublic != null); this.outputXmlDecl = this.isXmlOutput && ! this.output.OmitXmlDeclaration && ! this.omitXmlDeclCalled; } // // Constructor // internal SequentialOutput(Processor processor) { this.processor = processor; CacheOuptutProps(processor.Output); } public void OmitXmlDecl() { this.omitXmlDeclCalled = true; this.outputXmlDecl = false; } // // Particular outputs // void WriteStartElement(RecordBuilder record) { Debug.Assert(record.MainNode.NodeType == XmlNodeType.Element); BuilderInfo mainNode = record.MainNode; HtmlElementProps htmlProps = null; if (this.isHtmlOutput) { if (mainNode.Prefix.Length == 0) { htmlProps = mainNode.htmlProps; if (htmlProps == null && mainNode.search) { htmlProps = HtmlElementProps.GetProps(mainNode.LocalName); } record.Manager.CurrentElementScope.HtmlElementProps = htmlProps; mainNode.IsEmptyTag = false; } } else if (this.isXmlOutput) { if (mainNode.Depth == 0) { if( secondRoot && ( output.DoctypeSystem != null || output.Standalone ) ) { throw XsltException.Create(Res.Xslt_MultipleRoots); } secondRoot = true; } } if (this.outputDoctype) { WriteDoctype(mainNode); this.outputDoctype = false; } if (this.cdataElements != null && this.cdataElements.Contains(new XmlQualifiedName(mainNode.LocalName, mainNode.NamespaceURI)) && this.isXmlOutput) { record.Manager.CurrentElementScope.ToCData = true; } Indent(record); Write(s_LessThan); WriteName(mainNode.Prefix, mainNode.LocalName); WriteAttributes(record.AttributeList, record.AttributeCount, htmlProps); if (mainNode.IsEmptyTag) { Debug.Assert(! this.isHtmlOutput || mainNode.Prefix != null, "Html can't have abreviated elements"); Write(s_SlashGreaterThan); } else { Write(s_GreaterThan); } if(htmlProps != null && htmlProps.Head) { mainNode.Depth ++; Indent(record); mainNode.Depth --; Write(""); } } void WriteTextNode(RecordBuilder record) { BuilderInfo mainNode = record.MainNode; OutputScope scope = record.Manager.CurrentElementScope; scope.Mixed = true; if(scope.HtmlElementProps != null && scope.HtmlElementProps.NoEntities) { // script or stile Write(mainNode.Value); } else if (scope.ToCData) { WriteCDataSection(mainNode.Value); } else { WriteTextNode(mainNode); } } void WriteTextNode(BuilderInfo node) { for (int i = 0; i < node.TextInfoCount; i ++) { string text = node.TextInfo[i]; if (text == null) { // disableEscaping marker i++; Debug.Assert(i < node.TextInfoCount, "disableEscaping marker can't be last TextInfo record"); Write(node.TextInfo[i]); } else { WriteWithReplace(text, s_TextValueFind, s_TextValueReplace); } } } void WriteCDataSection(string value) { Write(s_CDataBegin); WriteCData(value); Write(s_CDataEnd); } void WriteDoctype(BuilderInfo mainNode) { Debug.Assert(this.outputDoctype == true, "It supposed to check this condition before actual call"); Debug.Assert(this.output.DoctypeSystem != null || (this.isHtmlOutput && this.output.DoctypePublic != null), "We set outputDoctype == true only if"); Indent(0); Write(s_DocType); if (this.isXmlOutput) { WriteName(mainNode.Prefix, mainNode.LocalName); } else { WriteName(string.Empty, "html"); } Write(s_Space); if (output.DoctypePublic != null) { Write(s_Public); Write(s_Quote); Write(output.DoctypePublic); Write(s_QuoteSpace); } else { Write(s_System); } if (output.DoctypeSystem != null) { Write(s_Quote); Write(output.DoctypeSystem); Write(s_Quote); } Write(s_GreaterThan); } void WriteXmlDeclaration() { Debug.Assert(this.outputXmlDecl == true, "It supposed to check this condition before actual call"); Debug.Assert(this.isXmlOutput && ! this.output.OmitXmlDeclaration, "We set outputXmlDecl == true only if"); this.outputXmlDecl = false; Indent(0); Write(s_LessThanQuestion); WriteName(string.Empty, "xml"); Write(s_VersionAll); if (this.encoding != null) { Write(s_EncodingStart); Write(this.encoding.WebName); Write(s_Quote); } if (output.HasStandalone) { Write(s_Standalone); Write(output.Standalone ? "yes" : "no"); Write(s_Quote); } Write(s_QuestionGreaterThan); } void WriteProcessingInstruction(RecordBuilder record) { Indent(record); WriteProcessingInstruction(record.MainNode); } void WriteProcessingInstruction(BuilderInfo node) { Write(s_LessThanQuestion); WriteName(node.Prefix, node.LocalName); Write(s_Space); Write(node.Value); if(this.isHtmlOutput) { Write(s_GreaterThan); } else { Write(s_QuestionGreaterThan); } } void WriteEndElement(RecordBuilder record) { BuilderInfo node = record.MainNode; HtmlElementProps htmlProps = record.Manager.CurrentElementScope.HtmlElementProps; if(htmlProps != null && htmlProps.Empty) { return; } Indent(record); Write(s_LessThanSlash); WriteName(record.MainNode.Prefix, record.MainNode.LocalName); Write(s_GreaterThan); } // // RecordOutput interface method implementation // public Processor.OutputResult RecordDone(RecordBuilder record) { if (output.Method == XsltOutput.OutputMethod.Unknown) { if (! DecideDefaultOutput(record.MainNode)) { CacheRecord(record); } else { OutputCachedRecords(); OutputRecord(record); } } else { OutputRecord(record); } record.Reset(); return Processor.OutputResult.Continue; } public void TheEnd() { OutputCachedRecords(); Close(); } private bool DecideDefaultOutput(BuilderInfo node) { XsltOutput.OutputMethod method = XsltOutput.OutputMethod.Xml; switch (node.NodeType) { case XmlNodeType.Element: if (node.NamespaceURI.Length == 0 && String.Compare("html", node.LocalName, StringComparison.OrdinalIgnoreCase) == 0) { method = XsltOutput.OutputMethod.Html; } break; case XmlNodeType.Text: case XmlNodeType.Whitespace: case XmlNodeType.SignificantWhitespace: if (xmlCharType.IsOnlyWhitespace(node.Value)) { return false; } method = XsltOutput.OutputMethod.Xml; break; default : return false; } if(this.processor.SetDefaultOutput(method)) { CacheOuptutProps(processor.Output); } return true; } private void CacheRecord(RecordBuilder record) { if (this.outputCache == null) { this.outputCache = new ArrayList(); } this.outputCache.Add(record.MainNode.Clone()); } private void OutputCachedRecords() { if (this.outputCache == null) { return; } for(int record = 0; record < this.outputCache.Count; record ++) { Debug.Assert(this.outputCache[record] is BuilderInfo); BuilderInfo info = (BuilderInfo) this.outputCache[record]; OutputRecord(info); } this.outputCache = null; } private void OutputRecord(RecordBuilder record) { BuilderInfo mainNode = record.MainNode; if(this.outputXmlDecl) { WriteXmlDeclaration(); } switch (mainNode.NodeType) { case XmlNodeType.Element: WriteStartElement(record); break; case XmlNodeType.Text: case XmlNodeType.Whitespace: case XmlNodeType.SignificantWhitespace: WriteTextNode(record); break; case XmlNodeType.CDATA: Debug.Fail("Should never get here"); break; case XmlNodeType.EntityReference: Write(s_Ampersand); WriteName(mainNode.Prefix, mainNode.LocalName); Write(s_Semicolon); break; case XmlNodeType.ProcessingInstruction: WriteProcessingInstruction(record); break; case XmlNodeType.Comment: Indent(record); Write(s_CommentBegin); Write(mainNode.Value); Write(s_CommentEnd); break; case XmlNodeType.Document: break; case XmlNodeType.DocumentType: Write(mainNode.Value); break; case XmlNodeType.EndElement: WriteEndElement(record); break; default: break; } } private void OutputRecord(BuilderInfo node) { if(this.outputXmlDecl) { WriteXmlDeclaration(); } Indent(0); // we can have only top level stuff here switch (node.NodeType) { case XmlNodeType.Element: Debug.Fail("Should never get here"); break; case XmlNodeType.Text: case XmlNodeType.Whitespace: case XmlNodeType.SignificantWhitespace: WriteTextNode(node); break; case XmlNodeType.CDATA: Debug.Fail("Should never get here"); break; case XmlNodeType.EntityReference: Write(s_Ampersand); WriteName(node.Prefix, node.LocalName); Write(s_Semicolon); break; case XmlNodeType.ProcessingInstruction: WriteProcessingInstruction(node); break; case XmlNodeType.Comment: Write(s_CommentBegin); Write(node.Value); Write(s_CommentEnd); break; case XmlNodeType.Document: break; case XmlNodeType.DocumentType: Write(node.Value); break; case XmlNodeType.EndElement: Debug.Fail("Should never get here"); break; default: break; } } // // Internal helpers // private void WriteName(string prefix, string name) { if (prefix != null && prefix.Length > 0) { Write(prefix); if (name != null && name.Length > 0) { Write(s_Colon); } else { return; } } Write(name); } private void WriteXmlAttributeValue(string value) { Debug.Assert(value != null); WriteWithReplace(value, s_XmlAttributeValueFind, s_XmlAttributeValueReplace); } private void WriteHtmlAttributeValue(string value) { Debug.Assert(value != null); int length = value.Length; int i = 0; while(i < length) { char ch = value[i]; i ++; switch (ch) { case '&': if(i != length && value[i] == '{') { // &{ hasn't to be encoded in HTML output. Write(ch); } else { Write(s_EnAmpersand); } break; case '"': Write(s_EnQuote); break; default: Write(ch); break; } } } private void WriteHtmlUri(string value) { Debug.Assert(value != null); Debug.Assert(this.isHtmlOutput); int length = value.Length; int i = 0; while(i < length) { char ch = value[i]; i ++; switch (ch) { case '&': if(i != length && value[i] == '{') { // &{ hasn't to be encoded in HTML output. Write(ch); } else { Write(s_EnAmpersand); } break; case '"': Write(s_EnQuote); break; case '\n': Write(s_EnNewLine); break; case '\r': Write(s_EnReturn); break; default: if(127 < ch) { if (this.utf8Encoding == null) { this.utf8Encoding = Encoding.UTF8; this.byteBuffer = new byte[utf8Encoding.GetMaxByteCount(1)]; } int bytes = this.utf8Encoding.GetBytes(value, i - 1, 1, this.byteBuffer, 0); for(int j = 0; j < bytes; j ++) { Write("%"); Write(((uint)this.byteBuffer[j]).ToString("X2", CultureInfo.InvariantCulture)); } } else { Write(ch); } break; } } } private void WriteWithReplace(string value, char[] find, string[] replace) { Debug.Assert(value != null); Debug.Assert(find.Length == replace.Length); int length = value.Length; int pos = 0; while(pos < length) { int newPos = value.IndexOfAny(find, pos); if (newPos == -1) { break; // not found; } // output clean leading part of the string while (pos < newPos) { Write(value[pos]); pos ++; } // output replacement char badChar = value[pos]; int i; for(i = find.Length - 1; 0 <= i; i --) { if(find[i] == badChar) { Write(replace[i]); break; } } Debug.Assert(0 <= i, "find char wasn't realy find"); pos ++; } // output rest of the string if(pos == 0) { Write(value); } else { while(pos < length) { Write(value[pos]); pos ++; } } } private void WriteCData(string value) { Debug.Assert(value != null); Write(value.Replace(s_CDataEnd, s_CDataSplit)); } private void WriteAttributes(ArrayList list, int count, HtmlElementProps htmlElementsProps) { Debug.Assert(count <= list.Count); for (int attrib = 0; attrib < count; attrib ++) { Debug.Assert(list[attrib] is BuilderInfo); BuilderInfo attribute = (BuilderInfo) list[attrib]; string attrValue = attribute.Value; bool abr = false, uri = false; { if(htmlElementsProps != null && attribute.Prefix.Length == 0) { HtmlAttributeProps htmlAttrProps = attribute.htmlAttrProps; if (htmlAttrProps == null && attribute.search) { htmlAttrProps = HtmlAttributeProps.GetProps(attribute.LocalName); } if(htmlAttrProps != null) { abr = htmlElementsProps.AbrParent && htmlAttrProps.Abr; uri = htmlElementsProps.UriParent && ( htmlAttrProps.Uri || htmlElementsProps.NameParent && htmlAttrProps.Name ); } } } Write(s_Space); WriteName(attribute.Prefix, attribute.LocalName); if(abr && 0 == string.Compare(attribute.LocalName, attrValue, StringComparison.OrdinalIgnoreCase) ) { // Since the name of the attribute = the value of the attribute, // this is a boolean attribute whose value should be suppressed continue; } Write(s_EqualQuote); if(uri) { WriteHtmlUri(attrValue); } else if(this.isHtmlOutput) { WriteHtmlAttributeValue(attrValue); } else { WriteXmlAttributeValue(attrValue); } Write(s_Quote); } } void Indent(RecordBuilder record) { if (! record.Manager.CurrentElementScope.Mixed) { Indent(record.MainNode.Depth); } } void Indent(int depth) { if(this.firstLine) { if (this.indentOutput) { this.firstLine = false; } return; // preven leading CRLF } Write(s_EndOfLine); for (int i = 2 * depth; 0 < i; i--) { Write(" "); } } // // Abstract methods internal abstract void Write(char outputChar); internal abstract void Write(string outputText); internal abstract void Close(); } }