2 // System.Xml.Serialization.SerializationCodeGenerator.cs:
5 // Lluis Sanchez Gual (lluis@ximian.com)
6 // Atsushi Enomoto (atsushi@ximian.com)
8 // (C) 2002, 2003 Ximian, Inc. http://www.ximian.com
9 // Copyright (C) 2006 Novell, Inc.
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:
21 // The above copyright notice and this permission notice shall be
22 // included in all copies or substantial portions of the Software.
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.
34 using System.Reflection;
35 using System.Xml.Serialization;
36 using System.Collections;
37 using System.Globalization;
40 using HookDir = System.Xml.Serialization.XmlMappingAccess;
42 namespace System.Xml.Serialization
44 internal class SerializationCodeGenerator
47 SerializationFormat _format;
51 Hashtable _uniqueNames = new Hashtable();
53 SerializerInfo _config;
54 ArrayList _mapsToGenerate = new ArrayList ();
55 ArrayList _fixupCallbacks;
56 ArrayList _referencedTypes = new ArrayList ();
57 GenerationResult[] _results;
58 GenerationResult _result;
59 XmlMapping[] _xmlMaps;
61 CodeIdentifiers classNames = new CodeIdentifiers ();
63 public SerializationCodeGenerator (XmlMapping[] xmlMaps): this (xmlMaps, null)
67 public SerializationCodeGenerator (XmlMapping[] xmlMaps, SerializerInfo config)
73 public SerializationCodeGenerator (XmlMapping xmlMap, SerializerInfo config)
75 _xmlMaps = new XmlMapping [] {xmlMap};
79 public static void Generate (string configFileName, string outputPath)
81 SerializationCodeGeneratorConfiguration cnf = null;
82 StreamReader sr = new StreamReader (configFileName);
85 XmlReflectionImporter ri = new XmlReflectionImporter ();
86 ri.AllowPrivateTypes = true;
87 XmlSerializer ser = new XmlSerializer (ri.ImportTypeMapping (typeof (SerializationCodeGeneratorConfiguration)));
88 cnf = (SerializationCodeGeneratorConfiguration) ser.Deserialize (sr);
95 if (outputPath == null) outputPath = "";
97 CodeIdentifiers ids = new CodeIdentifiers ();
98 if (cnf.Serializers != null)
100 foreach (SerializerInfo info in cnf.Serializers)
103 if (info.Assembly != null)
107 asm = Assembly.Load (info.Assembly);
109 asm = Assembly.LoadFrom (info.Assembly);
111 type = asm.GetType (info.ClassName, true);
114 type = Type.GetType (info.ClassName);
116 if (type == null) throw new InvalidOperationException ("Type " + info.ClassName + " not found");
118 string file = info.OutFileName;
119 if (file == null || file.Length == 0) {
120 int i = info.ClassName.LastIndexOf (".");
121 if (i != -1) file = info.ClassName.Substring (i+1);
122 else file = info.ClassName;
123 file = ids.AddUnique (file, type) + "Serializer.cs";
125 StreamWriter writer = new StreamWriter (Path.Combine (outputPath, file));
131 if (info.SerializationFormat == SerializationFormat.Literal) {
132 XmlReflectionImporter ri = new XmlReflectionImporter ();
133 map = ri.ImportTypeMapping (type);
136 SoapReflectionImporter ri = new SoapReflectionImporter ();
137 map = ri.ImportTypeMapping (type);
140 SerializationCodeGenerator gen = new SerializationCodeGenerator (map, info);
141 gen.GenerateSerializers (writer);
151 public void GenerateSerializers (TextWriter writer)
154 _results = new GenerationResult [_xmlMaps.Length];
156 WriteLine ("// It is automatically generated");
157 WriteLine ("using System;");
158 WriteLine ("using System.Xml;");
159 WriteLine ("using System.Xml.Schema;");
160 WriteLine ("using System.Xml.Serialization;");
161 WriteLine ("using System.Text;");
162 WriteLine ("using System.Collections;");
163 WriteLine ("using System.Globalization;");
164 if (_config != null && _config.NamespaceImports != null && _config.NamespaceImports.Length > 0) {
165 foreach (string ns in _config.NamespaceImports)
166 WriteLine ("using " + ns + ";");
170 string readerClassName = null;
171 string writerClassName = null;
172 string baseClassName = null;
173 string implClassName = null;
174 string namspace = null;
178 readerClassName = _config.ReaderClassName;
179 writerClassName = _config.WriterClassName;
180 baseClassName = _config.BaseSerializerClassName;
181 implClassName = _config.ImplementationClassName;
182 namspace = _config.Namespace;
185 if (readerClassName == null || readerClassName.Length == 0)
186 readerClassName = "GeneratedReader";
188 if (writerClassName == null || writerClassName.Length == 0)
189 writerClassName = "GeneratedWriter";
190 if (baseClassName == null || baseClassName.Length == 0)
191 baseClassName = "BaseXmlSerializer";
192 if (implClassName == null || implClassName.Length == 0)
193 implClassName = "XmlSerializerContract";
194 readerClassName = GetUniqueClassName (readerClassName);
195 writerClassName = GetUniqueClassName (writerClassName);
196 baseClassName = GetUniqueClassName (baseClassName);
197 implClassName = GetUniqueClassName (implClassName);
198 Hashtable mapsByNamespace = new Hashtable ();
199 Hashtable generatedMaps = new Hashtable ();
201 for (int n=0; n<_xmlMaps.Length; n++)
203 _typeMap = _xmlMaps [n];
204 if (_typeMap == null) continue;
206 _result = generatedMaps [_typeMap] as GenerationResult;
207 if (_result != null) {
208 _results[n] = _result;
212 _result = new GenerationResult ();
213 _results[n] = _result;
215 generatedMaps [_typeMap] = _result;
218 if (_typeMap is XmlTypeMapping) typeName = CodeIdentifier.MakeValid (((XmlTypeMapping)_typeMap).TypeData.CSharpName);
219 else typeName = ((XmlMembersMapping)_typeMap).ElementName;
221 _result.ReaderClassName = readerClassName;
222 _result.WriterClassName = writerClassName;
223 _result.BaseSerializerClassName = baseClassName;
224 _result.ImplementationClassName = implClassName;
226 if (namspace == null || namspace.Length == 0)
227 _result.Namespace = "Mono.GeneratedSerializers." + _typeMap.Format;
229 _result.Namespace = namspace;
231 _result.WriteMethodName = GetUniqueName ("rwo", _typeMap, "WriteRoot_" + typeName);
232 _result.ReadMethodName = GetUniqueName ("rro", _typeMap, "ReadRoot_" + typeName);
234 _result.Mapping = _typeMap;
236 ArrayList maps = (ArrayList) mapsByNamespace [_result.Namespace];
238 maps = new ArrayList ();
239 mapsByNamespace [_result.Namespace] = maps;
244 foreach (DictionaryEntry entry in mapsByNamespace)
246 ArrayList maps = (ArrayList) entry.Value;
248 WriteLine ("namespace " + entry.Key);
251 if (_config == null || !_config.NoReader)
252 GenerateReader (readerClassName, maps);
254 if (_config == null || !_config.NoWriter)
255 GenerateWriter (writerClassName, maps);
259 GenerateContract (maps);
267 public GenerationResult[] GenerationResults
269 get { return _results; }
272 public ArrayList ReferencedTypes
274 get { return _referencedTypes; }
277 void UpdateGeneratedTypes (ArrayList list)
279 for (int n=0; n<list.Count; n++)
281 XmlTypeMapping map = list[n] as XmlTypeMapping;
282 if (map != null && !_referencedTypes.Contains (map.TypeData.Type))
283 _referencedTypes.Add (map.TypeData.Type);
287 static string ToCSharpFullName (Type type)
289 return TypeData.ToCSharpName (type, true);
292 #region Writer Generation
294 //*******************************************************
295 // Contract generation
299 public void GenerateContract (ArrayList generatedMaps)
301 // Write the base serializer
303 if (generatedMaps.Count == 0) return;
305 GenerationResult main = (GenerationResult) generatedMaps[0];
307 string baseSerializerName = main.BaseSerializerClassName;
308 string access_mod = (_config == null || !_config.GenerateAsInternal) ? "public" : "internal";
311 WriteLine (access_mod + " class " + baseSerializerName + " : System.Xml.Serialization.XmlSerializer");
313 WriteLineInd ("protected override System.Xml.Serialization.XmlSerializationReader CreateReader () {");
314 WriteLine ("return new " + main.ReaderClassName + " ();");
318 WriteLineInd ("protected override System.Xml.Serialization.XmlSerializationWriter CreateWriter () {");
319 WriteLine ("return new " + main.WriterClassName + " ();");
323 WriteLineInd ("public override bool CanDeserialize (System.Xml.XmlReader xmlReader) {");
324 WriteLine ("return true;");
330 // Write a serializer for each imported map
332 foreach (GenerationResult res in generatedMaps)
334 res.SerializerClassName = GetUniqueClassName (res.Mapping.ElementName + "Serializer");
336 WriteLine (access_mod + " sealed class " + res.SerializerClassName + " : " + baseSerializerName);
338 WriteLineInd ("protected override void Serialize (object obj, System.Xml.Serialization.XmlSerializationWriter writer) {");
339 WriteLine ("((" + res.WriterClassName + ")writer)." + res.WriteMethodName + "(obj);");
343 WriteLineInd ("protected override object Deserialize (System.Xml.Serialization.XmlSerializationReader reader) {");
344 WriteLine ("return ((" + res.ReaderClassName + ")reader)." + res.ReadMethodName + "();");
351 WriteLine ("#if !TARGET_JVM"); // does it make sense? ;-)
352 WriteLine (access_mod + " class " + main.ImplementationClassName + " : System.Xml.Serialization.XmlSerializerImplementation");
355 WriteLine ("System.Collections.Hashtable readMethods = null;");
356 WriteLine ("System.Collections.Hashtable writeMethods = null;");
357 WriteLine ("System.Collections.Hashtable typedSerializers = null;");
360 WriteLineInd ("public override System.Xml.Serialization.XmlSerializationReader Reader {");
361 WriteLineInd ("get {");
362 WriteLine ("return new " + main.ReaderClassName + "();");
367 WriteLineInd ("public override System.Xml.Serialization.XmlSerializationWriter Writer {");
368 WriteLineInd ("get {");
369 WriteLine ("return new " + main.WriterClassName + "();");
374 WriteLineInd ("public override System.Collections.Hashtable ReadMethods {");
375 WriteLineInd ("get {");
376 WriteLineInd ("lock (this) {");
377 WriteLineInd ("if (readMethods == null) {");
378 WriteLine ("readMethods = new System.Collections.Hashtable ();");
379 foreach (GenerationResult res in generatedMaps)
380 WriteLine ("readMethods.Add (@\"" + res.Mapping.GetKey () + "\", @\"" + res.ReadMethodName + "\");");
382 WriteLine ("return readMethods;");
388 WriteLineInd ("public override System.Collections.Hashtable WriteMethods {");
389 WriteLineInd ("get {");
390 WriteLineInd ("lock (this) {");
391 WriteLineInd ("if (writeMethods == null) {");
392 WriteLine ("writeMethods = new System.Collections.Hashtable ();");
393 foreach (GenerationResult res in generatedMaps)
394 WriteLine ("writeMethods.Add (@\"" + res.Mapping.GetKey () + "\", @\"" + res.WriteMethodName + "\");");
396 WriteLine ("return writeMethods;");
402 WriteLineInd ("public override System.Collections.Hashtable TypedSerializers {");
403 WriteLineInd ("get {");
404 WriteLineInd ("lock (this) {");
405 WriteLineInd ("if (typedSerializers == null) {");
406 WriteLine ("typedSerializers = new System.Collections.Hashtable ();");
407 foreach (GenerationResult res in generatedMaps)
408 WriteLine ("typedSerializers.Add (@\"" + res.Mapping.GetKey () + "\", new " + res.SerializerClassName + "());");
410 WriteLine ("return typedSerializers;");
416 WriteLine ("public override XmlSerializer GetSerializer (Type type)");
418 WriteLine ("switch (type.FullName) {");
419 foreach (GenerationResult res in generatedMaps) {
420 if (res.Mapping is XmlTypeMapping) {
421 WriteLineInd ("case \"" + ((XmlTypeMapping) res.Mapping).TypeData.CSharpFullName + "\":");
422 WriteLine ("return (XmlSerializer) TypedSerializers [\"" + res.Mapping.GetKey () + "\"];");
427 WriteLine ("return base.GetSerializer (type);");
431 WriteLineInd ("public override bool CanSerialize (System.Type type) {");
432 foreach (GenerationResult res in generatedMaps) {
433 if (res.Mapping is XmlTypeMapping)
434 WriteLine ("if (type == typeof(" + (res.Mapping as XmlTypeMapping).TypeData.CSharpFullName + ")) return true;");
436 WriteLine ("return false;");
441 WriteLine ("#endif");
446 //*******************************************************
450 public void GenerateWriter (string writerClassName, ArrayList maps)
452 _mapsToGenerate = new ArrayList ();
456 if (_config == null || !_config.GenerateAsInternal)
457 WriteLine ("public class " + writerClassName + " : XmlSerializationWriter");
459 WriteLine ("internal class " + writerClassName + " : XmlSerializationWriter");
461 WriteLine ("const string xmlNamespace = \"http://www.w3.org/2000/xmlns/\";");
462 // ToBinHexString() is not public, so use reflection here.
463 WriteLine ("static readonly System.Reflection.MethodInfo toBinHexStringMethod = typeof (XmlConvert).GetMethod (\"ToBinHexString\", System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.NonPublic, null, new Type [] {typeof (byte [])}, null);");
464 WriteLine ("static string ToBinHexString (byte [] input)");
466 WriteLine ("return input == null ? null : (string) toBinHexStringMethod.Invoke (null, new object [] {input});");
469 for (int n=0; n<maps.Count; n++)
471 GenerationResult res = (GenerationResult) maps [n];
472 _typeMap = res.Mapping;
473 _format = _typeMap.Format;
476 GenerateWriteRoot ();
479 for (int n=0; n<_mapsToGenerate.Count; n++)
481 XmlTypeMapping map = (XmlTypeMapping) _mapsToGenerate[n];
482 GenerateWriteObject (map);
483 if (map.TypeData.SchemaType == SchemaTypes.Enum)
484 GenerateGetXmlEnumValue (map);
487 GenerateWriteInitCallbacks ();
488 UpdateGeneratedTypes (_mapsToGenerate);
493 void GenerateWriteRoot ()
495 WriteLine ("public void " +_result.WriteMethodName + " (object o)");
497 WriteLine ("WriteStartDocument ();");
499 if (_typeMap is XmlTypeMapping)
501 WriteLine (GetRootTypeName () + " ob = (" + GetRootTypeName () + ") o;");
502 XmlTypeMapping mp = (XmlTypeMapping) _typeMap;
503 if (mp.TypeData.SchemaType == SchemaTypes.Class || mp.TypeData.SchemaType == SchemaTypes.Array)
504 WriteLine ("TopLevelElement ();");
506 if (_format == SerializationFormat.Literal) {
507 WriteLine (GetWriteObjectName (mp) + " (ob, " + GetLiteral(mp.ElementName) + ", " + GetLiteral(mp.Namespace) + ", true, false, true);");
510 RegisterReferencingMap (mp);
511 WriteLine ("WritePotentiallyReferencingElement (" + GetLiteral(mp.ElementName) + ", " + GetLiteral(mp.Namespace) + ", ob, " + GetTypeOf(mp.TypeData) + ", true, false);");
514 else if (_typeMap is XmlMembersMapping) {
515 WriteLine ("object[] pars = (object[]) o;");
516 GenerateWriteMessage ((XmlMembersMapping) _typeMap);
519 throw new InvalidOperationException ("Unknown type map");
521 if (_format == SerializationFormat.Encoded)
522 WriteLine ("WriteReferencedElements ();");
528 void GenerateWriteMessage (XmlMembersMapping membersMap)
530 if (membersMap.HasWrapperElement) {
531 WriteLine ("TopLevelElement ();");
532 WriteLine ("WriteStartElement (" + GetLiteral (membersMap.ElementName) + ", " + GetLiteral (membersMap.Namespace) + ", (" + GetLiteral(_format == SerializationFormat.Encoded) + "));");
534 /* WriteLineInd ("if (Writer.LookupPrefix (XmlSchema.Namespace) == null)");
535 WriteLine ("WriteAttribute (\"xmlns\",\"xsd\",XmlSchema.Namespace,XmlSchema.Namespace);");
538 WriteLineInd ("if (Writer.LookupPrefix (XmlSchema.InstanceNamespace) == null)");
539 WriteLine ("WriteAttribute (\"xmlns\",\"xsi\",XmlSchema.InstanceNamespace,XmlSchema.InstanceNamespace);");
544 GenerateWriteObjectElement (membersMap, "pars", true);
546 if (membersMap.HasWrapperElement)
547 WriteLine ("WriteEndElement();");
550 void GenerateGetXmlEnumValue (XmlTypeMapping map)
552 EnumMap emap = (EnumMap) map.ObjectMap;
554 string xmlNamesArray = null;
555 string valuesArray = null;
558 // create static string[] holding XML names of enum constants
559 xmlNamesArray = GetUniqueName ("gxe", map, "_xmlNames" + map.XmlType);
560 Write ("static readonly string[] " + xmlNamesArray + " = { ");
561 for (int i = 0; i < emap.XmlNames.Length; i++) {
565 _writer.Write (emap.XmlNames[i]);
566 _writer.Write ('\"');
568 _writer.WriteLine (" };");
570 // create const long[] holding values of enum constants
571 valuesArray = GetUniqueName ("gve", map, "_values" + map.XmlType);
572 Write ("static readonly long[] " + valuesArray + " = { ");
573 for (int i = 0; i < emap.Values.Length; i++) {
576 _writer.Write (emap.Values[i].ToString (CultureInfo.InvariantCulture));
579 _writer.WriteLine (" };");
580 WriteLine (string.Empty);
583 WriteLine ("string " + GetGetEnumValueName (map) + " (" + GetTypeFullName (map.TypeData) + " val)");
587 WriteLineInd ("switch (val) {");
588 for (int i = 0; i < emap.EnumNames.Length; i++)
589 WriteLine ("case " + map.TypeData.CSharpFullName + ".@" + emap.EnumNames[i] + ": return " + GetLiteral (emap.XmlNames[i]) + ";");
592 WriteLineInd ("default:");
593 // FromEnum actually covers this case too, but we save some cycles here
594 WriteLine ("if (val.ToString () == \"0\") return string.Empty;");
595 Write ("return FromEnum ((long) val, " + xmlNamesArray + ", " + valuesArray);
597 _writer.Write (", typeof (");
598 _writer.Write (map.TypeData.CSharpFullName);
599 _writer.Write (").FullName");
601 _writer.Write (')'); // close FromEnum method call
602 WriteUni (";"); // end statement
605 WriteLine ("default: throw CreateInvalidEnumValueException ((long) val, typeof (" + map.TypeData.CSharpFullName + ").FullName);");
607 WriteLine ("default: return ((long)val).ToString(CultureInfo.InvariantCulture);");
617 void GenerateWriteObject (XmlTypeMapping typeMap)
619 WriteLine ("void " + GetWriteObjectName (typeMap) + " (" + GetTypeFullName (typeMap.TypeData) + " ob, string element, string namesp, bool isNullable, bool needType, bool writeWrappingElem)");
624 SetHookVar ("$TYPE", typeMap.TypeData.CSharpName);
625 SetHookVar ("$FULLTYPE", typeMap.TypeData.CSharpFullName);
626 SetHookVar ("$OBJECT", "ob");
627 SetHookVar ("$NULLABLE", "isNullable");
629 if (GenerateWriteHook (HookType.type, typeMap.TypeData.Type))
637 if (!typeMap.TypeData.IsValueType)
639 WriteLine ("if (((object)ob) == null)");
641 WriteLineInd ("if (isNullable)");
643 if (_format == SerializationFormat.Literal)
644 WriteLine ("WriteNullTagLiteral(element, namesp);");
646 WriteLine ("WriteNullTagEncoded (element, namesp);");
648 WriteLineUni ("return;");
653 if (typeMap.TypeData.SchemaType == SchemaTypes.XmlNode)
655 if (_format == SerializationFormat.Literal)
656 WriteLine ("WriteElementLiteral (ob, \"\", \"\", true, " + GetLiteral(typeMap.IsAny) + ");");
658 WriteLine ("WriteElementEncoded (ob, \"\", \"\", true, " + GetLiteral(typeMap.IsAny) + ");");
667 if (typeMap.TypeData.SchemaType == SchemaTypes.XmlSerializable)
669 WriteLine ("WriteSerializable (ob, element, namesp, isNullable, " + GetLiteral(!typeMap.IsAny) + ");");
678 ArrayList types = typeMap.DerivedTypes;
680 WriteLine ("System.Type type = ob.GetType ();");
681 WriteLine ("if (type == typeof(" + typeMap.TypeData.CSharpFullName + "))");
684 for (int n=0; n<types.Count; n++)
686 XmlTypeMapping map = (XmlTypeMapping)types[n];
688 WriteLineInd ("else if (type == typeof(" + map.TypeData.CSharpFullName + ")) { ");
689 WriteLine (GetWriteObjectName (map) + "((" + map.TypeData.CSharpFullName + ")ob, element, namesp, isNullable, true, writeWrappingElem);");
690 WriteLine ("return;");
694 if (typeMap.TypeData.Type == typeof (object)) {
695 WriteLineInd ("else {");
696 WriteLineInd ("if (ob.GetType().IsArray && typeof(XmlNode).IsAssignableFrom(ob.GetType().GetElementType())) {");
697 WriteLine ("Writer.WriteStartElement (" + GetLiteral (typeMap.ElementName) + ", " + GetLiteral (typeMap.Namespace) + ");");
698 WriteLineInd ("foreach (XmlNode node in (System.Collections.IEnumerable) ob)");
699 WriteLineUni ("node.WriteTo (Writer);");
700 WriteLineUni ("Writer.WriteEndElement ();");
702 WriteLineInd ("else");
703 WriteLineUni ("WriteTypedPrimitive (element, namesp, ob, true);");
704 WriteLine ("return;");
709 WriteLineInd ("else {");
710 WriteLine ("throw CreateUnknownTypeException (ob);");
714 WriteLineInd ("if (writeWrappingElem) {");
715 if (_format == SerializationFormat.Encoded) WriteLine ("needType = true;");
716 WriteLine ("WriteStartElement (element, namesp, ob);");
720 WriteLine ("if (needType) WriteXsiType(" + GetLiteral(typeMap.XmlType) + ", " + GetLiteral(typeMap.XmlTypeNamespace) + ");");
723 switch (typeMap.TypeData.SchemaType)
725 case SchemaTypes.Class: GenerateWriteObjectElement (typeMap, "ob", false); break;
726 case SchemaTypes.Array: GenerateWriteListElement (typeMap, "ob"); break;
727 case SchemaTypes.Primitive: GenerateWritePrimitiveElement (typeMap, "ob"); break;
728 case SchemaTypes.Enum: GenerateWriteEnumElement (typeMap, "ob"); break;
731 WriteLine ("if (writeWrappingElem) WriteEndElement (ob);");
740 void GenerateWriteObjectElement (XmlMapping xmlMap, string ob, bool isValueList)
742 XmlTypeMapping typeMap = xmlMap as XmlTypeMapping;
743 Type xmlMapType = (typeMap != null) ? typeMap.TypeData.Type : typeof(object[]);
745 ClassMap map = (ClassMap)xmlMap.ObjectMap;
746 if (!GenerateWriteHook (HookType.attributes, xmlMapType))
748 if (map.NamespaceDeclarations != null) {
749 WriteLine ("WriteNamespaceDeclarations ((XmlSerializerNamespaces) " + ob + ".@" + map.NamespaceDeclarations.Name + ");");
753 XmlTypeMapMember anyAttrMember = map.DefaultAnyAttributeMember;
754 if (anyAttrMember != null)
756 if (!GenerateWriteMemberHook (xmlMapType, anyAttrMember))
758 string cond = GenerateMemberHasValueCondition (anyAttrMember, ob, isValueList);
759 if (cond != null) WriteLineInd ("if (" + cond + ") {");
761 string tmpVar = GetObTempVar ();
762 WriteLine ("ICollection " + tmpVar + " = " + GenerateGetMemberValue (anyAttrMember, ob, isValueList) + ";");
763 WriteLineInd ("if (" + tmpVar + " != null) {");
765 string tmpVar2 = GetObTempVar ();
766 WriteLineInd ("foreach (XmlAttribute " + tmpVar2 + " in " + tmpVar + ")");
767 WriteLineInd ("if (" + tmpVar2 + ".NamespaceURI != xmlNamespace)");
768 WriteLine ("WriteXmlAttribute (" + tmpVar2 + ", " + ob + ");");
773 if (cond != null) WriteLineUni ("}");
779 ICollection attributes = map.AttributeMembers;
780 if (attributes != null)
782 foreach (XmlTypeMapMemberAttribute attr in attributes)
784 if (GenerateWriteMemberHook (xmlMapType, attr)) continue;
786 string val = GenerateGetMemberValue (attr, ob, isValueList);
787 string cond = GenerateMemberHasValueCondition (attr, ob, isValueList);
789 if (cond != null) WriteLineInd ("if (" + cond + ") {");
791 string strVal = GenerateGetStringValue (attr.MappedType, attr.TypeData, val, false);
792 WriteLine ("WriteAttribute (" + GetLiteral(attr.AttributeName) + ", " + GetLiteral(attr.Namespace) + ", " + strVal + ");");
794 if (cond != null) WriteLineUni ("}");
802 if (!GenerateWriteHook (HookType.elements, xmlMapType))
804 ICollection members = map.ElementMembers;
807 foreach (XmlTypeMapMemberElement member in members)
809 if (GenerateWriteMemberHook (xmlMapType, member)) continue;
811 string cond = GenerateMemberHasValueCondition (member, ob, isValueList);
812 if (cond != null) WriteLineInd ("if (" + cond + ") {");
814 string memberValue = GenerateGetMemberValue (member, ob, isValueList);
815 Type memType = member.GetType();
817 if (memType == typeof(XmlTypeMapMemberList))
819 GenerateWriteMemberElement ((XmlTypeMapElementInfo) member.ElementInfo[0], memberValue);
821 else if (memType == typeof(XmlTypeMapMemberFlatList))
823 WriteLineInd ("if (" + memberValue + " != null) {");
824 GenerateWriteListContent (ob, member.TypeData, ((XmlTypeMapMemberFlatList)member).ListMap, memberValue, false);
827 else if (memType == typeof(XmlTypeMapMemberAnyElement))
829 WriteLineInd ("if (" + memberValue + " != null) {");
830 GenerateWriteAnyElementContent ((XmlTypeMapMemberAnyElement)member, memberValue);
833 else if (memType == typeof(XmlTypeMapMemberAnyElement))
835 WriteLineInd ("if (" + memberValue + " != null) {");
836 GenerateWriteAnyElementContent ((XmlTypeMapMemberAnyElement)member, memberValue);
839 else if (memType == typeof(XmlTypeMapMemberAnyAttribute))
843 else if (memType == typeof(XmlTypeMapMemberElement))
845 if (member.ElementInfo.Count == 1) {
846 XmlTypeMapElementInfo elem = (XmlTypeMapElementInfo)member.ElementInfo[0];
847 GenerateWriteMemberElement (elem, GetCast(elem.TypeData, member.TypeData, memberValue));
849 else if (member.ChoiceMember != null)
851 string choiceValue = ob + ".@" + member.ChoiceMember;
852 foreach (XmlTypeMapElementInfo elem in member.ElementInfo) {
853 WriteLineInd ("if (" + choiceValue + " == " + GetLiteral(elem.ChoiceValue) + ") {");
854 GenerateWriteMemberElement (elem, GetCast(elem.TypeData, member.TypeData, memberValue));
860 // WriteLineInd ("if (" + memberValue + " == null) {");
861 // GenerateWriteMemberElement ((XmlTypeMapElementInfo)member.ElementInfo[0], memberValue);
862 // WriteLineUni ("}");
865 foreach (XmlTypeMapElementInfo elem in member.ElementInfo)
867 WriteLineInd ((first?"":"else ") + "if (" + memberValue + " is " + elem.TypeData.CSharpFullName + ") {");
868 GenerateWriteMemberElement (elem, GetCast(elem.TypeData, member.TypeData, memberValue));
875 throw new InvalidOperationException ("Unknown member type");
887 void GenerateWriteMemberElement (XmlTypeMapElementInfo elem, string memberValue)
889 switch (elem.TypeData.SchemaType)
891 case SchemaTypes.XmlNode:
892 string elemName = elem.WrappedElement ? elem.ElementName : "";
893 if (_format == SerializationFormat.Literal)
894 WriteMetCall ("WriteElementLiteral", memberValue, GetLiteral(elemName), GetLiteral(elem.Namespace), GetLiteral(elem.IsNullable), "false");
896 WriteMetCall ("WriteElementEncoded", memberValue, GetLiteral(elemName), GetLiteral(elem.Namespace), GetLiteral(elem.IsNullable), "false");
899 case SchemaTypes.Enum:
900 case SchemaTypes.Primitive:
901 if (_format == SerializationFormat.Literal)
902 GenerateWritePrimitiveValueLiteral (memberValue, elem.ElementName, elem.Namespace, elem.MappedType, elem.TypeData, elem.WrappedElement, elem.IsNullable);
904 GenerateWritePrimitiveValueEncoded (memberValue, elem.ElementName, elem.Namespace, new XmlQualifiedName (elem.TypeData.XmlType, elem.DataTypeNamespace), elem.MappedType, elem.TypeData, elem.WrappedElement, elem.IsNullable);
907 case SchemaTypes.Array:
908 WriteLineInd ("if (" + memberValue + " != null) {");
910 if (elem.MappedType.MultiReferenceType) {
911 WriteMetCall ("WriteReferencingElement", GetLiteral(elem.ElementName), GetLiteral(elem.Namespace), memberValue, GetLiteral(elem.IsNullable));
912 RegisterReferencingMap (elem.MappedType);
915 WriteMetCall ("WriteStartElement", GetLiteral(elem.ElementName), GetLiteral(elem.Namespace), memberValue);
916 GenerateWriteListContent (null, elem.TypeData, (ListMap) elem.MappedType.ObjectMap, memberValue, false);
917 WriteMetCall ("WriteEndElement", memberValue);
921 if (elem.IsNullable) {
922 WriteLineInd ("else");
923 if (_format == SerializationFormat.Literal)
924 WriteMetCall ("WriteNullTagLiteral", GetLiteral(elem.ElementName), GetLiteral(elem.Namespace));
926 WriteMetCall ("WriteNullTagEncoded", GetLiteral(elem.ElementName), GetLiteral(elem.Namespace));
932 case SchemaTypes.Class:
933 if (elem.MappedType.MultiReferenceType) {
934 RegisterReferencingMap (elem.MappedType);
935 if (elem.MappedType.TypeData.Type == typeof(object))
936 WriteMetCall ("WritePotentiallyReferencingElement", GetLiteral(elem.ElementName), GetLiteral(elem.Namespace), memberValue, "null", "false", GetLiteral(elem.IsNullable));
938 WriteMetCall ("WriteReferencingElement", GetLiteral(elem.ElementName), GetLiteral(elem.Namespace), memberValue, GetLiteral(elem.IsNullable));
941 WriteMetCall (GetWriteObjectName(elem.MappedType), memberValue, GetLiteral(elem.ElementName), GetLiteral(elem.Namespace), GetLiteral(elem.IsNullable), "false", "true");
944 case SchemaTypes.XmlSerializable:
945 WriteMetCall ("WriteSerializable", "(" + ToCSharpFullName (elem.MappedType.TypeData.Type) + ") " + memberValue, GetLiteral(elem.ElementName), GetLiteral(elem.Namespace), GetLiteral(elem.IsNullable));
949 throw new NotSupportedException ("Invalid value type");
953 void GenerateWriteListElement (XmlTypeMapping typeMap, string ob)
955 if (_format == SerializationFormat.Encoded)
958 string itemCount = GenerateGetListCount (typeMap.TypeData, ob);
959 GenerateGetArrayType ((ListMap) typeMap.ObjectMap, itemCount, out n, out ns);
962 if (ns != string.Empty)
963 arrayType = "FromXmlQualifiedName (new XmlQualifiedName(" + n + "," + ns + "))";
965 arrayType = GetLiteral (n);
967 WriteMetCall ("WriteAttribute", GetLiteral("arrayType"), GetLiteral(XmlSerializer.EncodingNamespace), arrayType);
969 GenerateWriteListContent (null, typeMap.TypeData, (ListMap) typeMap.ObjectMap, ob, false);
972 void GenerateWriteAnyElementContent (XmlTypeMapMemberAnyElement member, string memberValue)
974 bool singleElement = (member.TypeData.Type == typeof (XmlElement) || member.TypeData.Type == typeof (XmlNode));
977 var = GetObTempVar ();
981 var2 = GetObTempVar ();
982 WriteLineInd ("foreach (object " + var2 + " in " + memberValue + ") {");
984 WriteLine ("XmlNode " + var + " = " + var2 + " as XmlNode;");
985 WriteLine ("if (" + var + " == null && " + var2 + "!= null) throw new InvalidOperationException (\"A member with XmlAnyElementAttribute can only serialize and deserialize certain XmlNode types.\");");
987 string elem = GetObTempVar ();
988 WriteLine ("XmlNode " + elem + " = " + var + ";");
989 WriteLineInd ("if (" + elem + " is XmlElement) {");
991 if (!member.IsDefaultAny) {
992 for (int n=0; n<member.ElementInfo.Count; n++) {
993 XmlTypeMapElementInfo info = (XmlTypeMapElementInfo)member.ElementInfo[n];
994 string txt = "(" + elem + ".LocalName == " + GetLiteral(info.ElementName) + " && " + elem + ".NamespaceURI == " + GetLiteral(info.Namespace) + ")";
995 if (n == member.ElementInfo.Count-1) txt += ") {";
996 if (n == 0) WriteLineInd ("if (" + txt);
997 else WriteLine ("|| " + txt);
1001 if (_format == SerializationFormat.Literal)
1002 WriteLine ("WriteElementLiteral (" + elem + ", \"\", \"\", false, true);");
1004 WriteLine ("WriteElementEncoded (" + elem + ", \"\", \"\", false, true);");
1006 if (!member.IsDefaultAny) {
1008 WriteLineInd ("else");
1009 WriteLine ("throw CreateUnknownAnyElementException (" + elem + ".Name, " + elem + ".NamespaceURI);");
1013 WriteLine ("else " + elem + ".WriteTo (Writer);");
1019 void GenerateWritePrimitiveElement (XmlTypeMapping typeMap, string ob)
1021 string strVal = GenerateGetStringValue (typeMap, typeMap.TypeData, ob, false);
1022 WriteLine ("Writer.WriteString (" + strVal + ");");
1025 void GenerateWriteEnumElement (XmlTypeMapping typeMap, string ob)
1027 string strVal = GenerateGetEnumXmlValue (typeMap, ob);
1028 WriteLine ("Writer.WriteString (" + strVal + ");");
1031 string GenerateGetStringValue (XmlTypeMapping typeMap, TypeData type, string value, bool isNullable)
1033 if (type.SchemaType == SchemaTypes.Array) {
1034 string str = GetStrTempVar ();
1035 WriteLine ("string " + str + " = null;");
1036 WriteLineInd ("if (" + value + " != null) {");
1037 string res = GenerateWriteListContent (null, typeMap.TypeData, (ListMap)typeMap.ObjectMap, value, true);
1038 WriteLine (str + " = " + res + ".ToString ().Trim ();");
1042 else if (type.SchemaType == SchemaTypes.Enum) {
1044 return "(" + value + ").HasValue ? " + GenerateGetEnumXmlValue (typeMap, "(" + value + ").Value") + " : null";
1046 return GenerateGetEnumXmlValue (typeMap, value);
1048 else if (type.Type == typeof (XmlQualifiedName))
1049 return "FromXmlQualifiedName (" + value + ")";
1050 else if (value == null)
1053 return XmlCustomFormatter.GenerateToXmlString (type, value);
1056 string GenerateGetEnumXmlValue (XmlTypeMapping typeMap, string ob)
1058 return GetGetEnumValueName (typeMap) + " (" + ob + ")";
1061 string GenerateGetListCount (TypeData listType, string ob)
1063 if (listType.Type.IsArray)
1069 void GenerateGetArrayType (ListMap map, string itemCount, out string localName, out string ns)
1072 if (itemCount != "") arrayDim = "";
1073 else arrayDim = "[]";
1075 XmlTypeMapElementInfo info = (XmlTypeMapElementInfo) map.ItemInfo[0];
1076 if (info.TypeData.SchemaType == SchemaTypes.Array)
1079 GenerateGetArrayType ((ListMap)info.MappedType.ObjectMap, "", out nm, out ns);
1080 localName = nm + arrayDim;
1084 if (info.MappedType != null)
1086 localName = info.MappedType.XmlType + arrayDim;
1087 ns = info.MappedType.Namespace;
1091 localName = info.TypeData.XmlType + arrayDim;
1092 ns = info.DataTypeNamespace;
1095 if (itemCount != "") {
1096 localName = "\"" + localName + "[\" + " + itemCount + " + \"]\"";
1097 ns = GetLiteral (ns);
1101 string GenerateWriteListContent (string container, TypeData listType, ListMap map, string ob, bool writeToString)
1103 string targetString = null;
1107 targetString = GetStrTempVar ();
1108 WriteLine ("System.Text.StringBuilder " + targetString + " = new System.Text.StringBuilder();");
1111 if (listType.Type.IsArray)
1113 string itemVar = GetNumTempVar ();
1114 WriteLineInd ("for (int "+itemVar+" = 0; "+itemVar+" < " + ob + ".Length; "+itemVar+"++) {");
1115 GenerateListLoop (container, map, ob + "["+itemVar+"]", itemVar, listType.ListItemTypeData, targetString);
1118 else if (typeof(ICollection).IsAssignableFrom (listType.Type))
1120 string itemVar = GetNumTempVar ();
1121 WriteLineInd ("for (int "+itemVar+" = 0; "+itemVar+" < " + ob + ".Count; "+itemVar+"++) {");
1122 GenerateListLoop (container, map, ob + "["+itemVar+"]", itemVar, listType.ListItemTypeData, targetString);
1125 else if (typeof(IEnumerable).IsAssignableFrom (listType.Type))
1127 string itemVar = GetObTempVar ();
1128 WriteLineInd ("foreach (" + listType.ListItemTypeData.CSharpFullName + " " + itemVar + " in " + ob + ") {");
1129 GenerateListLoop (container, map, itemVar, null, listType.ListItemTypeData, targetString);
1133 throw new Exception ("Unsupported collection type");
1135 return targetString;
1138 void GenerateListLoop (string container, ListMap map, string item, string index, TypeData itemTypeData, string targetString)
1140 bool multichoice = (map.ItemInfo.Count > 1);
1142 if (map.ChoiceMember != null && container != null && index != null) {
1143 WriteLineInd ("if ((" + container + ".@" + map.ChoiceMember + " == null) || (" + index + " >= " + container + ".@" + map.ChoiceMember + ".Length))");
1144 WriteLine ("throw CreateInvalidChoiceIdentifierValueException (" + container + ".GetType().ToString(), \"" + map.ChoiceMember + "\");");
1149 WriteLine ("if (((object)" + item + ") == null) { }");
1151 foreach (XmlTypeMapElementInfo info in map.ItemInfo)
1153 if (map.ChoiceMember != null && multichoice)
1154 WriteLineInd ("else if (" + container + ".@" + map.ChoiceMember + "[" + index + "] == " + GetLiteral (info.ChoiceValue) + ") {");
1155 else if (multichoice)
1156 WriteLineInd ("else if (" + item + ".GetType() == typeof(" + info.TypeData.CSharpFullName + ")) {");
1158 if (targetString == null)
1159 GenerateWriteMemberElement (info, GetCast (info.TypeData, itemTypeData, item));
1162 string strVal = GenerateGetStringValue (info.MappedType, info.TypeData, GetCast (info.TypeData, itemTypeData, item), false);
1163 WriteLine (targetString + ".Append (" + strVal + ").Append (\" \");");
1171 WriteLine ("else throw CreateUnknownTypeException (" + item + ");");
1174 void GenerateWritePrimitiveValueLiteral (string memberValue, string name, string ns, XmlTypeMapping mappedType, TypeData typeData, bool wrapped, bool isNullable)
1177 string strVal = GenerateGetStringValue (mappedType, typeData, memberValue, false);
1178 WriteMetCall ("WriteValue", strVal);
1180 else if (isNullable) {
1181 if (typeData.Type == typeof(XmlQualifiedName))
1182 WriteMetCall ("WriteNullableQualifiedNameLiteral", GetLiteral(name), GetLiteral(ns), memberValue);
1184 string strVal = GenerateGetStringValue (mappedType, typeData, memberValue, true);
1185 WriteMetCall ("WriteNullableStringLiteral", GetLiteral(name), GetLiteral(ns), strVal);
1189 if (typeData.Type == typeof(XmlQualifiedName))
1190 WriteMetCall ("WriteElementQualifiedName", GetLiteral(name), GetLiteral(ns), memberValue);
1192 string strVal = GenerateGetStringValue (mappedType, typeData, memberValue, false);
1193 WriteMetCall ("WriteElementString", GetLiteral(name),GetLiteral(ns), strVal);
1198 void GenerateWritePrimitiveValueEncoded (string memberValue, string name, string ns, XmlQualifiedName xsiType, XmlTypeMapping mappedType, TypeData typeData, bool wrapped, bool isNullable)
1201 string strVal = GenerateGetStringValue (mappedType, typeData, memberValue, false);
1202 WriteMetCall ("WriteValue", strVal);
1204 else if (isNullable) {
1205 if (typeData.Type == typeof(XmlQualifiedName))
1206 WriteMetCall ("WriteNullableQualifiedNameEncoded", GetLiteral(name), GetLiteral(ns), memberValue, GetLiteral(xsiType));
1208 string strVal = GenerateGetStringValue (mappedType, typeData, memberValue, true);
1209 WriteMetCall ("WriteNullableStringEncoded", GetLiteral(name), GetLiteral(ns), strVal, GetLiteral(xsiType));
1213 if (typeData.Type == typeof(XmlQualifiedName))
1214 WriteMetCall ("WriteElementQualifiedName", GetLiteral(name), GetLiteral(ns), memberValue, GetLiteral(xsiType));
1216 string strVal = GenerateGetStringValue (mappedType, typeData, memberValue, false);
1217 WriteMetCall ("WriteElementString", GetLiteral(name),GetLiteral(ns), strVal, GetLiteral(xsiType));
1222 string GenerateGetMemberValue (XmlTypeMapMember member, string ob, bool isValueList)
1224 if (isValueList) return GetCast (member.TypeData, TypeTranslator.GetTypeData (typeof(object)), ob + "[" + member.GlobalIndex + "]");
1225 else return ob + ".@" + member.Name;
1228 string GenerateMemberHasValueCondition (XmlTypeMapMember member, string ob, bool isValueList)
1231 if (member.IsOptionalValueType)
1232 return ob + ".Length > " + Math.Max (member.GlobalIndex, member.SpecifiedGlobalIndex) + " && " + GetCast (typeof(bool), ob + "[" + member.SpecifiedGlobalIndex + "]");
1234 return ob + ".Length > " + member.GlobalIndex;
1236 else if (member.DefaultValue != System.DBNull.Value) {
1237 string mem = ob + ".@" + member.Name;
1238 if (member.DefaultValue == null)
1239 return mem + " != null";
1240 else if (member.TypeData.SchemaType == SchemaTypes.Enum)
1241 return mem + " != " + GetCast (member.TypeData, GetLiteral (member.DefaultValue));
1243 return mem + " != " + GetLiteral (member.DefaultValue);
1245 else if (member.HasSpecified) {
1246 var sb = new StringBuilder ();
1247 sb.AppendFormat ("{0}.@{1}Specified", ob, member.Name);
1248 if (member.HasShouldSerialize)
1249 sb.AppendFormat (" && {0}.@ShouldSerialize{1} ()", ob, member.Name);
1250 return sb.ToString ();
1251 } else if (member.HasShouldSerialize)
1252 return ob + ".@ShouldSerialize" + member.Name + " ()";
1257 void GenerateWriteInitCallbacks ()
1259 WriteLine ("protected override void InitCallbacks ()");
1262 if (_format == SerializationFormat.Encoded)
1264 foreach (XmlMapping xmap in _mapsToGenerate) {
1265 XmlTypeMapping map = xmap as XmlTypeMapping;
1267 WriteMetCall ("AddWriteCallback", GetTypeOf(map.TypeData), GetLiteral(map.XmlType), GetLiteral(map.Namespace), "new XmlSerializationWriteCallback (" + GetWriteObjectCallbackName (map) + ")");
1274 if (_format == SerializationFormat.Encoded)
1276 foreach (XmlTypeMapping xmap in _mapsToGenerate) {
1277 XmlTypeMapping map = xmap as XmlTypeMapping;
1278 if (map == null) continue;
1279 if (map.TypeData.SchemaType == SchemaTypes.Enum)
1280 WriteWriteEnumCallback (map);
1282 WriteWriteObjectCallback (map);
1287 void WriteWriteEnumCallback (XmlTypeMapping map)
1289 WriteLine ("void " + GetWriteObjectCallbackName (map) + " (object ob)");
1291 WriteMetCall (GetWriteObjectName(map), GetCast (map.TypeData, "ob"), GetLiteral(map.ElementName), GetLiteral(map.Namespace), "false", "true", "false");
1296 void WriteWriteObjectCallback (XmlTypeMapping map)
1298 WriteLine ("void " + GetWriteObjectCallbackName (map) + " (object ob)");
1300 WriteMetCall (GetWriteObjectName(map), GetCast (map.TypeData, "ob"), GetLiteral(map.ElementName), GetLiteral(map.Namespace), "false", "false", "false");
1307 #region Reader Generation
1309 //*******************************************************
1310 // Reader generation
1313 public void GenerateReader (string readerClassName, ArrayList maps)
1315 if (_config == null || !_config.GenerateAsInternal)
1316 WriteLine ("public class " + readerClassName + " : XmlSerializationReader");
1318 WriteLine ("internal class " + readerClassName + " : XmlSerializationReader");
1320 // FromBinHexString() is not public, so use reflection here.
1321 WriteLine ("static readonly System.Reflection.MethodInfo fromBinHexStringMethod = typeof (XmlConvert).GetMethod (\"FromBinHexString\", System.Reflection.BindingFlags.Static | System.Reflection.BindingFlags.NonPublic, null, new Type [] {typeof (string)}, null);");
1322 WriteLine ("static byte [] FromBinHexString (string input)");
1324 WriteLine ("return input == null ? null : (byte []) fromBinHexStringMethod.Invoke (null, new object [] {input});");
1327 _mapsToGenerate = new ArrayList ();
1328 _fixupCallbacks = new ArrayList ();
1331 for (int n=0; n<maps.Count; n++)
1333 GenerationResult res = (GenerationResult) maps [n];
1334 _typeMap = res.Mapping;
1335 _format = _typeMap.Format;
1338 GenerateReadRoot ();
1341 for (int n=0; n<_mapsToGenerate.Count; n++)
1343 XmlTypeMapping map = _mapsToGenerate [n] as XmlTypeMapping;
1344 if (map == null) continue;
1346 GenerateReadObject (map);
1347 if (map.TypeData.SchemaType == SchemaTypes.Enum)
1348 GenerateGetEnumValueMethod (map);
1351 GenerateReadInitCallbacks ();
1353 if (_format == SerializationFormat.Encoded)
1355 GenerateFixupCallbacks ();
1356 GenerateFillerCallbacks ();
1360 UpdateGeneratedTypes (_mapsToGenerate);
1363 void GenerateReadRoot ()
1365 WriteLine ("public object " + _result.ReadMethodName + " ()");
1367 WriteLine ("Reader.MoveToContent();");
1369 if (_typeMap is XmlTypeMapping)
1371 XmlTypeMapping typeMap = (XmlTypeMapping) _typeMap;
1373 if (_format == SerializationFormat.Literal)
1375 if (typeMap.TypeData.SchemaType == SchemaTypes.XmlNode) {
1376 if (typeMap.TypeData.Type == typeof (XmlDocument))
1377 WriteLine ("return ReadXmlDocument (false);");
1379 WriteLine ("return ReadXmlNode (false);");
1381 if (!typeMap.IsAny) {
1382 WriteLineInd ("if (Reader.LocalName != " + GetLiteral (typeMap.ElementName) + " || Reader.NamespaceURI != " + GetLiteral (typeMap.Namespace) + ")");
1383 WriteLine ("throw CreateUnknownNodeException();");
1387 WriteLine ("return " + GetReadObjectCall (typeMap, GetLiteral(typeMap.IsNullable), "true") + ";");
1392 WriteLine ("object ob = null;");
1393 WriteLine ("Reader.MoveToContent();");
1394 WriteLine ("if (Reader.NodeType == System.Xml.XmlNodeType.Element) ");
1397 WriteLineInd ("if (Reader.LocalName == " + GetLiteral(typeMap.ElementName) + " && Reader.NamespaceURI == " + GetLiteral (typeMap.Namespace) + ")");
1398 WriteLine ("ob = ReadReferencedElement();");
1400 if (!typeMap.IsAny) {
1401 WriteLineInd ("else ");
1402 WriteLine ("throw CreateUnknownNodeException();");
1406 WriteLineInd ("else ");
1407 WriteLine ("UnknownNode(null);");
1410 WriteLine ("ReadReferencedElements();");
1411 WriteLine ("return ob;");
1412 RegisterReferencingMap (typeMap);
1416 WriteLine ("return " + GenerateReadMessage ((XmlMembersMapping)_typeMap) + ";");
1423 string GenerateReadMessage (XmlMembersMapping typeMap)
1425 WriteLine ("object[] parameters = new object[" + typeMap.Count + "];");
1428 if (typeMap.HasWrapperElement)
1430 if (_format == SerializationFormat.Encoded)
1432 WriteLine ("while (Reader.NodeType == System.Xml.XmlNodeType.Element)");
1434 WriteLine ("string root = Reader.GetAttribute (\"root\", " + GetLiteral(XmlSerializer.EncodingNamespace) + ");");
1435 WriteLine ("if (root == null || System.Xml.XmlConvert.ToBoolean(root)) break;");
1436 WriteLine ("ReadReferencedElement ();");
1437 WriteLine ("Reader.MoveToContent ();");
1440 WriteLine ("if (Reader.NodeType != System.Xml.XmlNodeType.EndElement)");
1442 WriteLineInd ("if (Reader.IsEmptyElement) {");
1443 WriteLine ("Reader.Skip();");
1444 WriteLine ("Reader.MoveToContent();");
1446 WriteLineInd ("else {");
1447 WriteLine ("Reader.ReadStartElement();");
1448 GenerateReadMembers (typeMap, (ClassMap)typeMap.ObjectMap, "parameters", true, false);
1449 WriteLine ("ReadEndElement();");
1452 WriteLine ("Reader.MoveToContent();");
1457 // bug #79988: out parameters need to be initialized if
1458 // they are value types
1459 ClassMap classMap = (ClassMap) typeMap.ObjectMap;
1460 ArrayList members = classMap.AllMembers;
1461 for (int n = 0; n < members.Count; n++) {
1462 XmlTypeMapMember mem = (XmlTypeMapMember) members [n];
1463 if (!mem.IsReturnValue && mem.TypeData.IsValueType)
1464 GenerateSetMemberValueFromAttr (mem, "parameters",
1465 String.Format ("({0}) Activator.CreateInstance(typeof({0}), true)", mem.TypeData.FullTypeName), true);
1468 WriteLine ("while (Reader.NodeType != System.Xml.XmlNodeType.EndElement && Reader.ReadState == ReadState.Interactive)");
1470 WriteLine ("if (Reader.IsStartElement(" + GetLiteral(typeMap.ElementName) + ", " + GetLiteral(typeMap.Namespace) + "))");
1473 GenerateReadAttributeMembers (typeMap, (ClassMap)typeMap.ObjectMap, "parameters", true, ref dummy);
1474 WriteLine ("if (Reader.IsEmptyElement)");
1476 WriteLine ("Reader.Skip(); Reader.MoveToContent(); continue;");
1478 WriteLine ("Reader.ReadStartElement();");
1479 GenerateReadMembers (typeMap, (ClassMap)typeMap.ObjectMap, "parameters", true, false);
1480 WriteLine ("ReadEndElement();");
1481 WriteLine ("break;");
1483 WriteLineInd ("else ");
1484 WriteLine ("UnknownNode(null);");
1487 WriteLine ("Reader.MoveToContent();");
1492 GenerateReadMembers (typeMap, (ClassMap)typeMap.ObjectMap, "parameters", true, _format == SerializationFormat.Encoded);
1494 if (_format == SerializationFormat.Encoded)
1495 WriteLine ("ReadReferencedElements();");
1497 return "parameters";
1500 void GenerateReadObject (XmlTypeMapping typeMap)
1503 if (_format == SerializationFormat.Literal) {
1504 WriteLine ("public " + GetTypeFullName(typeMap.TypeData) + " " + GetReadObjectName (typeMap) + " (bool isNullable, bool checkType)");
1505 isNullable = "isNullable";
1508 WriteLine ("public object " + GetReadObjectName (typeMap) + " ()");
1509 isNullable = "true";
1516 SetHookVar ("$TYPE", typeMap.TypeData.CSharpName);
1517 SetHookVar ("$FULLTYPE", typeMap.TypeData.CSharpFullName);
1518 SetHookVar ("$NULLABLE", "isNullable");
1520 switch (typeMap.TypeData.SchemaType)
1522 case SchemaTypes.Class: GenerateReadClassInstance (typeMap, isNullable, "checkType"); break;
1523 case SchemaTypes.Array:
1524 string list = GenerateReadListElement (typeMap, null, isNullable, true);
1526 WriteLine ("return " + list + ";");
1528 case SchemaTypes.XmlNode: GenerateReadXmlNodeElement (typeMap, isNullable); break;
1529 case SchemaTypes.Primitive: GenerateReadPrimitiveElement (typeMap, isNullable); break;
1530 case SchemaTypes.Enum: GenerateReadEnumElement (typeMap, isNullable); break;
1531 case SchemaTypes.XmlSerializable: GenerateReadXmlSerializableElement (typeMap, isNullable); break;
1532 default: throw new Exception ("Unsupported map type");
1540 void GenerateReadClassInstance (XmlTypeMapping typeMap, string isNullable, string checkType)
1542 SetHookVar ("$OBJECT", "ob");
1543 if (!typeMap.TypeData.IsValueType)
1545 WriteLine (typeMap.TypeData.CSharpFullName + " ob = null;");
1547 if (GenerateReadHook (HookType.type, typeMap.TypeData.Type)) {
1548 WriteLine ("return ob;");
1552 if (_format == SerializationFormat.Literal) {
1553 WriteLine ("if (" + isNullable + " && ReadNull()) return null;");
1555 WriteLine ("if (checkType) ");
1559 WriteLine ("if (ReadNull()) return null;");
1565 WriteLine (typeMap.TypeData.CSharpFullName + String.Format (" ob = ({0}) Activator.CreateInstance(typeof({0}), true);", typeMap.TypeData.CSharpFullName));
1567 if (GenerateReadHook (HookType.type, typeMap.TypeData.Type)) {
1568 WriteLine ("return ob;");
1573 WriteLine ("System.Xml.XmlQualifiedName t = GetXsiType();");
1574 WriteLine ("if (t == null)");
1575 if (typeMap.TypeData.Type != typeof(object))
1578 WriteLine ("\treturn " + GetCast (typeMap.TypeData, "ReadTypedPrimitive (new System.Xml.XmlQualifiedName(\"anyType\", System.Xml.Schema.XmlSchema.Namespace))") + ";");
1580 foreach (XmlTypeMapping realMap in typeMap.DerivedTypes)
1582 WriteLineInd ("else if (t.Name == " + GetLiteral (realMap.XmlType) + " && t.Namespace == " + GetLiteral (realMap.XmlTypeNamespace) + ")");
1583 WriteLine ("return " + GetReadObjectCall(realMap, isNullable, checkType) + ";");
1587 WriteLine ("else if (t.Name != " + GetLiteral (typeMap.XmlType) + " || t.Namespace != " + GetLiteral (typeMap.XmlTypeNamespace) + ")");
1588 if (typeMap.TypeData.Type == typeof(object))
1589 WriteLine ("\treturn " + GetCast (typeMap.TypeData, "ReadTypedPrimitive (t)") + ";");
1591 WriteLine ("\tthrow CreateUnknownTypeException(t);");
1593 if (!typeMap.TypeData.IsValueType)
1595 if (_format == SerializationFormat.Literal)
1598 if (typeMap.TypeData.Type.IsAbstract) {
1600 WriteLine ("return ob;");
1605 WriteLine (String.Format ("ob = ({0}) Activator.CreateInstance(typeof({0}), true);", typeMap.TypeData.CSharpFullName));
1610 WriteLine ("Reader.MoveToElement();");
1613 GenerateReadMembers (typeMap, (ClassMap)typeMap.ObjectMap, "ob", false, false);
1618 WriteLine ("return ob;");
1621 void GenerateReadMembers (XmlMapping xmlMap, ClassMap map, string ob, bool isValueList, bool readBySoapOrder)
1623 XmlTypeMapping typeMap = xmlMap as XmlTypeMapping;
1624 Type xmlMapType = (typeMap != null) ? typeMap.TypeData.Type : typeof(object[]);
1628 GenerateReadAttributeMembers (xmlMap, map, ob, isValueList, ref first);
1632 WriteLine ("Reader.MoveToElement();");
1633 WriteLineInd ("if (Reader.IsEmptyElement) {");
1634 WriteLine ("Reader.Skip ();");
1635 GenerateSetListMembersDefaults (typeMap, map, ob, isValueList);
1636 WriteLine ("return " + ob + ";");
1640 WriteLine ("Reader.ReadStartElement();");
1645 WriteLine("Reader.MoveToContent();");
1648 if (!GenerateReadHook (HookType.elements, xmlMapType))
1650 string[] readFlag = null;
1651 if (map.ElementMembers != null && !readBySoapOrder)
1653 string readFlagsVars = string.Empty;
1654 readFlag = new string[map.ElementMembers.Count];
1656 foreach (XmlTypeMapMember mem in map.ElementMembers) {
1657 // The text collector doesn't need a flag
1658 if (!((mem is XmlTypeMapMemberElement) && ((XmlTypeMapMemberElement)mem).IsXmlTextCollector)) {
1659 readFlag[n] = GetBoolTempVar ();
1660 if (readFlagsVars.Length > 0) readFlagsVars += ", ";
1661 readFlagsVars += readFlag[n] + "=false";
1665 if (readFlagsVars.Length > 0) {
1666 readFlagsVars = "bool " + readFlagsVars;
1667 WriteLine (readFlagsVars + ";");
1669 foreach (XmlTypeMapElementInfo info in map.AllElementInfos)
1670 if (info.ExplicitOrder >= 0) {
1671 WriteLine ("int idx = -1;");
1677 string[] indexes = null;
1678 string[] flatLists = null;
1679 string[] flatListsChoices = null;
1681 if (map.FlatLists != null)
1683 indexes = new string[map.FlatLists.Count];
1684 flatLists = new string[map.FlatLists.Count];
1686 string code = "int ";
1687 for (int n=0; n<map.FlatLists.Count; n++)
1689 XmlTypeMapMemberElement mem = (XmlTypeMapMemberElement)map.FlatLists[n];
1690 indexes[n] = GetNumTempVar ();
1691 if (n > 0) code += ", ";
1692 code += indexes[n] + "=0";
1693 if (!MemberHasReadReplaceHook (xmlMapType, mem)) {
1694 flatLists[n] = GetObTempVar ();
1696 WriteLine (mem.TypeData.CSharpFullName + " " + flatLists[n] + ";");
1697 if (IsReadOnly (typeMap, mem, mem.TypeData, isValueList)) {
1698 rval = GenerateGetMemberValue (mem, ob, isValueList);
1699 WriteLine (flatLists[n] + " = " + rval + ";");
1700 } else if (mem.TypeData.Type.IsArray) {
1701 rval = GenerateInitializeList (mem.TypeData);
1702 WriteLine (flatLists[n] + " = " + rval + ";");
1704 WriteLine (flatLists[n] + " = " + GenerateGetMemberValue (mem, ob, isValueList) + ";");
1705 WriteLineInd ("if (((object)" + flatLists[n] + ") == null) {");
1706 WriteLine (flatLists[n] + " = " + GenerateInitializeList (mem.TypeData) + ";");
1707 GenerateSetMemberValue (mem, ob, flatLists[n], isValueList);
1712 if (mem.ChoiceMember != null) {
1713 if (flatListsChoices == null)
1714 flatListsChoices = new string [map.FlatLists.Count];
1715 flatListsChoices[n] = GetObTempVar ();
1716 string rval = GenerateInitializeList (mem.ChoiceTypeData);
1717 WriteLine (mem.ChoiceTypeData.CSharpFullName + " " + flatListsChoices[n] + " = " + rval + ";");
1720 WriteLine (code + ";");
1724 if (_format == SerializationFormat.Encoded && map.ElementMembers != null)
1726 _fixupCallbacks.Add (xmlMap);
1727 WriteLine ("Fixup fixup = new Fixup(" + ob + ", new XmlSerializationFixupCallback(" + GetFixupCallbackName (xmlMap) + "), " + map.ElementMembers.Count + ");");
1728 WriteLine ("AddFixup (fixup);");
1732 ArrayList infos = null;
1735 if (readBySoapOrder) {
1736 if (map.ElementMembers != null) maxInd = map.ElementMembers.Count;
1741 infos = new ArrayList ();
1742 infos.AddRange (map.AllElementInfos);
1743 maxInd = infos.Count;
1745 WriteLine ("while (Reader.NodeType != System.Xml.XmlNodeType.EndElement) ");
1747 WriteLine ("if (Reader.NodeType == System.Xml.XmlNodeType.Element) ");
1752 for (int ind = 0; ind < maxInd; ind++)
1754 XmlTypeMapElementInfo info = readBySoapOrder ? map.GetElement (ind) : (XmlTypeMapElementInfo) infos [ind];
1756 if (!readBySoapOrder)
1758 if (info.IsTextElement || info.IsUnnamedAnyElement) continue;
1759 string elemCond = first ? "" : "else ";
1761 if (info.ExplicitOrder >= 0)
1762 elemCond += "idx < " + info.ExplicitOrder + "&& ";
1763 if (!(info.Member.IsReturnValue && _format == SerializationFormat.Encoded)) {
1764 elemCond += "Reader.LocalName == " + GetLiteral (info.ElementName);
1765 if (!map.IgnoreMemberNamespace) elemCond += " && Reader.NamespaceURI == " + GetLiteral (info.Namespace);
1768 if (readFlag[info.Member.Index] != null)
1769 elemCond += "!" + readFlag[info.Member.Index] + ") {";
1771 elemCond += "true) {";
1772 WriteLineInd (elemCond);
1775 if (info.Member.GetType() == typeof (XmlTypeMapMemberList))
1777 if (_format == SerializationFormat.Encoded && info.MultiReferenceType)
1779 string list = GetObTempVar ();
1780 WriteLine ("object " + list + " = ReadReferencingElement (out fixup.Ids[" + info.Member.Index + "]);");
1781 RegisterReferencingMap (info.MappedType);
1783 WriteLineInd ("if (fixup.Ids[" + info.Member.Index + "] == null) {"); // Already read
1784 if (IsReadOnly (typeMap, info.Member, info.TypeData, isValueList))
1785 WriteLine ("throw CreateReadOnlyCollectionException (" + GetLiteral(info.TypeData.CSharpFullName) + ");");
1787 GenerateSetMemberValue (info.Member, ob, GetCast (info.Member.TypeData,list), isValueList);
1790 if (!info.MappedType.TypeData.Type.IsArray)
1792 WriteLineInd ("else {");
1793 if (IsReadOnly (typeMap, info.Member, info.TypeData, isValueList))
1794 WriteLine (list + " = " + GenerateGetMemberValue (info.Member, ob, isValueList) + ";");
1796 WriteLine (list + " = " + GenerateCreateList (info.MappedType.TypeData.Type) + ";");
1797 GenerateSetMemberValue (info.Member, ob, GetCast (info.Member.TypeData,list), isValueList);
1799 WriteLine ("AddFixup (new CollectionFixup (" + list + ", new XmlSerializationCollectionFixupCallback (" + GetFillListName(info.Member.TypeData) + "), fixup.Ids[" + info.Member.Index + "]));");
1800 WriteLine ("fixup.Ids[" + info.Member.Index + "] = null;"); // The member already has the value, no further fix needed.
1806 if (!GenerateReadMemberHook (xmlMapType, info.Member)) {
1807 if (IsReadOnly (typeMap, info.Member, info.TypeData, isValueList)) {
1808 GenerateReadListElement (info.MappedType, GenerateGetMemberValue (info.Member, ob, isValueList), GetLiteral(info.IsNullable), false);
1809 } else if (info.MappedType.TypeData.Type.IsArray) {
1810 if (info.IsNullable)
1811 GenerateSetMemberValue (info.Member, ob, GenerateReadListElement (info.MappedType, null, GetLiteral(info.IsNullable), true), isValueList);
1813 string list = GetObTempVar ();
1814 WriteLine (info.MappedType.TypeData.CSharpFullName + " " + list + " = " + GenerateReadListElement (info.MappedType, null, GetLiteral(info.IsNullable), true) + ";");
1815 WriteLineInd ("if (((object)" + list + ") != null) {");
1816 GenerateSetMemberValue (info.Member, ob, list, isValueList);
1820 string list = GetObTempVar ();
1821 WriteLine (info.MappedType.TypeData.CSharpFullName + " " + list + " = " + GenerateGetMemberValue (info.Member, ob, isValueList) + ";");
1822 WriteLineInd ("if (((object)" + list + ") == null) {");
1823 WriteLine (list + " = " + GenerateCreateList (info.MappedType.TypeData.Type) + ";");
1824 GenerateSetMemberValue (info.Member, ob, list, isValueList);
1826 GenerateReadListElement (info.MappedType, list, GetLiteral(info.IsNullable), true);
1831 if (!readBySoapOrder)
1832 WriteLine (readFlag[info.Member.Index] + " = true;");
1834 else if (info.Member.GetType() == typeof (XmlTypeMapMemberFlatList))
1836 XmlTypeMapMemberFlatList mem = (XmlTypeMapMemberFlatList)info.Member;
1837 if (!GenerateReadArrayMemberHook (xmlMapType, info.Member, indexes[mem.FlatArrayIndex])) {
1838 GenerateAddListValue (mem.TypeData, flatLists[mem.FlatArrayIndex], indexes[mem.FlatArrayIndex], GenerateReadObjectElement (info), !IsReadOnly (typeMap, info.Member, info.TypeData, isValueList));
1839 if (mem.ChoiceMember != null) {
1840 GenerateAddListValue (mem.ChoiceTypeData, flatListsChoices[mem.FlatArrayIndex], indexes[mem.FlatArrayIndex], GetLiteral (info.ChoiceValue), true);
1844 WriteLine (indexes[mem.FlatArrayIndex] + "++;");
1846 else if (info.Member.GetType() == typeof (XmlTypeMapMemberAnyElement))
1848 XmlTypeMapMemberAnyElement mem = (XmlTypeMapMemberAnyElement)info.Member;
1849 if (mem.TypeData.IsListType) {
1850 if (!GenerateReadArrayMemberHook (xmlMapType, info.Member, indexes[mem.FlatArrayIndex])) {
1851 GenerateAddListValue (mem.TypeData, flatLists[mem.FlatArrayIndex], indexes[mem.FlatArrayIndex], GetReadXmlNode (mem.TypeData.ListItemTypeData, false), true);
1854 WriteLine (indexes[mem.FlatArrayIndex] + "++;");
1857 if (!GenerateReadMemberHook (xmlMapType, info.Member)) {
1858 GenerateSetMemberValue (mem, ob, GetReadXmlNode(mem.TypeData, false), isValueList);
1863 else if (info.Member.GetType() == typeof(XmlTypeMapMemberElement))
1865 if (!readBySoapOrder)
1866 WriteLine (readFlag[info.Member.Index] + " = true;");
1867 if (info.ExplicitOrder >= 0)
1868 WriteLine ("idx = " + info.ExplicitOrder + ";");
1869 if (_format == SerializationFormat.Encoded)
1871 string val = GetObTempVar ();
1872 RegisterReferencingMap (info.MappedType);
1874 if (info.Member.TypeData.SchemaType != SchemaTypes.Primitive)
1875 WriteLine ("object " + val + " = ReadReferencingElement (out fixup.Ids[" + info.Member.Index + "]);");
1877 WriteLine ("object " + val + " = ReadReferencingElement (" + GetLiteral(info.Member.TypeData.XmlType) + ", " + GetLiteral(System.Xml.Schema.XmlSchema.Namespace) + ", out fixup.Ids[" + info.Member.Index + "]);");
1879 if (info.MultiReferenceType)
1880 WriteLineInd ("if (fixup.Ids[" + info.Member.Index + "] == null) {"); // already read
1882 WriteLineInd ("if (" + val + " != null) {"); // null value
1884 GenerateSetMemberValue (info.Member, ob, GetCast (info.Member.TypeData,val), isValueList);
1887 else if (!GenerateReadMemberHook (xmlMapType, info.Member)) {
1888 if (info.ChoiceValue != null) {
1889 XmlTypeMapMemberElement imem = (XmlTypeMapMemberElement) info.Member;
1890 WriteLine (ob + ".@" + imem.ChoiceMember + " = " + GetLiteral(info.ChoiceValue) + ";");
1892 GenerateSetMemberValue (info.Member, ob, GenerateReadObjectElement (info), isValueList);
1897 throw new InvalidOperationException ("Unknown member type");
1899 if (!readBySoapOrder)
1902 WriteLine ("Reader.MoveToContent();");
1906 if (!readBySoapOrder)
1908 if (!first) WriteLineInd ("else {");
1910 if (map.DefaultAnyElementMember != null)
1912 XmlTypeMapMemberAnyElement mem = map.DefaultAnyElementMember;
1913 if (mem.TypeData.IsListType) {
1914 if (!GenerateReadArrayMemberHook (xmlMapType, mem, indexes[mem.FlatArrayIndex])) {
1915 GenerateAddListValue (mem.TypeData, flatLists[mem.FlatArrayIndex], indexes[mem.FlatArrayIndex], GetReadXmlNode(mem.TypeData.ListItemTypeData, false), true);
1918 WriteLine (indexes[mem.FlatArrayIndex] + "++;");
1920 else if (! GenerateReadMemberHook (xmlMapType, mem)) {
1921 GenerateSetMemberValue (mem, ob, GetReadXmlNode(mem.TypeData, false), isValueList);
1926 if (!GenerateReadHook (HookType.unknownElement, xmlMapType)) {
1927 WriteLine ("UnknownNode (" + ob + ");");
1932 if (!first) WriteLineUni ("}");
1936 if (map.XmlTextCollector != null)
1938 WriteLine ("else if (Reader.NodeType == System.Xml.XmlNodeType.Text || Reader.NodeType == System.Xml.XmlNodeType.CDATA)");
1941 if (map.XmlTextCollector is XmlTypeMapMemberExpandable)
1943 XmlTypeMapMemberExpandable mem = (XmlTypeMapMemberExpandable)map.XmlTextCollector;
1944 XmlTypeMapMemberFlatList flatl = mem as XmlTypeMapMemberFlatList;
1945 TypeData itype = (flatl == null) ? mem.TypeData.ListItemTypeData : flatl.ListMap.FindTextElement().TypeData;
1947 if (!GenerateReadArrayMemberHook (xmlMapType, map.XmlTextCollector, indexes[mem.FlatArrayIndex])) {
1948 string val = (itype.Type == typeof (string)) ? "Reader.ReadString()" : GetReadXmlNode (itype, false);
1949 GenerateAddListValue (mem.TypeData, flatLists[mem.FlatArrayIndex], indexes[mem.FlatArrayIndex], val, true);
1952 WriteLine (indexes[mem.FlatArrayIndex] + "++;");
1954 else if (!GenerateReadMemberHook (xmlMapType, map.XmlTextCollector))
1956 XmlTypeMapMemberElement mem = (XmlTypeMapMemberElement) map.XmlTextCollector;
1957 XmlTypeMapElementInfo info = (XmlTypeMapElementInfo) mem.ElementInfo [0];
1958 string str = GetStrTempVar ();
1959 WriteLine ("string " + str + " = Reader.ReadString();");
1960 if (info.TypeData.Type == typeof (string))
1961 GenerateSetMemberValue (mem, ob, str, isValueList);
1963 GenerateSetMemberValue (mem, ob, GenerateGetValueFromXmlString (str, info.TypeData, info.MappedType, info.IsNullable), isValueList);
1971 WriteLine ("\tUnknownNode(" + ob + ");");
1973 WriteLine ("Reader.MoveToContent();");
1977 WriteLine ("Reader.MoveToContent();");
1979 if (flatLists != null)
1982 foreach (XmlTypeMapMemberExpandable mem in map.FlatLists)
1984 if (MemberHasReadReplaceHook (xmlMapType, mem)) continue;
1986 string list = flatLists[mem.FlatArrayIndex];
1987 if (mem.TypeData.Type.IsArray)
1988 WriteLine (list + " = (" + mem.TypeData.CSharpFullName + ") ShrinkArray (" + list + ", " + indexes[mem.FlatArrayIndex] + ", " + GetTypeOf(mem.TypeData.Type.GetElementType()) + ", true);");
1989 if (!IsReadOnly (typeMap, mem, mem.TypeData, isValueList) && mem.TypeData.Type.IsArray)
1990 GenerateSetMemberValue (mem, ob, list, isValueList);
1994 if (flatListsChoices != null)
1997 foreach (XmlTypeMapMemberExpandable mem in map.FlatLists)
1999 if (MemberHasReadReplaceHook (xmlMapType, mem)) continue;
2000 if (mem.ChoiceMember == null) continue;
2002 string list = flatListsChoices[mem.FlatArrayIndex];
2003 WriteLine (list + " = (" + mem.ChoiceTypeData.CSharpFullName + ") ShrinkArray (" + list + ", " + indexes[mem.FlatArrayIndex] + ", " + GetTypeOf(mem.ChoiceTypeData.Type.GetElementType()) + ", true);");
2004 WriteLine (ob + ".@" + mem.ChoiceMember + " = " + list + ";");
2008 GenerateSetListMembersDefaults (typeMap, map, ob, isValueList);
2016 WriteLine ("ReadEndElement();");
2020 void GenerateReadAttributeMembers (XmlMapping xmlMap, ClassMap map, string ob, bool isValueList, ref bool first)
2022 XmlTypeMapping typeMap = xmlMap as XmlTypeMapping;
2023 Type xmlMapType = (typeMap != null) ? typeMap.TypeData.Type : typeof(object[]);
2025 if (GenerateReadHook (HookType.attributes, xmlMapType))
2028 XmlTypeMapMember anyAttrMember = map.DefaultAnyAttributeMember;
2030 if (anyAttrMember != null)
2032 WriteLine ("int anyAttributeIndex = 0;");
2033 WriteLine (anyAttrMember.TypeData.CSharpFullName + " anyAttributeArray = null;");
2036 WriteLine ("while (Reader.MoveToNextAttribute())");
2039 if (map.AttributeMembers != null) {
2040 foreach (XmlTypeMapMemberAttribute at in map.AttributeMembers)
2042 WriteLineInd ((first?"":"else ") + "if (Reader.LocalName == " + GetLiteral (at.AttributeName) + " && Reader.NamespaceURI == " + GetLiteral (at.Namespace) + ") {");
2043 if (!GenerateReadMemberHook (xmlMapType, at)) {
2044 GenerateSetMemberValue (at, ob, GenerateGetValueFromXmlString ("Reader.Value", at.TypeData, at.MappedType, false), isValueList);
2051 WriteLineInd ((first?"":"else ") + "if (IsXmlnsAttribute (Reader.Name)) {");
2053 // If the map has NamespaceDeclarations,
2054 // then store this xmlns to the given member.
2055 // If the instance doesn't exist, then create.
2057 if (map.NamespaceDeclarations != null) {
2058 if (!GenerateReadMemberHook (xmlMapType, map.NamespaceDeclarations)) {
2059 string nss = ob + ".@" + map.NamespaceDeclarations.Name;
2060 WriteLine ("if (" + nss + " == null) " + nss + " = new XmlSerializerNamespaces ();");
2061 WriteLineInd ("if (Reader.Prefix == \"xmlns\")");
2062 WriteLine (nss + ".Add (Reader.LocalName, Reader.Value);");
2064 WriteLineInd ("else");
2065 WriteLine (nss + ".Add (\"\", Reader.Value);");
2072 WriteLineInd ("else {");
2074 if (anyAttrMember != null)
2076 if (!GenerateReadArrayMemberHook (xmlMapType, anyAttrMember, "anyAttributeIndex")) {
2077 WriteLine ("System.Xml.XmlAttribute attr = (System.Xml.XmlAttribute) Document.ReadNode(Reader);");
2078 if (typeof(System.Xml.Schema.XmlSchemaAnnotated).IsAssignableFrom (xmlMapType))
2079 WriteLine ("ParseWsdlArrayType (attr);");
2080 GenerateAddListValue (anyAttrMember.TypeData, "anyAttributeArray", "anyAttributeIndex", GetCast (anyAttrMember.TypeData.ListItemTypeData, "attr"), true);
2083 WriteLine ("anyAttributeIndex++;");
2086 if (!GenerateReadHook (HookType.unknownAttribute, xmlMapType)) {
2087 WriteLine ("UnknownNode (" + ob + ");");
2095 if (anyAttrMember != null && !MemberHasReadReplaceHook (xmlMapType, anyAttrMember))
2098 WriteLine("anyAttributeArray = (" + anyAttrMember.TypeData.CSharpFullName + ") ShrinkArray (anyAttributeArray, anyAttributeIndex, " + GetTypeOf(anyAttrMember.TypeData.Type.GetElementType()) + ", true);");
2099 GenerateSetMemberValue (anyAttrMember, ob, "anyAttributeArray", isValueList);
2102 WriteLine ("Reader.MoveToElement ();");
2107 void GenerateSetListMembersDefaults (XmlTypeMapping typeMap, ClassMap map, string ob, bool isValueList)
2109 if (map.ListMembers != null)
2111 ArrayList members = map.ListMembers;
2112 for (int n=0; n<members.Count; n++) {
2113 XmlTypeMapMember mem = (XmlTypeMapMember) members[n];
2114 if (IsReadOnly (typeMap, mem, mem.TypeData, isValueList))
2116 WriteLineInd ("if (" + GenerateGetMemberValue (mem, ob, isValueList) + " == null) {");
2117 GenerateSetMemberValue (mem, ob, GenerateInitializeList (mem.TypeData), isValueList);
2123 bool IsReadOnly (XmlTypeMapping map, XmlTypeMapMember member, TypeData memType, bool isValueList)
2125 if (isValueList) return !memType.HasPublicConstructor;
2126 else return member.IsReadOnly (map.TypeData.Type) || !memType.HasPublicConstructor;
2129 void GenerateSetMemberValue (XmlTypeMapMember member, string ob, string value, bool isValueList)
2131 GenerateSetMemberValue (member, ob, value, isValueList, false);
2134 void GenerateSetMemberValue (XmlTypeMapMember member, string ob, string value, bool isValueList, bool initializingMember)
2137 WriteLine (ob + "[" + member.GlobalIndex + "] = " + value + ";");
2138 if (member.IsOptionalValueType) {
2139 string val = initializingMember ? "false" : "true";
2140 WriteLine (ob + "[" + member.SpecifiedGlobalIndex + "] = " + val + ";");
2144 WriteLine (ob + ".@" + member.Name + " = " + value + ";");
2145 if (member.IsOptionalValueType && member.IsValueSpecifiedSettable ()) {
2146 string val = initializingMember ? "false" : "true";
2147 WriteLine (ob + "." + member.Name + "Specified = " + val + ";");
2152 void GenerateSetMemberValueFromAttr (XmlTypeMapMember member, string ob, string value, bool isValueList)
2154 // Enumeration values specified in custom attributes are stored as integer
2155 // values if the custom attribute property is of type object. So, it is
2156 // necessary to convert to the enum type before asigning the value to the field.
2158 if (member.TypeData.Type.IsEnum)
2159 value = GetCast (member.TypeData.Type, value);
2160 GenerateSetMemberValue (member, ob, value, isValueList, true);
2163 string GenerateReadObjectElement (XmlTypeMapElementInfo elem)
2165 switch (elem.TypeData.SchemaType)
2167 case SchemaTypes.XmlNode:
2168 return GetReadXmlNode (elem.TypeData, true);
2170 case SchemaTypes.Primitive:
2171 case SchemaTypes.Enum:
2172 return GenerateReadPrimitiveValue (elem);
2174 case SchemaTypes.Array:
2175 return GenerateReadListElement (elem.MappedType, null, GetLiteral(elem.IsNullable), true);
2177 case SchemaTypes.Class:
2178 return GetReadObjectCall (elem.MappedType, GetLiteral(elem.IsNullable), "true");
2180 case SchemaTypes.XmlSerializable:
2181 return GetCast (elem.TypeData, String.Format ("({0}) ReadSerializable (({0}) Activator.CreateInstance(typeof({0}), true))", elem.TypeData.CSharpFullName));
2184 throw new NotSupportedException ("Invalid value type");
2188 string GenerateReadPrimitiveValue (XmlTypeMapElementInfo elem)
2190 if (elem.TypeData.Type == typeof (XmlQualifiedName)) {
2191 if (elem.IsNullable) return "ReadNullableQualifiedName ()";
2192 else return "ReadElementQualifiedName ()";
2194 else if (elem.IsNullable) {
2195 string str = GetStrTempVar ();
2196 WriteLine ("string " + str + " = ReadNullableString ();");
2197 return GenerateGetValueFromXmlString (str, elem.TypeData, elem.MappedType, true);
2200 string str = GetStrTempVar ();
2201 WriteLine ("string " + str + " = Reader.ReadElementString ();");
2202 return GenerateGetValueFromXmlString (str, elem.TypeData, elem.MappedType, false);
2206 string GenerateGetValueFromXmlString (string value, TypeData typeData, XmlTypeMapping typeMap, bool isNullable)
2208 if (typeData.SchemaType == SchemaTypes.Array)
2209 return GenerateReadListString (typeMap, value);
2210 else if (typeData.SchemaType == SchemaTypes.Enum)
2211 return GenerateGetEnumValue (typeMap, value, isNullable);
2212 else if (typeData.Type == typeof (XmlQualifiedName))
2213 return "ToXmlQualifiedName (" + value + ")";
2215 return XmlCustomFormatter.GenerateFromXmlString (typeData, value);
2218 string GenerateReadListElement (XmlTypeMapping typeMap, string list, string isNullable, bool canCreateInstance)
2220 Type listType = typeMap.TypeData.Type;
2221 ListMap listMap = (ListMap)typeMap.ObjectMap;
2222 bool doNullCheck = typeMap.TypeData.Type.IsArray;
2224 if (canCreateInstance && typeMap.TypeData.HasPublicConstructor)
2227 list = GetObTempVar ();
2228 WriteLine (typeMap.TypeData.CSharpFullName + " " + list + " = null;");
2230 WriteLineInd ("if (!ReadNull()) {");
2231 WriteLine (list + " = " + GenerateCreateList (listType) + ";");
2234 WriteLineInd ("if (!ReadNull()) {");
2240 WriteLineInd ("if (((object)" + list + ") == null)");
2241 WriteLine ("throw CreateReadOnlyCollectionException (" + GetLiteral (typeMap.TypeData.CSharpFullName) + ");");
2243 doNullCheck = false;
2246 WriteLine ("throw CreateReadOnlyCollectionException (" + GetLiteral (typeMap.TypeData.CSharpFullName) + ");");
2251 WriteLineInd ("if (Reader.IsEmptyElement) {");
2252 WriteLine ("Reader.Skip();");
2253 if (listType.IsArray)
2254 WriteLine (list + " = (" + typeMap.TypeData.CSharpFullName + ") ShrinkArray (" + list + ", 0, " + GetTypeOf(listType.GetElementType()) + ", false);");
2257 WriteLineInd ("} else {");
2259 string index = GetNumTempVar ();
2260 WriteLine ("int " + index + " = 0;");
2261 WriteLine ("Reader.ReadStartElement();");
2262 WriteLine ("Reader.MoveToContent();");
2265 WriteLine ("while (Reader.NodeType != System.Xml.XmlNodeType.EndElement) ");
2267 WriteLine ("if (Reader.NodeType == System.Xml.XmlNodeType.Element) ");
2271 foreach (XmlTypeMapElementInfo elemInfo in listMap.ItemInfo)
2273 WriteLineInd ((first?"":"else ") + "if (Reader.LocalName == " + GetLiteral (elemInfo.ElementName) + " && Reader.NamespaceURI == " + GetLiteral (elemInfo.Namespace) + ") {");
2274 GenerateAddListValue (typeMap.TypeData, list, index, GenerateReadObjectElement (elemInfo), false);
2275 WriteLine (index + "++;");
2279 if (!first) WriteLine ("else UnknownNode (null);");
2280 else WriteLine ("UnknownNode (null);");
2283 WriteLine ("else UnknownNode (null);");
2285 WriteLine ("Reader.MoveToContent();");
2288 WriteLine ("ReadEndElement();");
2290 if (listType.IsArray)
2291 WriteLine (list + " = (" + typeMap.TypeData.CSharpFullName + ") ShrinkArray (" + list + ", " + index + ", " + GetTypeOf(listType.GetElementType()) + ", false);");
2300 string GenerateReadListString (XmlTypeMapping typeMap, string values)
2302 Type listType = typeMap.TypeData.Type;
2303 ListMap listMap = (ListMap)typeMap.ObjectMap;
2304 string itemType = ToCSharpFullName (listType.GetElementType());
2306 string list = GetObTempVar ();
2307 WriteLine (itemType + "[] " + list + ";");
2309 string var = GetStrTempVar ();
2310 WriteLine ("string " + var + " = " + values + ".Trim();");
2311 WriteLineInd ("if (" + var + " != string.Empty) {");
2313 string valueArray = GetObTempVar ();
2314 WriteLine ("string[] " + valueArray + " = " + var + ".Split (' ');");
2316 WriteLine (list + " = new " + GetArrayDeclaration (listType, valueArray + ".Length") + ";");
2318 XmlTypeMapElementInfo info = (XmlTypeMapElementInfo)listMap.ItemInfo[0];
2320 string index = GetNumTempVar ();
2321 WriteLineInd ("for (int " + index + " = 0; " + index + " < " + valueArray + ".Length; " + index + "++)");
2322 WriteLine (list + "[" + index + "] = " + GenerateGetValueFromXmlString (valueArray + "[" + index + "]", info.TypeData, info.MappedType, info.IsNullable) + ";");
2326 WriteLine ("\t" + list + " = new " + GetArrayDeclaration (listType, "0") + ";");
2331 string GetArrayDeclaration (Type type, string length)
2333 Type t = type.GetElementType();
2334 System.Text.StringBuilder sb = new System.Text.StringBuilder ();
2335 sb.Append ('[').Append (length).Append (']');
2338 t = t.GetElementType();
2340 sb.Insert (0, ToCSharpFullName (t));
2341 return sb.ToString ();
2344 void GenerateAddListValue (TypeData listType, string list, string index, string value, bool canCreateInstance)
2346 Type type = listType.Type;
2349 WriteLine (list + " = (" + ToCSharpFullName (type) + ") EnsureArrayIndex (" + list + ", " + index + ", " + GetTypeOf(type.GetElementType()) + ");");
2350 WriteLine (list + "[" + index + "] = " + value + ";");
2352 else // Must be IEnumerable
2354 WriteLine ("if (((object)" + list + ") == null)");
2355 if (canCreateInstance)
2356 WriteLine ("\t" + list + String.Format (" = ({0}) Activator.CreateInstance(typeof({0}), true);", listType.CSharpFullName));
2358 WriteLine ("\tthrow CreateReadOnlyCollectionException (" + GetLiteral (listType.CSharpFullName) + ");");
2360 WriteLine (list + ".Add (" + value + ");");
2364 string GenerateCreateList (Type listType)
2366 if (listType.IsArray)
2367 return "(" + ToCSharpFullName (listType) + ") EnsureArrayIndex (null, 0, " + GetTypeOf(listType.GetElementType()) + ")";
2369 return "new " + ToCSharpFullName (listType) + "()";
2372 string GenerateInitializeList (TypeData listType)
2374 if (listType.Type.IsArray)
2377 return "new " + listType.CSharpFullName + "()";
2380 void GenerateFillerCallbacks ()
2382 foreach (TypeData td in _listsToFill)
2384 string metName = GetFillListName (td);
2385 WriteLine ("void " + metName + " (object list, object source)");
2387 WriteLine ("if (((object)list) == null) throw CreateReadOnlyCollectionException (" + GetLiteral (td.CSharpFullName) + ");");
2390 WriteLine (td.CSharpFullName + " dest = (" + td.CSharpFullName + ") list;");
2391 WriteLine ("foreach (object ob in (IEnumerable)source)");
2392 WriteLine ("\tdest.Add (" + GetCast (td.ListItemTypeData, "ob") + ");");
2398 void GenerateReadXmlNodeElement (XmlTypeMapping typeMap, string isNullable)
2400 WriteLine ("return " + GetReadXmlNode (typeMap.TypeData, false) + ";");
2403 void GenerateReadPrimitiveElement (XmlTypeMapping typeMap, string isNullable)
2405 WriteLine ("XmlQualifiedName t = GetXsiType();");
2406 WriteLine ("if (t == null) t = new XmlQualifiedName (" + GetLiteral(typeMap.XmlType) + ", " + GetLiteral(typeMap.Namespace) + ");");
2407 WriteLine ("return " + GetCast (typeMap.TypeData, "ReadTypedPrimitive (t)") + ";");
2410 void GenerateReadEnumElement (XmlTypeMapping typeMap, string isNullable)
2412 WriteLine ("Reader.ReadStartElement ();");
2413 WriteLine (typeMap.TypeData.CSharpFullName + " res = " + GenerateGetEnumValue (typeMap, "Reader.ReadString()", false) + ";");
2414 WriteLineInd ("if (Reader.NodeType != XmlNodeType.None)");
2415 WriteLineUni ("Reader.ReadEndElement ();");
2416 WriteLine ("return res;");
2419 string GenerateGetEnumValue (XmlTypeMapping typeMap, string val, bool isNullable)
2422 return "(" + val + ") != null ? " + GetGetEnumValueName (typeMap) + " (" + val + ") : (" + typeMap.TypeData.CSharpFullName + "?) null";
2424 return GetGetEnumValueName (typeMap) + " (" + val + ")";
2427 void GenerateGetEnumValueMethod (XmlTypeMapping typeMap)
2429 string metName = GetGetEnumValueName (typeMap);
2430 EnumMap map = (EnumMap) typeMap.ObjectMap;
2434 string switchMethod = metName + "_Switch";
2435 WriteLine (typeMap.TypeData.CSharpFullName + " " + metName + " (string xmlName)");
2437 WriteLine ("xmlName = xmlName.Trim();");
2438 WriteLine ("if (xmlName.Length == 0) return (" + typeMap.TypeData.CSharpFullName + ")0;");
2439 WriteLine (typeMap.TypeData.CSharpFullName + " sb = (" + typeMap.TypeData.CSharpFullName + ")0;");
2440 WriteLine ("string[] enumNames = xmlName.Split (null);");
2441 WriteLine ("foreach (string name in enumNames)");
2443 WriteLine ("if (name == string.Empty) continue;");
2444 WriteLine ("sb |= " + switchMethod + " (name); ");
2446 WriteLine ("return sb;");
2449 metName = switchMethod;
2452 WriteLine (typeMap.TypeData.CSharpFullName + " " + metName + " (string xmlName)");
2454 GenerateGetSingleEnumValue (typeMap, "xmlName");
2459 void GenerateGetSingleEnumValue (XmlTypeMapping typeMap, string val)
2461 EnumMap map = (EnumMap) typeMap.ObjectMap;
2462 WriteLine ("switch (" + val + ")");
2464 foreach (EnumMap.EnumMapMember mem in map.Members)
2466 WriteLine ("case " + GetLiteral (mem.XmlName) + ": return " + typeMap.TypeData.CSharpFullName + ".@" + mem.EnumName + ";");
2468 WriteLineInd ("default:");
2469 WriteLine ("throw CreateUnknownConstantException (" + val + ", typeof(" + typeMap.TypeData.CSharpFullName + "));");
2474 void GenerateReadXmlSerializableElement (XmlTypeMapping typeMap, string isNullable)
2476 WriteLine ("Reader.MoveToContent ();");
2477 WriteLine ("if (Reader.NodeType == XmlNodeType.Element)");
2480 WriteLineInd ("if (Reader.LocalName == " + GetLiteral (typeMap.ElementName) + " && Reader.NamespaceURI == " + GetLiteral (typeMap.Namespace) + ")");
2481 WriteLine (String.Format ("\treturn ({0}) ReadSerializable (({0}) Activator.CreateInstance(typeof({0}), true));", typeMap.TypeData.CSharpFullName));
2483 if (!typeMap.IsAny) {
2485 WriteLine ("\tthrow CreateUnknownNodeException ();");
2488 WriteLine ("else UnknownNode (null);");
2490 WriteLine ("return default (" + typeMap.TypeData.CSharpFullName + ");");
2493 void GenerateReadInitCallbacks ()
2495 WriteLine ("protected override void InitCallbacks ()");
2498 if (_format == SerializationFormat.Encoded)
2500 foreach (XmlMapping xmap in _mapsToGenerate)
2502 XmlTypeMapping map = xmap as XmlTypeMapping;
2503 if (map == null) continue;
2504 if (map.TypeData.SchemaType == SchemaTypes.Class || map.TypeData.SchemaType == SchemaTypes.Enum)
2505 WriteMetCall ("AddReadCallback", GetLiteral (map.XmlType), GetLiteral(map.Namespace), GetTypeOf(map.TypeData.Type), "new XmlSerializationReadCallback (" + GetReadObjectName (map) + ")");
2512 WriteLine ("protected override void InitIDs ()");
2518 void GenerateFixupCallbacks ()
2520 foreach (XmlMapping map in _fixupCallbacks)
2522 bool isList = map is XmlMembersMapping;
2523 string tname = !isList ? ((XmlTypeMapping)map).TypeData.CSharpFullName : "object[]";
2524 WriteLine ("void " + GetFixupCallbackName (map) + " (object obfixup)");
2526 WriteLine ("Fixup fixup = (Fixup)obfixup;");
2527 WriteLine (tname + " source = (" + tname + ") fixup.Source;");
2528 WriteLine ("string[] ids = fixup.Ids;");
2531 ClassMap cmap = (ClassMap)map.ObjectMap;
2532 ICollection members = cmap.ElementMembers;
2533 if (members != null) {
2534 foreach (XmlTypeMapMember member in members)
2536 WriteLineInd ("if (ids[" + member.Index + "] != null)");
2537 string val = "GetTarget(ids[" + member.Index + "])";
2538 if (!isList) val = GetCast (member.TypeData, val);
2539 GenerateSetMemberValue (member, "source", val, isList);
2548 string GetReadXmlNode (TypeData type, bool wrapped)
2550 if (type.Type == typeof (XmlDocument))
2551 return GetCast (type, TypeTranslator.GetTypeData (typeof(XmlDocument)), "ReadXmlDocument (" + GetLiteral(wrapped) + ")");
2553 return GetCast (type, TypeTranslator.GetTypeData (typeof(XmlNode)), "ReadXmlNode (" + GetLiteral(wrapped) + ")");
2558 #region Helper methods
2560 //*******************************************************
2564 ArrayList _listsToFill = new ArrayList ();
2565 Hashtable _hookVariables;
2566 Stack _hookContexts;
2567 Stack _hookOpenHooks;
2570 public HookType HookType;
2572 public string Member;
2573 public HookDir Direction;
2578 _hookContexts = new Stack ();
2579 _hookOpenHooks = new Stack ();
2580 _hookVariables = new Hashtable ();
2583 void PushHookContext ()
2585 _hookContexts.Push (_hookVariables);
2586 _hookVariables = (Hashtable) _hookVariables.Clone ();
2589 void PopHookContext ()
2591 _hookVariables = (Hashtable) _hookContexts.Pop ();
2594 void SetHookVar (string var, string value)
2596 _hookVariables [var] = value;
2599 bool GenerateReadHook (HookType hookType, Type type)
2601 return GenerateHook (hookType, HookDir.Read, type, null);
2604 bool GenerateWriteHook (HookType hookType, Type type)
2606 return GenerateHook (hookType, HookDir.Write, type, null);
2609 bool GenerateWriteMemberHook (Type type, XmlTypeMapMember member)
2611 SetHookVar ("$MEMBER", member.Name);
2612 return GenerateHook (HookType.member, HookDir.Write, type, member.Name);
2615 bool GenerateReadMemberHook (Type type, XmlTypeMapMember member)
2617 SetHookVar ("$MEMBER", member.Name);
2618 return GenerateHook (HookType.member, HookDir.Read, type, member.Name);
2621 bool GenerateReadArrayMemberHook (Type type, XmlTypeMapMember member, string index)
2623 SetHookVar ("$INDEX", index);
2624 return GenerateReadMemberHook (type, member);
2627 bool MemberHasReadReplaceHook (Type type, XmlTypeMapMember member)
2629 if (_config == null) return false;
2630 return _config.GetHooks (HookType.member, HookDir.Read, HookAction.Replace, type, member.Name).Count > 0;
2633 bool GenerateHook (HookType hookType, HookDir dir, Type type, string member)
2635 GenerateHooks (hookType, dir, type, null, HookAction.InsertBefore);
2636 if (GenerateHooks (hookType, dir, type, null, HookAction.Replace))
2638 GenerateHooks (hookType, dir, type, null, HookAction.InsertAfter);
2643 HookInfo hi = new HookInfo ();
2644 hi.HookType = hookType;
2648 _hookOpenHooks.Push (hi);
2653 void GenerateEndHook ()
2655 HookInfo hi = (HookInfo) _hookOpenHooks.Pop();
2656 GenerateHooks (hi.HookType, hi.Direction, hi.Type, hi.Member, HookAction.InsertAfter);
2659 bool GenerateHooks (HookType hookType, HookDir dir, Type type, string member, HookAction action)
2661 if (_config == null) return false;
2662 ArrayList hooks = _config.GetHooks (hookType, dir, action, type, null);
2663 if (hooks.Count == 0) return false;
2664 foreach (Hook hook in hooks)
2666 string code = hook.GetCode (action);
2667 foreach (DictionaryEntry de in _hookVariables)
2668 code = code.Replace ((string)de.Key, (string)de.Value);
2669 WriteMultilineCode (code);
2674 string GetRootTypeName ()
2676 if (_typeMap is XmlTypeMapping) return GetTypeFullName (((XmlTypeMapping)_typeMap).TypeData);
2677 else return "object[]";
2680 string GetNumTempVar ()
2682 return "n" + (_tempVarId++);
2685 string GetObTempVar ()
2687 return "o" + (_tempVarId++);
2690 string GetStrTempVar ()
2692 return "s" + (_tempVarId++);
2695 string GetBoolTempVar ()
2697 return "b" + (_tempVarId++);
2700 string GetUniqueName (string uniqueGroup, object ob, string name)
2702 name = CodeIdentifier.MakeValid (name.Replace ("[]","_array"));
2703 Hashtable names = (Hashtable) _uniqueNames [uniqueGroup];
2704 if (names == null) {
2705 names = new Hashtable ();
2706 _uniqueNames [uniqueGroup] = names;
2709 string res = (string) names [ob];
2710 if (res != null) return res;
2712 foreach (string n in names.Values)
2713 if (n == name) return GetUniqueName (uniqueGroup, ob, name + (_methodId++));
2719 void RegisterReferencingMap (XmlTypeMapping typeMap)
2721 if (typeMap != null && !_mapsToGenerate.Contains (typeMap))
2722 _mapsToGenerate.Add (typeMap);
2725 string GetWriteObjectName (XmlTypeMapping typeMap)
2727 if (!_mapsToGenerate.Contains (typeMap)) _mapsToGenerate.Add (typeMap);
2728 return GetUniqueName ("rw", typeMap, "WriteObject_" + typeMap.XmlType);
2731 string GetReadObjectName (XmlTypeMapping typeMap)
2733 if (!_mapsToGenerate.Contains (typeMap)) _mapsToGenerate.Add (typeMap);
2734 return GetUniqueName ("rr", typeMap, "ReadObject_" + typeMap.XmlType);
2737 string GetGetEnumValueName (XmlTypeMapping typeMap)
2739 if (!_mapsToGenerate.Contains (typeMap)) _mapsToGenerate.Add (typeMap);
2740 return GetUniqueName ("ge", typeMap, "GetEnumValue_" + typeMap.XmlType);
2743 string GetWriteObjectCallbackName (XmlTypeMapping typeMap)
2745 if (!_mapsToGenerate.Contains (typeMap)) _mapsToGenerate.Add (typeMap);
2746 return GetUniqueName ("wc", typeMap, "WriteCallback_" + typeMap.XmlType);
2749 string GetFixupCallbackName (XmlMapping typeMap)
2751 if (!_mapsToGenerate.Contains (typeMap)) _mapsToGenerate.Add (typeMap);
2753 if (typeMap is XmlTypeMapping)
2754 return GetUniqueName ("fc", typeMap, "FixupCallback_" + ((XmlTypeMapping)typeMap).XmlType);
2756 return GetUniqueName ("fc", typeMap, "FixupCallback__Message");
2759 string GetUniqueClassName (string s)
2761 return classNames.AddUnique (CodeIdentifier.MakeValid (s), null);
2764 string GetReadObjectCall (XmlTypeMapping typeMap, string isNullable, string checkType)
2766 if (_format == SerializationFormat.Literal)
2767 return GetReadObjectName (typeMap) + " (" + isNullable + ", " + checkType + ")";
2769 return GetCast (typeMap.TypeData, GetReadObjectName (typeMap) + " ()");
2772 string GetFillListName (TypeData td)
2774 if (!_listsToFill.Contains (td)) _listsToFill.Add (td);
2775 return GetUniqueName ("fl", td, "Fill_" + CodeIdentifier.MakeValid (td.CSharpName));
2778 string GetCast (TypeData td, TypeData tdval, string val)
2780 if (td.CSharpFullName == tdval.CSharpFullName) return val;
2781 else return GetCast (td, val);
2784 string GetCast (TypeData td, string val)
2786 return "((" + GetTypeFullName (td) + ") " + val + ")";
2789 string GetCast (Type td, string val)
2791 return "((" + ToCSharpFullName (td) + ") " + val + ")";
2794 string GetTypeOf (TypeData td)
2796 return "typeof(" + GetTypeFullName (td) + ")";
2799 string GetTypeOf (Type td)
2801 return "typeof(" + ToCSharpFullName (td) + ")";
2804 string GetTypeFullName (TypeData td) {
2805 if (td.IsNullable && td.IsValueType)
2806 return td.CSharpFullName + "?";
2808 return td.CSharpFullName;
2811 string GetLiteral (object ob)
2813 if (ob == null) return "null";
2814 if (ob is string) return "\"" + ob.ToString().Replace("\"","\"\"") + "\"";
2815 if (ob is char) return (char) ob == '\'' ? "'\\''" : "'" + ob.ToString () + "'";
2816 if (ob is DateTime) return "new DateTime (" + ((DateTime) ob).Ticks + ")";
2818 if (ob is DateTimeOffset) return "new DateTimeOffset (" + ((DateTimeOffset) ob).Ticks + ")";
2820 if (ob is TimeSpan) return "new TimeSpan (" + ((TimeSpan) ob).Ticks + ")";
2821 if (ob is bool) return ((bool)ob) ? "true" : "false";
2822 if (ob is XmlQualifiedName) {
2823 XmlQualifiedName qn = (XmlQualifiedName)ob;
2824 return "new XmlQualifiedName (" + GetLiteral(qn.Name) + "," + GetLiteral(qn.Namespace) + ")";
2827 string typeName = ToCSharpFullName (ob.GetType ());
2828 StringBuilder sb = new StringBuilder ();
2829 string namedValue = Enum.Format (ob.GetType (), ob, "g");
2830 string[] names = namedValue.Split (',');
2831 foreach (string name in names) {
2832 // individual named constants can be seperated by a comma
2833 // combined with some additional whitespace characters
2834 string cleanName = name.Trim ();
2835 if (cleanName.Length == 0)
2841 sb.Append (typeName);
2843 sb.Append (cleanName);
2845 return sb.ToString ();
2848 return (ob is IFormattable) ? ((IFormattable) ob).ToString (null, CultureInfo.InvariantCulture) : ob.ToString ();
2851 void WriteLineInd (string code)
2857 void WriteLineUni (string code)
2859 if (_indent > 0) _indent--;
2863 void Write (string code)
2865 if (code.Length > 0)
2866 _writer.Write (new String ('\t', _indent));
2867 _writer.Write (code);
2870 void WriteUni (string code)
2872 if (_indent > 0) _indent--;
2873 _writer.Write (code);
2874 _writer.WriteLine (string.Empty);
2877 void WriteLine (string code)
2879 if (code.Length > 0)
2880 _writer.Write (new String ('\t',_indent));
2881 _writer.WriteLine (code);
2884 void WriteMultilineCode (string code)
2886 string tabs = new string ('\t',_indent);
2887 code = code.Replace ("\r","");
2888 code = code.Replace ("\t","");
2889 while (code.StartsWith ("\n")) code = code.Substring (1);
2890 while (code.EndsWith ("\n")) code = code.Substring (0, code.Length - 1);
2891 code = code.Replace ("\n", "\n" + tabs);
2895 string Params (params string[] pars)
2898 foreach (string p in pars)
2900 if (res != "") res += ", ";
2906 void WriteMetCall (string method, params string[] pars)
2908 WriteLine (method + " (" + Params (pars) + ");");
2920 internal class GenerationResult
2922 public XmlMapping Mapping;
2923 public string ReaderClassName;
2924 public string ReadMethodName;
2925 public string WriterClassName;
2926 public string WriteMethodName;
2927 public string Namespace;
2929 public string SerializerClassName;
2931 public string BaseSerializerClassName;
2932 public string ImplementationClassName;