e79aa3c0ed
Former-commit-id: a2155e9bd80020e49e72e86c44da02a8ac0e57a4
623 lines
20 KiB
C#
623 lines
20 KiB
C#
//------------------------------------------------------------------------------
|
|
// <copyright file="ReaderOutput.cs" company="Microsoft">
|
|
// Copyright (c) Microsoft Corporation. All rights reserved.
|
|
// </copyright>
|
|
// <owner current="true" primary="true">[....]</owner>
|
|
//------------------------------------------------------------------------------
|
|
|
|
namespace System.Xml.Xsl.XsltOld {
|
|
using Res = System.Xml.Utils.Res;
|
|
using System;
|
|
using System.Globalization;
|
|
using System.Diagnostics;
|
|
using System.IO;
|
|
using System.Text;
|
|
using System.Xml;
|
|
using System.Xml.XPath;
|
|
using System.Collections;
|
|
|
|
internal class ReaderOutput : XmlReader, RecordOutput {
|
|
private Processor processor;
|
|
private XmlNameTable nameTable;
|
|
|
|
// Main node + Fields Collection
|
|
private RecordBuilder builder;
|
|
private BuilderInfo mainNode;
|
|
private ArrayList attributeList;
|
|
private int attributeCount;
|
|
private BuilderInfo attributeValue;
|
|
|
|
// OutputScopeManager
|
|
private OutputScopeManager manager;
|
|
|
|
// Current position in the list
|
|
private int currentIndex;
|
|
private BuilderInfo currentInfo;
|
|
|
|
// Reader state
|
|
private ReadState state = ReadState.Initial;
|
|
private bool haveRecord;
|
|
|
|
// Static default record
|
|
static BuilderInfo s_DefaultInfo = new BuilderInfo();
|
|
|
|
XmlEncoder encoder = new XmlEncoder();
|
|
XmlCharType xmlCharType = XmlCharType.Instance;
|
|
|
|
internal ReaderOutput(Processor processor) {
|
|
Debug.Assert(processor != null);
|
|
Debug.Assert(processor.NameTable != null);
|
|
|
|
this.processor = processor;
|
|
this.nameTable = processor.NameTable;
|
|
|
|
Reset();
|
|
}
|
|
|
|
// XmlReader abstract methods implementation
|
|
public override XmlNodeType NodeType {
|
|
get {
|
|
CheckCurrentInfo();
|
|
return this.currentInfo.NodeType;
|
|
}
|
|
}
|
|
|
|
public override string Name {
|
|
get {
|
|
CheckCurrentInfo();
|
|
string prefix = Prefix;
|
|
string localName = LocalName;
|
|
|
|
if (prefix != null && prefix.Length > 0) {
|
|
if (localName.Length > 0) {
|
|
return nameTable.Add(prefix + ":" + localName);
|
|
}
|
|
else {
|
|
return prefix;
|
|
}
|
|
}
|
|
else {
|
|
return localName;
|
|
}
|
|
}
|
|
}
|
|
|
|
public override string LocalName {
|
|
get {
|
|
CheckCurrentInfo();
|
|
return this.currentInfo.LocalName;
|
|
}
|
|
}
|
|
|
|
public override string NamespaceURI {
|
|
get {
|
|
CheckCurrentInfo();
|
|
return this.currentInfo.NamespaceURI;
|
|
}
|
|
}
|
|
|
|
public override string Prefix {
|
|
get {
|
|
CheckCurrentInfo();
|
|
return this.currentInfo.Prefix;
|
|
}
|
|
}
|
|
|
|
public override bool HasValue {
|
|
get {
|
|
return XmlReader.HasValueInternal(NodeType);
|
|
}
|
|
}
|
|
|
|
public override string Value {
|
|
get {
|
|
CheckCurrentInfo();
|
|
return this.currentInfo.Value;
|
|
}
|
|
}
|
|
|
|
public override int Depth {
|
|
get {
|
|
CheckCurrentInfo();
|
|
return this.currentInfo.Depth;
|
|
}
|
|
}
|
|
|
|
public override string BaseURI {
|
|
get {
|
|
return string.Empty;
|
|
}
|
|
}
|
|
|
|
public override bool IsEmptyElement {
|
|
get {
|
|
CheckCurrentInfo();
|
|
return this.currentInfo.IsEmptyTag;
|
|
}
|
|
}
|
|
|
|
public override char QuoteChar {
|
|
get { return encoder.QuoteChar; }
|
|
}
|
|
|
|
public override bool IsDefault {
|
|
get { return false; }
|
|
}
|
|
|
|
public override XmlSpace XmlSpace {
|
|
get { return this.manager != null ? this.manager.XmlSpace : XmlSpace.None; }
|
|
}
|
|
|
|
public override string XmlLang {
|
|
get { return this.manager != null ? this.manager.XmlLang : string.Empty; }
|
|
}
|
|
|
|
// Attribute Accessors
|
|
|
|
public override int AttributeCount {
|
|
get { return this.attributeCount; }
|
|
}
|
|
|
|
public override string GetAttribute(string name) {
|
|
int ordinal;
|
|
if (FindAttribute(name, out ordinal)) {
|
|
Debug.Assert(ordinal >= 0);
|
|
return((BuilderInfo)this.attributeList[ordinal]).Value;
|
|
}
|
|
else {
|
|
Debug.Assert(ordinal == -1);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public override string GetAttribute(string localName, string namespaceURI) {
|
|
int ordinal;
|
|
if (FindAttribute(localName, namespaceURI, out ordinal)) {
|
|
Debug.Assert(ordinal >= 0);
|
|
return((BuilderInfo)this.attributeList[ordinal]).Value;
|
|
}
|
|
else {
|
|
Debug.Assert(ordinal == -1);
|
|
return null;
|
|
}
|
|
}
|
|
|
|
public override string GetAttribute(int i) {
|
|
BuilderInfo attribute = GetBuilderInfo(i);
|
|
return attribute.Value;
|
|
}
|
|
|
|
public override string this [int i] {
|
|
get { return GetAttribute(i); }
|
|
}
|
|
|
|
public override string this [string name] {
|
|
get { return GetAttribute(name); }
|
|
}
|
|
|
|
public override string this [string name, string namespaceURI] {
|
|
get { return GetAttribute(name, namespaceURI); }
|
|
}
|
|
|
|
public override bool MoveToAttribute(string name) {
|
|
int ordinal;
|
|
if (FindAttribute(name, out ordinal)) {
|
|
Debug.Assert(ordinal >= 0);
|
|
SetAttribute(ordinal);
|
|
return true;
|
|
}
|
|
else {
|
|
Debug.Assert(ordinal == -1);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public override bool MoveToAttribute(string localName, string namespaceURI) {
|
|
int ordinal;
|
|
if (FindAttribute(localName, namespaceURI, out ordinal)) {
|
|
Debug.Assert(ordinal >= 0);
|
|
SetAttribute(ordinal);
|
|
return true;
|
|
}
|
|
else {
|
|
Debug.Assert(ordinal == -1);
|
|
return false;
|
|
}
|
|
}
|
|
|
|
public override void MoveToAttribute(int i) {
|
|
if (i < 0 || this.attributeCount <= i) {
|
|
throw new ArgumentOutOfRangeException("i");
|
|
}
|
|
SetAttribute(i);
|
|
}
|
|
|
|
public override bool MoveToFirstAttribute() {
|
|
if (this.attributeCount <= 0) {
|
|
Debug.Assert(this.attributeCount == 0);
|
|
return false;
|
|
}
|
|
else {
|
|
SetAttribute(0);
|
|
return true;
|
|
}
|
|
}
|
|
|
|
public override bool MoveToNextAttribute() {
|
|
if (this.currentIndex + 1 < this.attributeCount) {
|
|
SetAttribute(this.currentIndex + 1);
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
public override bool MoveToElement() {
|
|
if (NodeType == XmlNodeType.Attribute || this.currentInfo == this.attributeValue) {
|
|
SetMainNode();
|
|
return true;
|
|
}
|
|
return false;
|
|
}
|
|
|
|
// Moving through the Stream
|
|
|
|
public override bool Read() {
|
|
Debug.Assert(this.processor != null || this.state == ReadState.Closed);
|
|
|
|
if (this.state != ReadState.Interactive) {
|
|
if (this.state == ReadState.Initial) {
|
|
state = ReadState.Interactive;
|
|
}
|
|
else {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
while (true) { // while -- to ignor empty whitespace nodes.
|
|
if (this.haveRecord) {
|
|
this.processor.ResetOutput();
|
|
this.haveRecord = false;
|
|
}
|
|
|
|
this.processor.Execute();
|
|
|
|
if (this.haveRecord) {
|
|
CheckCurrentInfo();
|
|
// check text nodes on whitespaces;
|
|
switch (this.NodeType) {
|
|
case XmlNodeType.Text :
|
|
if (xmlCharType.IsOnlyWhitespace(this.Value)) {
|
|
this.currentInfo.NodeType = XmlNodeType.Whitespace;
|
|
goto case XmlNodeType.Whitespace;
|
|
}
|
|
Debug.Assert(this.Value.Length != 0, "It whould be Whitespace in this case");
|
|
break;
|
|
case XmlNodeType.Whitespace :
|
|
if(this.Value.Length == 0) {
|
|
continue; // ignoring emty text nodes
|
|
}
|
|
if (this.XmlSpace == XmlSpace.Preserve) {
|
|
this.currentInfo.NodeType = XmlNodeType.SignificantWhitespace;
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
else {
|
|
Debug.Assert(this.processor.ExecutionDone);
|
|
this.state = ReadState.EndOfFile;
|
|
Reset();
|
|
}
|
|
|
|
return this.haveRecord;
|
|
}
|
|
}
|
|
|
|
public override bool EOF {
|
|
get { return this.state == ReadState.EndOfFile; }
|
|
}
|
|
|
|
public override void Close() {
|
|
this.processor = null;
|
|
this.state = ReadState.Closed;
|
|
Reset();
|
|
}
|
|
|
|
public override ReadState ReadState {
|
|
get { return this.state; }
|
|
}
|
|
|
|
// Whole Content Read Methods
|
|
public override string ReadString() {
|
|
string result = string.Empty;
|
|
|
|
if (NodeType == XmlNodeType.Element || NodeType == XmlNodeType.Attribute || this.currentInfo == this.attributeValue) {
|
|
if(this.mainNode.IsEmptyTag) {
|
|
return result;
|
|
}
|
|
if (! Read()) {
|
|
throw new InvalidOperationException(Res.GetString(Res.Xml_InvalidOperation));
|
|
}
|
|
}
|
|
|
|
StringBuilder sb = null;
|
|
bool first = true;
|
|
|
|
while(true) {
|
|
switch (NodeType) {
|
|
case XmlNodeType.Text:
|
|
case XmlNodeType.Whitespace:
|
|
case XmlNodeType.SignificantWhitespace:
|
|
// case XmlNodeType.CharacterEntity:
|
|
if (first) {
|
|
result = this.Value;
|
|
first = false;
|
|
} else {
|
|
if (sb == null) {
|
|
sb = new StringBuilder(result);
|
|
}
|
|
sb.Append(this.Value);
|
|
}
|
|
if (! Read())
|
|
throw new InvalidOperationException(Res.GetString(Res.Xml_InvalidOperation));
|
|
break;
|
|
default:
|
|
return (sb == null) ? result : sb.ToString();
|
|
}
|
|
}
|
|
}
|
|
|
|
public override string ReadInnerXml() {
|
|
if (ReadState == ReadState.Interactive) {
|
|
if (NodeType == XmlNodeType.Element && ! IsEmptyElement) {
|
|
StringOutput output = new StringOutput(this.processor);
|
|
output.OmitXmlDecl();
|
|
int depth = Depth;
|
|
|
|
Read(); // skeep begin Element
|
|
while (depth < Depth) { // process content
|
|
Debug.Assert(this.builder != null);
|
|
output.RecordDone(this.builder);
|
|
Read();
|
|
}
|
|
Debug.Assert(NodeType == XmlNodeType.EndElement);
|
|
Read(); // skeep end element
|
|
|
|
output.TheEnd();
|
|
return output.Result;
|
|
}
|
|
else if(NodeType == XmlNodeType.Attribute) {
|
|
return encoder.AtributeInnerXml(Value);
|
|
}
|
|
else {
|
|
Read();
|
|
}
|
|
}
|
|
return string.Empty;
|
|
}
|
|
|
|
public override string ReadOuterXml() {
|
|
if (ReadState == ReadState.Interactive) {
|
|
if (NodeType == XmlNodeType.Element) {
|
|
StringOutput output = new StringOutput(this.processor);
|
|
output.OmitXmlDecl();
|
|
bool emptyElement = IsEmptyElement;
|
|
int depth = Depth;
|
|
// process current record
|
|
output.RecordDone(this.builder);
|
|
Read();
|
|
// process internal elements & text nodes
|
|
while(depth < Depth) {
|
|
Debug.Assert(this.builder != null);
|
|
output.RecordDone(this.builder);
|
|
Read();
|
|
}
|
|
// process end element
|
|
if (! emptyElement) {
|
|
output.RecordDone(this.builder);
|
|
Read();
|
|
}
|
|
|
|
output.TheEnd();
|
|
return output.Result;
|
|
}
|
|
else if(NodeType == XmlNodeType.Attribute) {
|
|
return encoder.AtributeOuterXml(Name, Value);
|
|
}
|
|
else {
|
|
Read();
|
|
}
|
|
}
|
|
return string.Empty;
|
|
}
|
|
|
|
//
|
|
// Nametable and Namespace Helpers
|
|
//
|
|
|
|
public override XmlNameTable NameTable {
|
|
get {
|
|
Debug.Assert(this.nameTable != null);
|
|
return this.nameTable;
|
|
}
|
|
}
|
|
|
|
public override string LookupNamespace(string prefix) {
|
|
prefix = this.nameTable.Get(prefix);
|
|
|
|
if (this.manager != null && prefix != null) {
|
|
return this.manager.ResolveNamespace(prefix);
|
|
}
|
|
return null;
|
|
}
|
|
|
|
public override void ResolveEntity() {
|
|
Debug.Assert(NodeType != XmlNodeType.EntityReference);
|
|
|
|
if (NodeType != XmlNodeType.EntityReference) {
|
|
throw new InvalidOperationException(Res.GetString(Res.Xml_InvalidOperation));
|
|
}
|
|
}
|
|
|
|
public override bool ReadAttributeValue() {
|
|
if (ReadState != ReadState.Interactive || NodeType != XmlNodeType.Attribute) {
|
|
return false;
|
|
}
|
|
|
|
if (this.attributeValue == null) {
|
|
this.attributeValue = new BuilderInfo();
|
|
this.attributeValue.NodeType = XmlNodeType.Text;
|
|
}
|
|
if (this.currentInfo == this.attributeValue) {
|
|
return false;
|
|
}
|
|
|
|
this.attributeValue.Value = this.currentInfo.Value;
|
|
this.attributeValue.Depth = this.currentInfo.Depth + 1;
|
|
this.currentInfo = this.attributeValue;
|
|
|
|
return true;
|
|
}
|
|
|
|
//
|
|
// RecordOutput interface method implementation
|
|
//
|
|
|
|
public Processor.OutputResult RecordDone(RecordBuilder record) {
|
|
this.builder = record;
|
|
this.mainNode = record.MainNode;
|
|
this.attributeList = record.AttributeList;
|
|
this.attributeCount = record.AttributeCount;
|
|
this.manager = record.Manager;
|
|
|
|
this.haveRecord = true;
|
|
SetMainNode();
|
|
|
|
return Processor.OutputResult.Interrupt;
|
|
}
|
|
|
|
public void TheEnd() {
|
|
// nothing here, was taken care of by RecordBuilder
|
|
}
|
|
|
|
//
|
|
// Implementation internals
|
|
//
|
|
|
|
private void SetMainNode() {
|
|
this.currentIndex = -1;
|
|
this.currentInfo = this.mainNode;
|
|
}
|
|
|
|
private void SetAttribute(int attrib) {
|
|
Debug.Assert(0 <= attrib && attrib < this.attributeCount);
|
|
Debug.Assert(0 <= attrib && attrib < this.attributeList.Count);
|
|
Debug.Assert(this.attributeList[attrib] is BuilderInfo);
|
|
|
|
this.currentIndex = attrib;
|
|
this.currentInfo = (BuilderInfo) this.attributeList[attrib];
|
|
}
|
|
|
|
private BuilderInfo GetBuilderInfo(int attrib) {
|
|
if (attrib < 0 || this.attributeCount <= attrib) {
|
|
throw new ArgumentOutOfRangeException("attrib");
|
|
}
|
|
|
|
Debug.Assert(this.attributeList[attrib] is BuilderInfo);
|
|
|
|
return(BuilderInfo) this.attributeList[attrib];
|
|
}
|
|
|
|
private bool FindAttribute(String localName, String namespaceURI, out int attrIndex) {
|
|
if (namespaceURI == null) {
|
|
namespaceURI = string.Empty;
|
|
}
|
|
if (localName == null) {
|
|
localName = string.Empty;
|
|
}
|
|
|
|
for (int index = 0; index < this.attributeCount; index ++) {
|
|
Debug.Assert(this.attributeList[index] is BuilderInfo);
|
|
|
|
BuilderInfo attribute = (BuilderInfo) this.attributeList[index];
|
|
if (attribute.NamespaceURI == namespaceURI && attribute.LocalName == localName) {
|
|
attrIndex = index;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
attrIndex = -1;
|
|
return false;
|
|
}
|
|
|
|
private bool FindAttribute(String name, out int attrIndex) {
|
|
if (name == null) {
|
|
name = string.Empty;
|
|
}
|
|
|
|
for (int index = 0; index < this.attributeCount; index ++) {
|
|
Debug.Assert(this.attributeList[index] is BuilderInfo);
|
|
|
|
BuilderInfo attribute = (BuilderInfo) this.attributeList[index];
|
|
if (attribute.Name == name) {
|
|
attrIndex = index;
|
|
return true;
|
|
}
|
|
}
|
|
|
|
attrIndex = -1;
|
|
return false;
|
|
}
|
|
|
|
private void Reset() {
|
|
this.currentIndex = -1;
|
|
this.currentInfo = s_DefaultInfo;
|
|
this.mainNode = s_DefaultInfo;
|
|
this.manager = null;
|
|
}
|
|
|
|
[System.Diagnostics.Conditional("DEBUG")]
|
|
private void CheckCurrentInfo() {
|
|
Debug.Assert(this.currentInfo != null);
|
|
Debug.Assert(this.attributeCount == 0 || this.attributeList != null);
|
|
Debug.Assert((this.currentIndex == -1) == (this.currentInfo == this.mainNode));
|
|
Debug.Assert((this.currentIndex == -1) || (this.currentInfo == this.attributeValue || this.attributeList[this.currentIndex] is BuilderInfo && this.attributeList[this.currentIndex] == this.currentInfo));
|
|
}
|
|
|
|
private class XmlEncoder {
|
|
private StringBuilder buffer = null;
|
|
private XmlTextEncoder encoder = null;
|
|
|
|
private void Init() {
|
|
buffer = new StringBuilder();
|
|
encoder = new XmlTextEncoder(new StringWriter(buffer, CultureInfo.InvariantCulture));
|
|
}
|
|
|
|
public string AtributeInnerXml(string value) {
|
|
if(encoder == null) Init();
|
|
buffer .Length = 0; // clean buffer
|
|
encoder.StartAttribute(/*save:*/false);
|
|
encoder.Write(value);
|
|
encoder.EndAttribute();
|
|
return buffer.ToString();
|
|
}
|
|
|
|
public string AtributeOuterXml(string name, string value) {
|
|
if(encoder == null) Init();
|
|
buffer .Length = 0; // clean buffer
|
|
buffer .Append(name);
|
|
buffer .Append('=');
|
|
buffer .Append(QuoteChar);
|
|
encoder.StartAttribute(/*save:*/false);
|
|
encoder.Write(value);
|
|
encoder.EndAttribute();
|
|
buffer .Append(QuoteChar);
|
|
return buffer.ToString();
|
|
}
|
|
|
|
public char QuoteChar {
|
|
get { return '"'; }
|
|
}
|
|
}
|
|
}
|
|
}
|