1418 lines
55 KiB
C#
Raw Normal View History

//------------------------------------------------------------------------------
// <copyright file="HtmlTextWriter.cs" company="Microsoft">
// Copyright (c) Microsoft Corporation. All rights reserved.
// </copyright>
//------------------------------------------------------------------------------
// HtmlTextWriter.cs
//
namespace System.Web.UI {
using System;
using System.Collections;
using System.Collections.Specialized;
using System.IO;
using System.Text;
using System.Globalization;
using System.Security.Permissions;
using System.Web.UI.WebControls;
using System.Web.Util;
public class HtmlTextWriter : TextWriter {
private Layout _currentLayout = new Layout(HorizontalAlign.NotSet, true /* wrap */);
private Layout _currentWrittenLayout = null;
internal virtual bool RenderDivAroundHiddenInputs {
get {
return true;
}
}
public virtual void EnterStyle(Style style, HtmlTextWriterTag tag) {
if (!style.IsEmpty || tag != HtmlTextWriterTag.Span) {
style.AddAttributesToRender(this);
RenderBeginTag(tag);
}
}
public virtual void ExitStyle(System.Web.UI.WebControls.Style style, HtmlTextWriterTag tag) {
// Review: This requires that the style doesn't change between beginstyle/endstyle.
if (!style.IsEmpty || tag != HtmlTextWriterTag.Span) {
RenderEndTag();
}
}
internal virtual void OpenDiv() {
OpenDiv(_currentLayout,
(_currentLayout != null) && (_currentLayout.Align != HorizontalAlign.NotSet),
(_currentLayout != null) && !_currentLayout.Wrap);
}
private void OpenDiv(Layout layout, bool writeHorizontalAlign, bool writeWrapping) {
WriteBeginTag("div");
if (writeHorizontalAlign) {
String alignment;
switch (layout.Align) {
case HorizontalAlign.Right:
alignment = "text-align:right";
break;
case HorizontalAlign.Center:
alignment = "text-align:center";
break;
default:
alignment = "text-align:left";
break;
}
WriteAttribute("style", alignment);
}
if (writeWrapping) {
WriteAttribute("mode",
layout.Wrap == true ? "wrap" : "nowrap");
}
Write('>');
_currentWrittenLayout = layout;
}
public virtual bool IsValidFormAttribute(String attribute) {
return true;
}
private TextWriter writer;
private int indentLevel;
private bool tabsPending;
private string tabString;
public const char TagLeftChar = '<';
public const char TagRightChar = '>';
public const string SelfClosingChars = " /";
public const string SelfClosingTagEnd = " />";
public const string EndTagLeftChars = "</";
public const char DoubleQuoteChar = '"';
public const char SingleQuoteChar = '\'';
public const char SpaceChar = ' ';
public const char EqualsChar = '=';
public const char SlashChar = '/';
public const string EqualsDoubleQuoteString = "=\"";
public const char SemicolonChar = ';';
public const char StyleEqualsChar = ':';
public const string DefaultTabString = "\t";
// The DesignerRegion attribute name must be kept in sync with
// System.Web.UI.Design.DesignerRegion.DesignerRegionNameAttribute
internal const string DesignerRegionAttributeName = "_designerRegion";
private static Hashtable _tagKeyLookupTable;
private static Hashtable _attrKeyLookupTable;
private static TagInformation[] _tagNameLookupArray;
private static AttributeInformation[] _attrNameLookupArray;
private RenderAttribute[] _attrList;
private int _attrCount;
private int _endTagCount;
private TagStackEntry[] _endTags;
private HttpWriter _httpWriter;
private int _inlineCount;
private bool _isDescendant;
private RenderStyle[] _styleList;
private int _styleCount;
private int _tagIndex;
private HtmlTextWriterTag _tagKey;
private string _tagName;
static HtmlTextWriter() {
// register known tags
_tagKeyLookupTable = new Hashtable((int)HtmlTextWriterTag.Xml + 1);
_tagNameLookupArray = new TagInformation[(int)HtmlTextWriterTag.Xml + 1];
RegisterTag(String.Empty, HtmlTextWriterTag.Unknown, TagType.Other);
RegisterTag("a", HtmlTextWriterTag.A, TagType.Inline);
RegisterTag("acronym", HtmlTextWriterTag.Acronym, TagType.Inline);
RegisterTag("address", HtmlTextWriterTag.Address, TagType.Other);
RegisterTag("area", HtmlTextWriterTag.Area, TagType.NonClosing);
RegisterTag("b", HtmlTextWriterTag.B, TagType.Inline);
RegisterTag("base", HtmlTextWriterTag.Base, TagType.NonClosing);
RegisterTag("basefont", HtmlTextWriterTag.Basefont, TagType.NonClosing);
RegisterTag("bdo", HtmlTextWriterTag.Bdo, TagType.Inline);
RegisterTag("bgsound", HtmlTextWriterTag.Bgsound, TagType.NonClosing);
RegisterTag("big", HtmlTextWriterTag.Big, TagType.Inline);
RegisterTag("blockquote", HtmlTextWriterTag.Blockquote, TagType.Other);
RegisterTag("body", HtmlTextWriterTag.Body, TagType.Other);
// Devdiv 852940, BR is a self-closing tag
RegisterTag("br", HtmlTextWriterTag.Br,
BinaryCompatibility.Current.TargetsAtLeastFramework46 ? TagType.NonClosing : TagType.Other);
RegisterTag("button", HtmlTextWriterTag.Button, TagType.Inline);
RegisterTag("caption", HtmlTextWriterTag.Caption, TagType.Other);
RegisterTag("center", HtmlTextWriterTag.Center, TagType.Other);
RegisterTag("cite", HtmlTextWriterTag.Cite, TagType.Inline);
RegisterTag("code", HtmlTextWriterTag.Code, TagType.Inline);
RegisterTag("col", HtmlTextWriterTag.Col, TagType.NonClosing);
RegisterTag("colgroup", HtmlTextWriterTag.Colgroup, TagType.Other);
RegisterTag("del", HtmlTextWriterTag.Del, TagType.Inline);
RegisterTag("dd", HtmlTextWriterTag.Dd, TagType.Inline);
RegisterTag("dfn", HtmlTextWriterTag.Dfn, TagType.Inline);
RegisterTag("dir", HtmlTextWriterTag.Dir, TagType.Other);
RegisterTag("div", HtmlTextWriterTag.Div, TagType.Other);
RegisterTag("dl", HtmlTextWriterTag.Dl, TagType.Other);
RegisterTag("dt", HtmlTextWriterTag.Dt, TagType.Inline);
RegisterTag("em", HtmlTextWriterTag.Em, TagType.Inline);
RegisterTag("embed", HtmlTextWriterTag.Embed, TagType.NonClosing);
RegisterTag("fieldset", HtmlTextWriterTag.Fieldset, TagType.Other);
RegisterTag("font", HtmlTextWriterTag.Font, TagType.Inline);
RegisterTag("form", HtmlTextWriterTag.Form, TagType.Other);
RegisterTag("frame", HtmlTextWriterTag.Frame, TagType.NonClosing);
RegisterTag("frameset", HtmlTextWriterTag.Frameset, TagType.Other);
RegisterTag("h1", HtmlTextWriterTag.H1, TagType.Other);
RegisterTag("h2", HtmlTextWriterTag.H2, TagType.Other);
RegisterTag("h3", HtmlTextWriterTag.H3, TagType.Other);
RegisterTag("h4", HtmlTextWriterTag.H4, TagType.Other);
RegisterTag("h5", HtmlTextWriterTag.H5, TagType.Other);
RegisterTag("h6", HtmlTextWriterTag.H6, TagType.Other);
RegisterTag("head", HtmlTextWriterTag.Head, TagType.Other);
RegisterTag("hr", HtmlTextWriterTag.Hr, TagType.NonClosing);
RegisterTag("html", HtmlTextWriterTag.Html, TagType.Other);
RegisterTag("i", HtmlTextWriterTag.I, TagType.Inline);
RegisterTag("iframe", HtmlTextWriterTag.Iframe, TagType.Other);
RegisterTag("img", HtmlTextWriterTag.Img, TagType.NonClosing);
RegisterTag("input", HtmlTextWriterTag.Input, TagType.NonClosing);
RegisterTag("ins", HtmlTextWriterTag.Ins, TagType.Inline);
RegisterTag("isindex", HtmlTextWriterTag.Isindex, TagType.NonClosing);
RegisterTag("kbd", HtmlTextWriterTag.Kbd, TagType.Inline);
RegisterTag("label", HtmlTextWriterTag.Label, TagType.Inline);
RegisterTag("legend", HtmlTextWriterTag.Legend, TagType.Other);
RegisterTag("li", HtmlTextWriterTag.Li, TagType.Inline);
RegisterTag("link", HtmlTextWriterTag.Link, TagType.NonClosing);
RegisterTag("map", HtmlTextWriterTag.Map, TagType.Other);
RegisterTag("marquee", HtmlTextWriterTag.Marquee, TagType.Other);
RegisterTag("menu", HtmlTextWriterTag.Menu, TagType.Other);
RegisterTag("meta", HtmlTextWriterTag.Meta, TagType.NonClosing);
RegisterTag("nobr", HtmlTextWriterTag.Nobr, TagType.Inline);
RegisterTag("noframes", HtmlTextWriterTag.Noframes, TagType.Other);
RegisterTag("noscript", HtmlTextWriterTag.Noscript, TagType.Other);
RegisterTag("object", HtmlTextWriterTag.Object, TagType.Other);
RegisterTag("ol", HtmlTextWriterTag.Ol, TagType.Other);
RegisterTag("option", HtmlTextWriterTag.Option, TagType.Other);
RegisterTag("p", HtmlTextWriterTag.P, TagType.Inline);
RegisterTag("param", HtmlTextWriterTag.Param, TagType.Other);
RegisterTag("pre", HtmlTextWriterTag.Pre, TagType.Other);
RegisterTag("ruby", HtmlTextWriterTag.Ruby, TagType.Other);
RegisterTag("rt", HtmlTextWriterTag.Rt, TagType.Other);
RegisterTag("q", HtmlTextWriterTag.Q, TagType.Inline);
RegisterTag("s", HtmlTextWriterTag.S, TagType.Inline);
RegisterTag("samp", HtmlTextWriterTag.Samp, TagType.Inline);
RegisterTag("script", HtmlTextWriterTag.Script, TagType.Other);
RegisterTag("select", HtmlTextWriterTag.Select, TagType.Other);
RegisterTag("small", HtmlTextWriterTag.Small, TagType.Other);
RegisterTag("span", HtmlTextWriterTag.Span, TagType.Inline);
RegisterTag("strike", HtmlTextWriterTag.Strike, TagType.Inline);
RegisterTag("strong", HtmlTextWriterTag.Strong, TagType.Inline);
RegisterTag("style", HtmlTextWriterTag.Style, TagType.Other);
RegisterTag("sub", HtmlTextWriterTag.Sub, TagType.Inline);
RegisterTag("sup", HtmlTextWriterTag.Sup, TagType.Inline);
RegisterTag("table", HtmlTextWriterTag.Table, TagType.Other);
RegisterTag("tbody", HtmlTextWriterTag.Tbody, TagType.Other);
RegisterTag("td", HtmlTextWriterTag.Td, TagType.Inline);
RegisterTag("textarea", HtmlTextWriterTag.Textarea, TagType.Inline);
RegisterTag("tfoot", HtmlTextWriterTag.Tfoot, TagType.Other);
RegisterTag("th", HtmlTextWriterTag.Th, TagType.Inline);
RegisterTag("thead", HtmlTextWriterTag.Thead, TagType.Other);
RegisterTag("title", HtmlTextWriterTag.Title, TagType.Other);
RegisterTag("tr", HtmlTextWriterTag.Tr, TagType.Other);
RegisterTag("tt", HtmlTextWriterTag.Tt, TagType.Inline);
RegisterTag("u", HtmlTextWriterTag.U, TagType.Inline);
RegisterTag("ul", HtmlTextWriterTag.Ul, TagType.Other);
RegisterTag("var", HtmlTextWriterTag.Var, TagType.Inline);
RegisterTag("wbr", HtmlTextWriterTag.Wbr, TagType.NonClosing);
RegisterTag("xml", HtmlTextWriterTag.Xml, TagType.Other);
// register known attributes
_attrKeyLookupTable = new Hashtable((int)HtmlTextWriterAttribute.VCardName + 1);
_attrNameLookupArray = new AttributeInformation[(int)HtmlTextWriterAttribute.VCardName + 1];
RegisterAttribute("abbr", HtmlTextWriterAttribute.Abbr, true);
RegisterAttribute("accesskey", HtmlTextWriterAttribute.Accesskey, true);
RegisterAttribute("align", HtmlTextWriterAttribute.Align, false);
RegisterAttribute("alt", HtmlTextWriterAttribute.Alt, true);
RegisterAttribute("autocomplete", HtmlTextWriterAttribute.AutoComplete, false);
RegisterAttribute("axis", HtmlTextWriterAttribute.Axis, true);
RegisterAttribute("background", HtmlTextWriterAttribute.Background, true, true);
RegisterAttribute("bgcolor", HtmlTextWriterAttribute.Bgcolor, false);
RegisterAttribute("border", HtmlTextWriterAttribute.Border, false);
RegisterAttribute("bordercolor", HtmlTextWriterAttribute.Bordercolor, false);
RegisterAttribute("cellpadding", HtmlTextWriterAttribute.Cellpadding, false);
RegisterAttribute("cellspacing", HtmlTextWriterAttribute.Cellspacing, false);
RegisterAttribute("checked", HtmlTextWriterAttribute.Checked, false);
RegisterAttribute("class", HtmlTextWriterAttribute.Class, true);
RegisterAttribute("cols", HtmlTextWriterAttribute.Cols, false);
RegisterAttribute("colspan", HtmlTextWriterAttribute.Colspan, false);
RegisterAttribute("content", HtmlTextWriterAttribute.Content, true);
RegisterAttribute("coords", HtmlTextWriterAttribute.Coords, false);
RegisterAttribute("dir", HtmlTextWriterAttribute.Dir, false);
RegisterAttribute("disabled", HtmlTextWriterAttribute.Disabled, false);
RegisterAttribute("for", HtmlTextWriterAttribute.For, false);
RegisterAttribute("headers", HtmlTextWriterAttribute.Headers, true);
RegisterAttribute("height", HtmlTextWriterAttribute.Height, false);
RegisterAttribute("href", HtmlTextWriterAttribute.Href, true, true);
RegisterAttribute("id", HtmlTextWriterAttribute.Id, false);
RegisterAttribute("longdesc", HtmlTextWriterAttribute.Longdesc, true, true);
RegisterAttribute("maxlength", HtmlTextWriterAttribute.Maxlength, false);
RegisterAttribute("multiple", HtmlTextWriterAttribute.Multiple, false);
RegisterAttribute("name", HtmlTextWriterAttribute.Name, false);
RegisterAttribute("nowrap", HtmlTextWriterAttribute.Nowrap, false);
RegisterAttribute("onclick", HtmlTextWriterAttribute.Onclick, true);
RegisterAttribute("onchange", HtmlTextWriterAttribute.Onchange, true);
RegisterAttribute("readonly", HtmlTextWriterAttribute.ReadOnly, false);
RegisterAttribute("rel", HtmlTextWriterAttribute.Rel, false);
RegisterAttribute("rows", HtmlTextWriterAttribute.Rows, false);
RegisterAttribute("rowspan", HtmlTextWriterAttribute.Rowspan, false);
RegisterAttribute("rules", HtmlTextWriterAttribute.Rules, false);
RegisterAttribute("scope", HtmlTextWriterAttribute.Scope, false);
RegisterAttribute("selected", HtmlTextWriterAttribute.Selected, false);
RegisterAttribute("shape", HtmlTextWriterAttribute.Shape, false);
RegisterAttribute("size", HtmlTextWriterAttribute.Size, false);
RegisterAttribute("src", HtmlTextWriterAttribute.Src, true, true);
RegisterAttribute("style", HtmlTextWriterAttribute.Style, false);
RegisterAttribute("tabindex", HtmlTextWriterAttribute.Tabindex, false);
RegisterAttribute("target", HtmlTextWriterAttribute.Target, false);
RegisterAttribute("title", HtmlTextWriterAttribute.Title, true);
RegisterAttribute("type", HtmlTextWriterAttribute.Type, false);
RegisterAttribute("usemap", HtmlTextWriterAttribute.Usemap, false);
RegisterAttribute("valign", HtmlTextWriterAttribute.Valign, false);
RegisterAttribute("value", HtmlTextWriterAttribute.Value, true);
RegisterAttribute("vcard_name", HtmlTextWriterAttribute.VCardName, false);
RegisterAttribute("width", HtmlTextWriterAttribute.Width, false);
RegisterAttribute("wrap", HtmlTextWriterAttribute.Wrap, false);
RegisterAttribute(DesignerRegionAttributeName, HtmlTextWriterAttribute.DesignerRegion, false);
}
public override Encoding Encoding {
get {
return writer.Encoding;
}
}
// Gets or sets the new line character to use.
public override string NewLine {
get {
return writer.NewLine;
}
set {
writer.NewLine = value;
}
}
// Gets or sets the number of spaces to indent.
public int Indent {
get {
return indentLevel;
}
set {
Debug.Assert(value >= 0, "Bogus Indent... probably caused by mismatched Indent++ and Indent--");
if (value < 0) {
value = 0;
}
indentLevel = value;
}
}
//Gets or sets the TextWriter to use.
public TextWriter InnerWriter {
get {
return writer;
}
set {
writer = value;
_httpWriter = value as HttpWriter;
}
}
/*
//
*/
public virtual void BeginRender() {
}
//Closes the document being written to.
public override void Close() {
writer.Close();
}
public virtual void EndRender() {
}
public virtual void EnterStyle(System.Web.UI.WebControls.Style style) {
EnterStyle(style, HtmlTextWriterTag.Span);
}
public virtual void ExitStyle(System.Web.UI.WebControls.Style style) {
ExitStyle(style, HtmlTextWriterTag.Span);
}
public override void Flush() {
writer.Flush();
}
protected virtual void OutputTabs() {
if (tabsPending) {
for (int i=0; i < indentLevel; i++) {
writer.Write(tabString);
}
tabsPending = false;
}
}
//Writes a string to the text stream.
public override void Write(string s) {
if (tabsPending) {
OutputTabs();
}
writer.Write(s);
}
//Writes the text representation of a Boolean value to the text stream.
public override void Write(bool value) {
if (tabsPending) {
OutputTabs();
}
writer.Write(value);
}
//Writes a character to the text stream.
public override void Write(char value) {
if (tabsPending) {
OutputTabs();
}
writer.Write(value);
}
// Writes a character array to the text stream.
public override void Write(char[] buffer) {
if (tabsPending) {
OutputTabs();
}
writer.Write(buffer);
}
// Writes a subarray of characters to the text stream.
public override void Write(char[] buffer, int index, int count) {
if (tabsPending) {
OutputTabs();
}
writer.Write(buffer, index, count);
}
// Writes the text representation of a Double to the text stream.
public override void Write(double value) {
if (tabsPending) {
OutputTabs();
}
writer.Write(value);
}
// Writes the text representation of a Single to the text stream.
public override void Write(float value) {
if (tabsPending) {
OutputTabs();
}
writer.Write(value);
}
// Writes the text representation of an integer to the text stream.
public override void Write(int value) {
if (tabsPending) {
OutputTabs();
}
writer.Write(value);
}
// Writes the text representation of an 8-byte integer to the text stream.
public override void Write(long value) {
if (tabsPending) {
OutputTabs();
}
writer.Write(value);
}
// Writes the text representation of an object to the text stream.
public override void Write(object value) {
if (tabsPending) {
OutputTabs();
}
writer.Write(value);
}
// Writes out a formatted string, using the same semantics as specified.
public override void Write(string format, object arg0) {
if (tabsPending) {
OutputTabs();
}
writer.Write(format, arg0);
}
// Writes out a formatted string, using the same semantics as specified.
public override void Write(string format, object arg0, object arg1) {
if (tabsPending) {
OutputTabs();
}
writer.Write(format, arg0, arg1);
}
// Writes out a formatted string, using the same semantics as specified.
public override void Write(string format, params object[] arg) {
if (tabsPending) {
OutputTabs();
}
writer.Write(format, arg);
}
// Writes the specified string to a line without tabs.
public void WriteLineNoTabs(string s) {
writer.WriteLine(s);
tabsPending = true;
}
// Writes the specified string followed by a line terminator to the text stream.
public override void WriteLine(string s) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(s);
tabsPending = true;
}
// Writes a line terminator.
public override void WriteLine() {
writer.WriteLine();
tabsPending = true;
}
// Writes the text representation of a Boolean followed by a line terminator to the text stream.
public override void WriteLine(bool value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
public override void WriteLine(char value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
public override void WriteLine(char[] buffer) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(buffer);
tabsPending = true;
}
public override void WriteLine(char[] buffer, int index, int count) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(buffer, index, count);
tabsPending = true;
}
public override void WriteLine(double value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
public override void WriteLine(float value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
public override void WriteLine(int value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
public override void WriteLine(long value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
public override void WriteLine(object value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
public override void WriteLine(string format, object arg0) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(format, arg0);
tabsPending = true;
}
public override void WriteLine(string format, object arg0, object arg1) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(format, arg0, arg1);
tabsPending = true;
}
public override void WriteLine(string format, params object[] arg) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(format, arg);
tabsPending = true;
}
[CLSCompliant(false)]
public override void WriteLine(UInt32 value) {
if (tabsPending) {
OutputTabs();
}
writer.WriteLine(value);
tabsPending = true;
}
protected static void RegisterTag(string name, HtmlTextWriterTag key) {
RegisterTag(name, key, TagType.Other);
}
private static void RegisterTag(string name, HtmlTextWriterTag key, TagType type) {
string nameLCase = name.ToLower(CultureInfo.InvariantCulture);
_tagKeyLookupTable.Add(nameLCase, key);
// Pre-resolve the end tag
string endTag = null;
if (type != TagType.NonClosing && key != HtmlTextWriterTag.Unknown) {
endTag = EndTagLeftChars + nameLCase + TagRightChar.ToString(CultureInfo.InvariantCulture);
}
if ((int)key < _tagNameLookupArray.Length) {
_tagNameLookupArray[(int)key] = new TagInformation(name, type, endTag);
}
}
protected static void RegisterAttribute(string name, HtmlTextWriterAttribute key) {
RegisterAttribute(name, key, false);
}
private static void RegisterAttribute(string name, HtmlTextWriterAttribute key, bool encode) {
RegisterAttribute(name, key, encode, false);
}
private static void RegisterAttribute(string name, HtmlTextWriterAttribute key, bool encode, bool isUrl) {
string nameLCase = name.ToLower(CultureInfo.InvariantCulture);
_attrKeyLookupTable.Add(nameLCase, key);
if ((int)key < _attrNameLookupArray.Length) {
_attrNameLookupArray[(int)key] = new AttributeInformation(name, encode, isUrl);
}
}
protected static void RegisterStyle(string name, HtmlTextWriterStyle key) {
CssTextWriter.RegisterAttribute(name, key);
}
public HtmlTextWriter(TextWriter writer) : this(writer, DefaultTabString) {
}
public HtmlTextWriter(TextWriter writer, string tabString) : base(CultureInfo.InvariantCulture) {
this.writer = writer;
this.tabString = tabString;
indentLevel = 0;
tabsPending = false;
// If it's an http writer, save it
_httpWriter = writer as HttpWriter;
_isDescendant = (GetType() != typeof(HtmlTextWriter));
_attrCount = 0;
_styleCount = 0;
_endTagCount = 0;
_inlineCount = 0;
}
protected HtmlTextWriterTag TagKey {
get {
return _tagKey;
}
set {
_tagIndex = (int) value;
if (_tagIndex < 0 || _tagIndex >= _tagNameLookupArray.Length) {
throw new ArgumentOutOfRangeException("value");
}
_tagKey = value;
// If explicitly setting to uknown, keep the old tag name. This allows a string tag
// to be set without clobbering it if setting TagKey to itself.
if (value != HtmlTextWriterTag.Unknown) {
_tagName = _tagNameLookupArray[_tagIndex].name;
}
}
}
protected string TagName {
get {
return _tagName;
}
set {
_tagName = value;
_tagKey = GetTagKey(_tagName);
_tagIndex = (int) _tagKey;
Debug.Assert(_tagIndex >= 0 && _tagIndex < _tagNameLookupArray.Length);
}
}
public virtual void AddAttribute(string name,string value) {
HtmlTextWriterAttribute attributeKey = GetAttributeKey(name);
value = EncodeAttributeValue(attributeKey, value);
AddAttribute(name, value, attributeKey);
}
//do not fix this spelling error
//believe it or not, it is a backwards breaking change for languages that
//support late binding with named parameters VB.Net
public virtual void AddAttribute(string name,string value, bool fEndode) {
value = EncodeAttributeValue(value, fEndode);
AddAttribute(name, value, GetAttributeKey(name));
}
public virtual void AddAttribute(HtmlTextWriterAttribute key,string value) {
int attributeIndex = (int) key;
if (attributeIndex >= 0 && attributeIndex < _attrNameLookupArray.Length) {
AttributeInformation info = _attrNameLookupArray[attributeIndex];
AddAttribute(info.name,value,key, info.encode, info.isUrl);
}
}
public virtual void AddAttribute(HtmlTextWriterAttribute key,string value, bool fEncode) {
int attributeIndex = (int) key;
if (attributeIndex >= 0 && attributeIndex < _attrNameLookupArray.Length) {
AttributeInformation info = _attrNameLookupArray[attributeIndex];
AddAttribute(info.name,value,key, fEncode, info.isUrl);
}
}
protected virtual void AddAttribute(string name, string value, HtmlTextWriterAttribute key) {
AddAttribute(name, value, key, false, false);
}
private void AddAttribute(string name, string value, HtmlTextWriterAttribute key, bool encode, bool isUrl) {
if(_attrList == null) {
_attrList = new RenderAttribute[20];
}
else if (_attrCount >= _attrList.Length) {
RenderAttribute[] newArray = new RenderAttribute[_attrList.Length * 2];
Array.Copy(_attrList, newArray, _attrList.Length);
_attrList = newArray;
}
RenderAttribute attr;
attr.name = name;
attr.value = value;
attr.key = key;
attr.encode = encode;
attr.isUrl = isUrl;
_attrList[_attrCount] = attr;
_attrCount++;
}
public virtual void AddStyleAttribute(string name, string value) {
AddStyleAttribute(name, value, CssTextWriter.GetStyleKey(name));
}
public virtual void AddStyleAttribute(HtmlTextWriterStyle key, string value) {
AddStyleAttribute(CssTextWriter.GetStyleName(key), value, key);
}
protected virtual void AddStyleAttribute(string name, string value, HtmlTextWriterStyle key) {
if(_styleList == null) {
_styleList = new RenderStyle[20];
}
else if (_styleCount > _styleList.Length) {
RenderStyle[] newArray = new RenderStyle[_styleList.Length * 2];
Array.Copy(_styleList, newArray, _styleList.Length);
_styleList = newArray;
}
RenderStyle style;
style.name = name;
style.key = key;
string attributeValue = value;
if (CssTextWriter.IsStyleEncoded(key)) {
// note that only css attributes in an inline style value need to be attribute encoded
// since CssTextWriter is used to render both embedded stylesheets and style attributes
// the attribute encoding is done here.
attributeValue = HttpUtility.HtmlAttributeEncode(value);
}
style.value = attributeValue;
_styleList[_styleCount] = style;
_styleCount++;
}
protected string EncodeAttributeValue(string value, bool fEncode) {
if (value == null) {
return null;
}
if (!fEncode)
return value;
return HttpUtility.HtmlAttributeEncode(value);
}
protected virtual string EncodeAttributeValue(HtmlTextWriterAttribute attrKey, string value) {
bool encode = true;
if (0 <= (int)attrKey && (int)attrKey < _attrNameLookupArray.Length) {
encode = _attrNameLookupArray[(int)attrKey].encode;
}
return EncodeAttributeValue(value, encode);
}
// This does minimal URL encoding by converting spaces in the url to "%20".
protected string EncodeUrl(string url) {
// VSWhidbey 454348: escaped spaces in UNC share paths don't work in IE, so
// we're not going to encode if it's a share.
if (!UrlPath.IsUncSharePath(url)) {
return HttpUtility.UrlPathEncode(url);
}
return url;
}
protected HtmlTextWriterAttribute GetAttributeKey(string attrName) {
if (!String.IsNullOrEmpty(attrName)) {
object key = _attrKeyLookupTable[attrName.ToLower(CultureInfo.InvariantCulture)];
if (key != null)
return(HtmlTextWriterAttribute)key;
}
return(HtmlTextWriterAttribute)(-1);
}
protected string GetAttributeName(HtmlTextWriterAttribute attrKey) {
if ((int)attrKey >= 0 && (int)attrKey < _attrNameLookupArray.Length)
return _attrNameLookupArray[(int)attrKey].name;
return string.Empty;
}
protected HtmlTextWriterStyle GetStyleKey(string styleName) {
return CssTextWriter.GetStyleKey(styleName);
}
protected string GetStyleName(HtmlTextWriterStyle styleKey) {
return CssTextWriter.GetStyleName(styleKey);
}
protected virtual HtmlTextWriterTag GetTagKey(string tagName) {
if (!String.IsNullOrEmpty(tagName)) {
object key = _tagKeyLookupTable[tagName.ToLower(CultureInfo.InvariantCulture)];
if (key != null)
return(HtmlTextWriterTag)key;
}
return HtmlTextWriterTag.Unknown;
}
protected virtual string GetTagName(HtmlTextWriterTag tagKey) {
int tagIndex = (int) tagKey;
if (tagIndex >= 0 && tagIndex < _tagNameLookupArray.Length)
return _tagNameLookupArray[tagIndex].name;
return string.Empty;
}
protected bool IsAttributeDefined(HtmlTextWriterAttribute key) {
for (int i = 0; i < _attrCount; i++) {
if (_attrList[i].key == key) {
return true;
}
}
return false;
}
protected bool IsAttributeDefined(HtmlTextWriterAttribute key, out string value) {
value = null;
for (int i = 0; i < _attrCount; i++) {
if (_attrList[i].key == key) {
value = _attrList[i].value;
return true;
}
}
return false;
}
protected bool IsStyleAttributeDefined(HtmlTextWriterStyle key) {
for (int i = 0; i < _styleCount; i++) {
if (_styleList[i].key == key) {
return true;
}
}
return false;
}
protected bool IsStyleAttributeDefined(HtmlTextWriterStyle key, out string value) {
value = null;
for (int i = 0; i < _styleCount; i++) {
if (_styleList[i].key == key) {
value = _styleList[i].value;
return true;
}
}
return false;
}
protected virtual bool OnAttributeRender(string name, string value, HtmlTextWriterAttribute key) {
return true;
}
protected virtual bool OnStyleAttributeRender(string name, string value, HtmlTextWriterStyle key) {
return true;
}
protected virtual bool OnTagRender(string name, HtmlTextWriterTag key) {
return true;
}
protected string PopEndTag() {
if (_endTagCount <= 0) {
throw new InvalidOperationException(SR.GetString(SR.HTMLTextWriterUnbalancedPop));
}
_endTagCount--;
TagKey = _endTags[_endTagCount].tagKey;
return _endTags[_endTagCount].endTagText;
}
protected void PushEndTag(string endTag) {
if(_endTags == null) {
_endTags = new TagStackEntry[16];
}
else if (_endTagCount >= _endTags.Length) {
TagStackEntry[] newArray = new TagStackEntry[_endTags.Length * 2];
Array.Copy(_endTags, newArray, _endTags.Length);
_endTags = newArray;
}
_endTags[_endTagCount].tagKey = _tagKey;
_endTags[_endTagCount].endTagText= endTag;
_endTagCount++;
}
// This calls filers out all attributes and style attributes by calling OnAttributeRender
// and OnStyleAttributeRender on all properites and updates the lists</para>
protected virtual void FilterAttributes() {
// Create the filtered list of styles
int newStyleCount = 0;
for (int i = 0; i < _styleCount; i++) {
RenderStyle style = _styleList[i];
if (OnStyleAttributeRender(style.name, style.value, style.key)) {
// Update the list. This can be done in place
_styleList[newStyleCount] = style;
newStyleCount++;
}
}
// Update the count
_styleCount = newStyleCount;
// Create the filtered list of attributes
int newAttrCount = 0;
for (int i = 0; i < _attrCount; i++) {
RenderAttribute attr = _attrList[i];
if (OnAttributeRender(attr.name, attr.value, attr.key)) {
// Update the list. This can be done in place
_attrList[newAttrCount] = attr;
newAttrCount++;
}
}
// Update the count
_attrCount = newAttrCount;
}
public virtual void RenderBeginTag(string tagName) {
this.TagName = tagName;
RenderBeginTag(_tagKey);
}
public virtual void RenderBeginTag(HtmlTextWriterTag tagKey) {
this.TagKey = tagKey;
bool renderTag = true;
if (_isDescendant) {
renderTag = OnTagRender(_tagName, _tagKey);
// Inherited renderers will be expecting to be able to filter any of the attributes at this point
FilterAttributes();
// write text before begin tag
string textBeforeTag = RenderBeforeTag();
if (textBeforeTag != null) {
if (tabsPending) {
OutputTabs();
}
writer.Write(textBeforeTag);
}
}
// gather information about this tag.
TagInformation tagInfo = _tagNameLookupArray[_tagIndex];
TagType tagType = tagInfo.tagType;
bool renderEndTag = renderTag && (tagType != TagType.NonClosing);
string endTag = renderEndTag? tagInfo.closingTag : null;
// write the begin tag
if (renderTag) {
if (tabsPending) {
OutputTabs();
}
writer.Write(TagLeftChar);
writer.Write(_tagName);
string styleValue = null;
for (int i = 0; i < _attrCount; i++) {
RenderAttribute attr = _attrList[i];
if (attr.key == HtmlTextWriterAttribute.Style) {
// append style attribute in with other styles
styleValue = attr.value;
}
else {
writer.Write(SpaceChar);
writer.Write(attr.name);
if (attr.value != null) {
writer.Write(EqualsDoubleQuoteString);
string attrValue = attr.value;
if (attr.isUrl) {
if (attr.key != HtmlTextWriterAttribute.Href || !attrValue.StartsWith("javascript:", StringComparison.Ordinal)) {
attrValue = EncodeUrl(attrValue);
}
}
if (attr.encode) {
WriteHtmlAttributeEncode(attrValue);
}
else {
writer.Write(attrValue);
}
writer.Write(DoubleQuoteChar);
}
}
}
if (_styleCount > 0 || styleValue != null) {
writer.Write(SpaceChar);
writer.Write("style");
writer.Write(EqualsDoubleQuoteString);
CssTextWriter.WriteAttributes(writer, _styleList, _styleCount);
if (styleValue != null) {
writer.Write(styleValue);
}
writer.Write(DoubleQuoteChar);
}
if (tagType == TagType.NonClosing) {
writer.Write(SelfClosingTagEnd);
}
else {
writer.Write(TagRightChar);
}
}
string textBeforeContent = RenderBeforeContent();
if (textBeforeContent != null) {
if (tabsPending) {
OutputTabs();
}
writer.Write(textBeforeContent);
}
// write text before the content
if (renderEndTag) {
if (tagType == TagType.Inline) {
_inlineCount += 1;
}
else {
// writeline and indent before rendering content
WriteLine();
Indent++;
}
// Manually build end tags for unknown tag types.
if (endTag == null) {
endTag = EndTagLeftChars + _tagName + TagRightChar.ToString(CultureInfo.InvariantCulture);
}
}
if (_isDescendant) {
// append text after the tag
string textAfterTag = RenderAfterTag();
if (textAfterTag != null) {
endTag = (endTag == null) ? textAfterTag : textAfterTag + endTag;
}
// build end content and push it on stack to write in RenderEndTag
// prepend text after the content
string textAfterContent = RenderAfterContent();
if (textAfterContent != null) {
endTag = (endTag == null) ? textAfterContent : textAfterContent + endTag;
}
}
// push end tag onto stack
PushEndTag(endTag);
// flush attribute and style lists for next tag
_attrCount = 0;
_styleCount = 0;
}
public virtual void RenderEndTag() {
string endTag = PopEndTag();
if (endTag != null) {
if (_tagNameLookupArray[_tagIndex].tagType == TagType.Inline) {
_inlineCount -= 1;
// Never inject crlfs at end of inline tags.
//
Write(endTag);
}
else {
// unindent if not an inline tag
WriteLine();
this.Indent--;
Write(endTag);
}
}
}
protected virtual string RenderBeforeTag() {
return null;
}
protected virtual string RenderBeforeContent() {
return null;
}
protected virtual string RenderAfterContent() {
return null;
}
protected virtual string RenderAfterTag() {
return null;
}
public virtual void WriteAttribute(string name, string value) {
WriteAttribute(name, value, false /*encode*/);
}
public virtual void WriteAttribute(string name, string value, bool fEncode) {
writer.Write(SpaceChar);
writer.Write(name);
if (value != null) {
writer.Write(EqualsDoubleQuoteString);
if (fEncode) {
WriteHtmlAttributeEncode(value);
}
else {
writer.Write(value);
}
writer.Write(DoubleQuoteChar);
}
}
public virtual void WriteBeginTag(string tagName) {
if (tabsPending) {
OutputTabs();
}
writer.Write(TagLeftChar);
writer.Write(tagName);
}
public virtual void WriteBreak() {
// Space between br and / is for improved html compatibility. See XHTML 1.0 specification, section C.2.
Write("<br />");
}
// DevDiv 33149: A backward compat. switch for Everett rendering
internal void WriteObsoleteBreak() {
Write("<br>");
}
public virtual void WriteFullBeginTag(string tagName) {
if (tabsPending) {
OutputTabs();
}
writer.Write(TagLeftChar);
writer.Write(tagName);
writer.Write(TagRightChar);
}
public virtual void WriteEndTag(string tagName) {
if (tabsPending) {
OutputTabs();
}
writer.Write(TagLeftChar);
writer.Write(SlashChar);
writer.Write(tagName);
writer.Write(TagRightChar);
}
public virtual void WriteStyleAttribute(string name, string value) {
WriteStyleAttribute(name, value, false /*encode*/);
}
public virtual void WriteStyleAttribute(string name, string value, bool fEncode) {
writer.Write(name);
writer.Write(StyleEqualsChar);
if (fEncode) {
WriteHtmlAttributeEncode(value);
}
else {
writer.Write(value);
}
writer.Write(SemicolonChar);
}
internal void WriteUTF8ResourceString(IntPtr pv, int offset, int size, bool fAsciiOnly) {
// If we have an http writer, we can use the faster code path. Otherwise,
// get a String out of the resource and write it.
if (_httpWriter != null) {
if (tabsPending) {
OutputTabs();
}
_httpWriter.WriteUTF8ResourceString(pv, offset, size, fAsciiOnly);
}
else {
Write(StringResourceManager.ResourceToString(pv, offset, size));
}
}
public virtual void WriteEncodedUrl(String url) {
int i = url.IndexOf('?');
if (i != -1) {
WriteUrlEncodedString(url.Substring(0, i), false);
Write(url.Substring(i));
}
else {
WriteUrlEncodedString(url, false);
}
}
public virtual void WriteEncodedUrlParameter(String urlText) {
WriteUrlEncodedString(urlText, true);
}
public virtual void WriteEncodedText(String text) {
if (text == null) {
throw new ArgumentNullException("text");
}
const char NBSP = '\u00A0';
// When inner text is retrieved for a text control, &nbsp; is
// decoded to 0x00A0 (code point for nbsp in Unicode).
// HtmlEncode doesn't encode 0x00A0 to &nbsp;, we need to do it
// manually here.
int length = text.Length;
int pos = 0;
while (pos < length) {
int nbsp = text.IndexOf(NBSP, pos);
if (nbsp < 0) {
HttpUtility.HtmlEncode(pos == 0 ? text : text.Substring(pos, length - pos), this);
pos = length;
}
else {
if (nbsp > pos) {
HttpUtility.HtmlEncode(text.Substring(pos, nbsp - pos), this);
}
Write("&nbsp;");
pos = nbsp + 1;
}
}
}
protected void WriteUrlEncodedString(String text, bool argument) {
int length = text.Length;
for (int i = 0; i < length; i++) {
char ch = text[i];
if (HttpEncoderUtility.IsUrlSafeChar(ch)) {
Write(ch);
}
else if ( !argument &&
(ch == '/' ||
ch == ':' ||
ch == '#' ||
ch == ','
)
) {
Write(ch);
}
else if (ch == ' ' && argument) {
Write('+');
}
// for chars that their code number is less than 128 and have
// not been handled above
else if ((ch & 0xff80) == 0) {
Write('%');
Write(HttpEncoderUtility.IntToHex((ch >> 4) & 0xf));
Write(HttpEncoderUtility.IntToHex((ch) & 0xf));
}
else {
// VSWhidbey 448625: For DBCS characters, use UTF8 encoding
// which can be handled by IIS5 and above.
Write(HttpUtility.UrlEncodeNonAscii(Char.ToString(ch), Encoding.UTF8));
}
}
}
internal void WriteHtmlAttributeEncode(string s) {
HttpUtility.HtmlAttributeEncode(s, _httpWriter ?? writer);
}
internal class Layout {
private bool _wrap;
private HorizontalAlign _align;
/*
public Layout(Layout currentLayout) {
Align = currentLayout.Align;
Wrap = currentLayout.Wrap;
}
*/
public Layout(HorizontalAlign alignment, bool wrapping) {
Align = alignment;
Wrap = wrapping;
}
public bool Wrap
{
get
{
return _wrap;
}
set
{
_wrap = value;
}
}
public HorizontalAlign Align
{
get
{
return _align;
}
set
{
_align = value;
}
}
/*
public bool Compare(Layout layout) {
return Wrap == layout.Wrap && Align == layout.Align;
}
public void MergeWith(Layout layout) {
if (Align == HorizontalAlign.NotSet) {
Align = layout.Align;
}
if (Wrap) {
Wrap = layout.Wrap;
}
}
*/
}
private struct TagStackEntry {
public HtmlTextWriterTag tagKey;
public string endTagText;
}
private struct RenderAttribute {
public string name;
public string value;
public HtmlTextWriterAttribute key;
public bool encode;
public bool isUrl;
}
private struct AttributeInformation {
public string name;
public bool isUrl;
public bool encode;
public AttributeInformation(string name, bool encode, bool isUrl) {
this.name = name;
this.encode = encode;
this.isUrl = isUrl;
}
}
private enum TagType {
Inline,
NonClosing,
Other,
}
private struct TagInformation {
public string name;
public TagType tagType;
public string closingTag;
public TagInformation(string name, TagType tagType, string closingTag) {
this.name = name;
this.tagType = tagType;
this.closingTag = closingTag;
}
}
}
}