//------------------------------------------------------------------------------ // // Copyright (c) Microsoft Corporation. All rights reserved. // // Microsoft //------------------------------------------------------------------------------ namespace System.Xml.Serialization { using System; using System.Xml.Schema; using System.Collections; using System.Collections.Generic; using System.ComponentModel; using System.Reflection; using System.Xml.Serialization.Configuration; using System.CodeDom; using System.CodeDom.Compiler; using System.Collections.Specialized; using System.Globalization; using System.Security.Permissions; using System.Xml.Serialization.Advanced; #if DEBUG using System.Diagnostics; #endif /// /// /// /// [To be supplied.] /// public class XmlSchemaImporter : SchemaImporter { /// /// /// [To be supplied.] /// public XmlSchemaImporter(XmlSchemas schemas) : base(schemas, CodeGenerationOptions.GenerateProperties, null, new ImportContext()) {} /// /// /// [To be supplied.] /// public XmlSchemaImporter(XmlSchemas schemas, CodeIdentifiers typeIdentifiers) : base(schemas, CodeGenerationOptions.GenerateProperties, null, new ImportContext(typeIdentifiers, false)) {} /// /// /// [To be supplied.] /// public XmlSchemaImporter(XmlSchemas schemas, CodeIdentifiers typeIdentifiers, CodeGenerationOptions options) : base(schemas, options, null, new ImportContext(typeIdentifiers, false)) {} /// /// /// [To be supplied.] /// public XmlSchemaImporter(XmlSchemas schemas, CodeGenerationOptions options, ImportContext context) : base(schemas, options, null, context){} /// /// /// [To be supplied.] /// public XmlSchemaImporter(XmlSchemas schemas, CodeGenerationOptions options, CodeDomProvider codeProvider, ImportContext context) : base(schemas, options, codeProvider, context){ } /// /// /// [To be supplied.] /// public XmlTypeMapping ImportDerivedTypeMapping(XmlQualifiedName name, Type baseType) { return ImportDerivedTypeMapping(name, baseType, false); } internal bool GenerateOrder { get { return (Options & CodeGenerationOptions.GenerateOrder) != 0; } } internal TypeMapping GetDefaultMapping(TypeFlags flags) { PrimitiveMapping mapping = new PrimitiveMapping(); mapping.TypeDesc = Scope.GetTypeDesc("string", XmlSchema.Namespace, flags); mapping.TypeName = mapping.TypeDesc.DataType.Name; mapping.Namespace = XmlSchema.Namespace; return mapping; } /// /// /// [To be supplied.] /// public XmlTypeMapping ImportDerivedTypeMapping(XmlQualifiedName name, Type baseType, bool baseTypeCanBeIndirect) { ElementAccessor element = ImportElement(name, typeof(TypeMapping), baseType); if (element.Mapping is StructMapping) { MakeDerived((StructMapping)element.Mapping, baseType, baseTypeCanBeIndirect); } else if (baseType != null) { if (element.Mapping is ArrayMapping) { // in the case of the ArrayMapping we can use the top-level StructMapping, because it does not have base base type element.Mapping = ((ArrayMapping)element.Mapping).TopLevelMapping; MakeDerived((StructMapping)element.Mapping, baseType, baseTypeCanBeIndirect); } else { // Element '{0}' from namespace '{1}' is not a complex type and cannot be used as a {2}. throw new InvalidOperationException(Res.GetString(Res.XmlBadBaseElement, name.Name, name.Namespace, baseType.FullName)); } } return new XmlTypeMapping(Scope, element); } /// /// /// [To be supplied.] /// public XmlTypeMapping ImportSchemaType(XmlQualifiedName typeName) { return ImportSchemaType(typeName, null, false); } /// /// /// [To be supplied.] /// public XmlTypeMapping ImportSchemaType(XmlQualifiedName typeName, Type baseType) { return ImportSchemaType(typeName, baseType, false); } /// /// /// [To be supplied.] /// public XmlTypeMapping ImportSchemaType(XmlQualifiedName typeName, Type baseType, bool baseTypeCanBeIndirect) { TypeMapping typeMapping = ImportType(typeName, typeof(TypeMapping), baseType, TypeFlags.CanBeElementValue, true); typeMapping.ReferencedByElement = false; ElementAccessor accessor = new ElementAccessor(); accessor.IsTopLevelInSchema = true; // false accessor.Name = typeName.Name; accessor.Namespace = typeName.Namespace; accessor.Mapping = typeMapping; if (typeMapping is SpecialMapping && ((SpecialMapping)typeMapping).NamedAny) accessor.Any = true; accessor.IsNullable = typeMapping.TypeDesc.IsNullable; accessor.Form = XmlSchemaForm.Qualified; if (accessor.Mapping is StructMapping) { MakeDerived((StructMapping)accessor.Mapping, baseType, baseTypeCanBeIndirect); } else if (baseType != null) { if (accessor.Mapping is ArrayMapping) { // in the case of the ArrayMapping we can use the top-level StructMapping, because it does not have base base type accessor.Mapping = ((ArrayMapping)accessor.Mapping).TopLevelMapping; MakeDerived((StructMapping)accessor.Mapping, baseType, baseTypeCanBeIndirect); } else { // Type '{0}' from namespace '{1}' is not a complex type and cannot be used as a {2}. throw new InvalidOperationException(Res.GetString(Res.XmlBadBaseType, typeName.Name, typeName.Namespace, baseType.FullName)); } } return new XmlTypeMapping(Scope, accessor); } /// /// /// [To be supplied.] /// public XmlTypeMapping ImportTypeMapping(XmlQualifiedName name) { return ImportDerivedTypeMapping(name, null); } /// /// /// [To be supplied.] /// public XmlMembersMapping ImportMembersMapping(XmlQualifiedName name) { return new XmlMembersMapping(Scope, ImportElement(name, typeof(MembersMapping), null), XmlMappingAccess.Read | XmlMappingAccess.Write); } /// public XmlMembersMapping ImportAnyType(XmlQualifiedName typeName, string elementName) { TypeMapping typeMapping = ImportType(typeName, typeof(MembersMapping), null, TypeFlags.CanBeElementValue, true); MembersMapping mapping = typeMapping as MembersMapping; if (mapping == null) { XmlSchemaComplexType type = new XmlSchemaComplexType(); XmlSchemaSequence seq = new XmlSchemaSequence(); type.Particle = seq; XmlSchemaElement element = new XmlSchemaElement(); element.Name = elementName; element.SchemaTypeName = typeName; seq.Items.Add(element); mapping = ImportMembersType(type, typeName.Namespace, elementName); } if (mapping.Members.Length != 1 || !mapping.Members[0].Accessor.Any) return null; mapping.Members[0].Name = elementName; ElementAccessor accessor = new ElementAccessor(); accessor.Name = elementName; accessor.Namespace = typeName.Namespace; accessor.Mapping = mapping; accessor.Any = true; XmlSchemaObject xso = Schemas.SchemaSet.GlobalTypes[typeName]; if (xso != null) { XmlSchema schema = xso.Parent as XmlSchema; if (schema != null) { accessor.Form = schema.ElementFormDefault == XmlSchemaForm.None ? XmlSchemaForm.Unqualified : schema.ElementFormDefault; } } XmlMembersMapping members = new XmlMembersMapping(Scope, accessor, XmlMappingAccess.Read | XmlMappingAccess.Write); return members; } /// /// /// [To be supplied.] /// public XmlMembersMapping ImportMembersMapping(XmlQualifiedName[] names) { return ImportMembersMapping(names, null, false); } /// /// /// [To be supplied.] /// public XmlMembersMapping ImportMembersMapping(XmlQualifiedName[] names, Type baseType, bool baseTypeCanBeIndirect) { CodeIdentifiers memberScope = new CodeIdentifiers(); memberScope.UseCamelCasing = true; MemberMapping[] members = new MemberMapping[names.Length]; for (int i = 0; i < names.Length; i++) { XmlQualifiedName name = names[i]; ElementAccessor accessor = ImportElement(name, typeof(TypeMapping), baseType); if (baseType != null && accessor.Mapping is StructMapping) MakeDerived((StructMapping)accessor.Mapping, baseType, baseTypeCanBeIndirect); MemberMapping member = new MemberMapping(); member.Name = CodeIdentifier.MakeValid(Accessor.UnescapeName(accessor.Name)); member.Name = memberScope.AddUnique(member.Name, member); member.TypeDesc = accessor.Mapping.TypeDesc; member.Elements = new ElementAccessor[] { accessor }; members[i] = member; } MembersMapping mapping = new MembersMapping(); mapping.HasWrapperElement = false; mapping.TypeDesc = Scope.GetTypeDesc(typeof(object[])); mapping.Members = members; ElementAccessor element = new ElementAccessor(); element.Mapping = mapping; return new XmlMembersMapping(Scope, element, XmlMappingAccess.Read | XmlMappingAccess.Write); } public XmlMembersMapping ImportMembersMapping(string name, string ns, SoapSchemaMember[] members) { XmlSchemaComplexType type = new XmlSchemaComplexType(); XmlSchemaSequence seq = new XmlSchemaSequence(); type.Particle = seq; foreach (SoapSchemaMember member in members) { XmlSchemaElement element = new XmlSchemaElement(); element.Name = member.MemberName; element.SchemaTypeName = member.MemberType; seq.Items.Add(element); } MembersMapping mapping = ImportMembersType(type, null, name); ElementAccessor accessor = new ElementAccessor(); accessor.Name = Accessor.EscapeName(name); accessor.Namespace = ns; accessor.Mapping = mapping; accessor.IsNullable = false; accessor.Form = XmlSchemaForm.Qualified; return new XmlMembersMapping(Scope, accessor, XmlMappingAccess.Read | XmlMappingAccess.Write); } ElementAccessor ImportElement(XmlQualifiedName name, Type desiredMappingType, Type baseType) { XmlSchemaElement element = FindElement(name); ElementAccessor accessor = (ElementAccessor)ImportedElements[element]; if (accessor != null) return accessor; accessor = ImportElement(element, string.Empty, desiredMappingType, baseType, name.Namespace, true); ElementAccessor existing = (ElementAccessor)ImportedElements[element]; if (existing != null) { return existing; } ImportedElements.Add(element, accessor); return accessor; } ElementAccessor ImportElement(XmlSchemaElement element, string identifier, Type desiredMappingType, Type baseType, string ns, bool topLevelElement) { if (!element.RefName.IsEmpty) { // we cannot re-use the accessor for the element refs ElementAccessor topAccessor = ImportElement(element.RefName, desiredMappingType, baseType); if (element.IsMultipleOccurrence && topAccessor.Mapping is ArrayMapping) { ElementAccessor refAccessor = topAccessor.Clone(); refAccessor.IsTopLevelInSchema = false; refAccessor.Mapping.ReferencedByElement = true; return refAccessor; } return topAccessor; } if (element.Name.Length == 0) { XmlQualifiedName parentType = XmlSchemas.GetParentName(element); throw new InvalidOperationException(Res.GetString(Res.XmlElementHasNoName, parentType.Name, parentType.Namespace)); } string unescapedName = Accessor.UnescapeName(element.Name); if (identifier.Length == 0) identifier = CodeIdentifier.MakeValid(unescapedName); else identifier += CodeIdentifier.MakePascal(unescapedName); TypeMapping mapping = ImportElementType(element, identifier, desiredMappingType, baseType, ns); ElementAccessor accessor = new ElementAccessor(); accessor.IsTopLevelInSchema = element.Parent is XmlSchema; accessor.Name = element.Name; accessor.Namespace = ns; accessor.Mapping = mapping; accessor.IsOptional = element.MinOccurs == 0m; if (element.DefaultValue != null) { accessor.Default = element.DefaultValue; } else if (element.FixedValue != null) { accessor.Default = element.FixedValue; accessor.IsFixed = true; } if (mapping is SpecialMapping && ((SpecialMapping)mapping).NamedAny) accessor.Any = true; accessor.IsNullable = element.IsNillable; if (topLevelElement) { accessor.Form = XmlSchemaForm.Qualified; } else { accessor.Form = ElementForm(ns, element); } return accessor; } TypeMapping ImportElementType(XmlSchemaElement element, string identifier, Type desiredMappingType, Type baseType, string ns) { TypeMapping mapping; if (!element.SchemaTypeName.IsEmpty) { mapping = ImportType(element.SchemaTypeName, desiredMappingType, baseType, TypeFlags.CanBeElementValue, false); if (!mapping.ReferencedByElement) { object type = FindType(element.SchemaTypeName, TypeFlags.CanBeElementValue); XmlSchemaObject parent = element; while (parent.Parent != null && type != parent) { parent = parent.Parent; } mapping.ReferencedByElement = (type != parent); } } else if (element.SchemaType != null) { if (element.SchemaType is XmlSchemaComplexType) mapping = ImportType((XmlSchemaComplexType)element.SchemaType, ns, identifier, desiredMappingType, baseType, TypeFlags.CanBeElementValue); else mapping = ImportDataType((XmlSchemaSimpleType)element.SchemaType, ns, identifier, baseType, TypeFlags.CanBeElementValue | TypeFlags.CanBeAttributeValue | TypeFlags.CanBeTextValue, false); mapping.ReferencedByElement = true; } else if (!element.SubstitutionGroup.IsEmpty) mapping = ImportElementType(FindElement(element.SubstitutionGroup), identifier, desiredMappingType, baseType, ns); else { if (desiredMappingType == typeof(MembersMapping)) { mapping = ImportMembersType(new XmlSchemaType(), ns, identifier); } else { mapping = ImportRootMapping(); } } if (!(desiredMappingType.IsAssignableFrom(mapping.GetType()))) throw new InvalidOperationException(Res.GetString(Res.XmlElementImportedTwice, element.Name, ns, mapping.GetType().Name, desiredMappingType.Name)); // let the extensions to run if (!mapping.TypeDesc.IsMappedType) { RunSchemaExtensions(mapping, element.SchemaTypeName, element.SchemaType, element, TypeFlags.CanBeElementValue); } return mapping; } void RunSchemaExtensions(TypeMapping mapping, XmlQualifiedName qname, XmlSchemaType type, XmlSchemaObject context, TypeFlags flags) { string typeName = null; SchemaImporterExtension typeOwner = null; CodeCompileUnit compileUnit = new CodeCompileUnit(); CodeNamespace mainNamespace = new CodeNamespace(); compileUnit.Namespaces.Add(mainNamespace); if (!qname.IsEmpty) { typeName = FindExtendedType(qname.Name, qname.Namespace, context, compileUnit, mainNamespace, out typeOwner); } else if (type != null) { typeName = FindExtendedType(type, context, compileUnit, mainNamespace, out typeOwner); } else if (context is XmlSchemaAny) { typeName = FindExtendedAnyElement((XmlSchemaAny)context, ((flags & TypeFlags.CanBeTextValue) != 0), compileUnit, mainNamespace, out typeOwner); } if (typeName != null && typeName.Length > 0) { // check if the type name is valid typeName = typeName.Replace('+', '.'); try { CodeGenerator.ValidateIdentifiers(new CodeTypeReference(typeName)); } catch (ArgumentException) { if (qname.IsEmpty) { throw new InvalidOperationException(Res.GetString(Res.XmlImporterExtensionBadLocalTypeName, typeOwner.GetType().FullName, typeName)); } else { throw new InvalidOperationException(Res.GetString(Res.XmlImporterExtensionBadTypeName, typeOwner.GetType().FullName, qname.Name, qname.Namespace, typeName)); } } // foreach (CodeNamespace ns in compileUnit.Namespaces) { CodeGenerator.ValidateIdentifiers(ns); } // mapping.TypeDesc = mapping.TypeDesc.CreateMappedTypeDesc(new MappedTypeDesc(typeName, qname.Name, qname.Namespace, type, context, typeOwner, mainNamespace, compileUnit.ReferencedAssemblies)); if (mapping is ArrayMapping) { TypeMapping top = ((ArrayMapping)mapping).TopLevelMapping; top.TypeName = mapping.TypeName; top.TypeDesc = mapping.TypeDesc; } else { mapping.TypeName = qname.IsEmpty ? null : typeName; } } } string GenerateUniqueTypeName(string desiredName, string ns) { int i = 1; string typeName = desiredName; while (true) { XmlQualifiedName qname = new XmlQualifiedName(typeName, ns); object type = Schemas.Find(qname, typeof(XmlSchemaType)); if (type == null) { break; } typeName = desiredName + i.ToString(CultureInfo.InvariantCulture); i++; } typeName = CodeIdentifier.MakeValid(typeName); return TypeIdentifiers.AddUnique(typeName, typeName); } [PermissionSet(SecurityAction.InheritanceDemand, Name="FullTrust")] internal override void ImportDerivedTypes(XmlQualifiedName baseName) { foreach (XmlSchema schema in Schemas) { if (Schemas.IsReference(schema)) continue; if (XmlSchemas.IsDataSet(schema)) continue; XmlSchemas.Preprocess(schema); foreach (object item in schema.SchemaTypes.Values) { if (item is XmlSchemaType) { XmlSchemaType type = (XmlSchemaType)item; if (type.DerivedFrom == baseName && TypesInUse[type.Name, schema.TargetNamespace] == null) { ImportType(type.QualifiedName, typeof(TypeMapping), null, TypeFlags.CanBeElementValue, false); } } } } } TypeMapping ImportType(XmlQualifiedName name, Type desiredMappingType, Type baseType, TypeFlags flags, bool addref) { if (name.Name == Soap.UrType && name.Namespace == XmlSchema.Namespace) return ImportRootMapping(); object type = FindType(name, flags); TypeMapping mapping = (TypeMapping)ImportedMappings[type]; if (mapping != null && desiredMappingType.IsAssignableFrom(mapping.GetType())) return mapping; if (addref) AddReference(name, TypesInUse, Res.XmlCircularTypeReference); if (type is XmlSchemaComplexType) { mapping = ImportType((XmlSchemaComplexType)type, name.Namespace, name.Name, desiredMappingType, baseType, flags); } else if (type is XmlSchemaSimpleType) mapping = ImportDataType((XmlSchemaSimpleType)type, name.Namespace, name.Name, baseType, flags, false); else throw new InvalidOperationException(Res.GetString(Res.XmlInternalError)); if (addref && name.Namespace != XmlSchema.Namespace) RemoveReference(name, TypesInUse); return mapping; } TypeMapping ImportType(XmlSchemaComplexType type, string typeNs, string identifier, Type desiredMappingType, Type baseType, TypeFlags flags) { if (type.Redefined != null) { // we do not support redefine in the current version throw new NotSupportedException(Res.GetString(Res.XmlUnsupportedRedefine, type.Name, typeNs)); } if (desiredMappingType == typeof(TypeMapping)) { TypeMapping mapping = null; if (baseType == null) { if ((mapping = ImportArrayMapping(type, identifier, typeNs, false)) == null) { mapping = ImportAnyMapping(type, identifier, typeNs, false); } } if (mapping == null) { mapping = ImportStructType(type, typeNs, identifier, baseType, false); if (mapping != null && type.Name != null && type.Name.Length != 0) ImportDerivedTypes(new XmlQualifiedName(identifier, typeNs)); } return mapping; } else if (desiredMappingType == typeof(MembersMapping)) return ImportMembersType(type, typeNs, identifier); else throw new ArgumentException(Res.GetString(Res.XmlInternalError), "desiredMappingType"); } MembersMapping ImportMembersType(XmlSchemaType type, string typeNs, string identifier) { if (!type.DerivedFrom.IsEmpty) throw new InvalidOperationException(Res.GetString(Res.XmlMembersDeriveError)); CodeIdentifiers memberScope = new CodeIdentifiers(); memberScope.UseCamelCasing = true; bool needExplicitOrder = false; MemberMapping[] members = ImportTypeMembers(type, typeNs, identifier, memberScope, new CodeIdentifiers(), new NameTable(), ref needExplicitOrder, false, false); MembersMapping mappings = new MembersMapping(); mappings.HasWrapperElement = true; mappings.TypeDesc = Scope.GetTypeDesc(typeof(object[])); mappings.Members = members; return mappings; } StructMapping ImportStructType(XmlSchemaType type, string typeNs, string identifier, Type baseType, bool arrayLike) { TypeDesc baseTypeDesc = null; TypeMapping baseMapping = null; bool isRootType = false; if (!type.DerivedFrom.IsEmpty) { baseMapping = ImportType(type.DerivedFrom, typeof(TypeMapping), null, TypeFlags.CanBeElementValue | TypeFlags.CanBeTextValue, false); if (baseMapping is StructMapping) baseTypeDesc = ((StructMapping)baseMapping).TypeDesc; else if (baseMapping is ArrayMapping) { baseMapping = ((ArrayMapping)baseMapping).TopLevelMapping; if (baseMapping != null) { baseMapping.ReferencedByTopLevelElement = false; baseMapping.ReferencedByElement = true; baseTypeDesc = baseMapping.TypeDesc; } } else baseMapping = null; } if (baseTypeDesc == null && baseType != null) baseTypeDesc = Scope.GetTypeDesc(baseType); if (baseMapping == null) { baseMapping = GetRootMapping(); isRootType = true; } Mapping previousMapping = (Mapping)ImportedMappings[type]; if (previousMapping != null) { if (previousMapping is StructMapping) { return (StructMapping)previousMapping; } else if (arrayLike && previousMapping is ArrayMapping){ ArrayMapping arrayMapping = (ArrayMapping)previousMapping; if (arrayMapping.TopLevelMapping != null) { return arrayMapping.TopLevelMapping; } } else { throw new InvalidOperationException(Res.GetString(Res.XmlTypeUsedTwice, type.QualifiedName.Name, type.QualifiedName.Namespace)); } } StructMapping structMapping = new StructMapping(); structMapping.IsReference = Schemas.IsReference(type); TypeFlags flags = TypeFlags.Reference; if (type is XmlSchemaComplexType) { if (((XmlSchemaComplexType)type).IsAbstract) flags |= TypeFlags.Abstract; } identifier = Accessor.UnescapeName(identifier); string typeName = type.Name == null || type.Name.Length == 0 ? GenerateUniqueTypeName(identifier, typeNs) : GenerateUniqueTypeName(identifier); structMapping.TypeDesc = new TypeDesc(typeName, typeName, TypeKind.Struct, baseTypeDesc, flags); structMapping.Namespace = typeNs; structMapping.TypeName = type.Name == null || type.Name.Length == 0 ? null : identifier; structMapping.BaseMapping = (StructMapping)baseMapping; if (!arrayLike) ImportedMappings.Add(type, structMapping); CodeIdentifiers members = new CodeIdentifiers(); CodeIdentifiers membersScope = structMapping.BaseMapping.Scope.Clone(); members.AddReserved(typeName); membersScope.AddReserved(typeName); AddReservedIdentifiersForDataBinding(members); if (isRootType) AddReservedIdentifiersForDataBinding(membersScope); bool needExplicitOrder = false; structMapping.Members = ImportTypeMembers(type, typeNs, identifier, members, membersScope, structMapping, ref needExplicitOrder, true, true); if (!IsAllGroup(type)) { if (needExplicitOrder && !GenerateOrder) { structMapping.SetSequence(); } else if (GenerateOrder) { structMapping.IsSequence = true; } } for (int i = 0; i < structMapping.Members.Length; i++) { StructMapping declaringMapping; MemberMapping baseMember = ((StructMapping)baseMapping).FindDeclaringMapping(structMapping.Members[i], out declaringMapping, structMapping.TypeName); if (baseMember != null && baseMember.TypeDesc != structMapping.Members[i].TypeDesc) throw new InvalidOperationException(Res.GetString(Res.XmlIllegalOverride, type.Name, baseMember.Name, baseMember.TypeDesc.FullName, structMapping.Members[i].TypeDesc.FullName, declaringMapping.TypeDesc.FullName)); } structMapping.Scope = membersScope; Scope.AddTypeMapping(structMapping); return structMapping; } bool IsAllGroup(XmlSchemaType type) { TypeItems items = GetTypeItems(type); return (items.Particle != null) && (items.Particle is XmlSchemaAll); } StructMapping ImportStructDataType(XmlSchemaSimpleType dataType, string typeNs, string identifier, Type baseType) { identifier = Accessor.UnescapeName(identifier); string typeName = GenerateUniqueTypeName(identifier); StructMapping structMapping = new StructMapping(); structMapping.IsReference = Schemas.IsReference(dataType); TypeFlags flags = TypeFlags.Reference; TypeDesc baseTypeDesc = Scope.GetTypeDesc(baseType); structMapping.TypeDesc = new TypeDesc(typeName, typeName, TypeKind.Struct, baseTypeDesc, flags); structMapping.Namespace = typeNs; structMapping.TypeName = identifier; CodeIdentifiers members = new CodeIdentifiers(); members.AddReserved(typeName); AddReservedIdentifiersForDataBinding(members); ImportTextMember(members, new CodeIdentifiers(), null); structMapping.Members = (MemberMapping[])members.ToArray(typeof(MemberMapping)); structMapping.Scope = members; Scope.AddTypeMapping(structMapping); return structMapping; } class TypeItems { internal XmlSchemaObjectCollection Attributes = new XmlSchemaObjectCollection(); internal XmlSchemaAnyAttribute AnyAttribute; internal XmlSchemaGroupBase Particle; internal XmlQualifiedName baseSimpleType; internal bool IsUnbounded; } MemberMapping[] ImportTypeMembers(XmlSchemaType type, string typeNs, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, INameScope elementsScope, ref bool needExplicitOrder, bool order, bool allowUnboundedElements) { TypeItems items = GetTypeItems(type); bool mixed = IsMixed(type); if (mixed) { // check if we can transfer the attribute to the base class XmlSchemaType t = type; while (!t.DerivedFrom.IsEmpty) { t = FindType(t.DerivedFrom, TypeFlags.CanBeElementValue | TypeFlags.CanBeTextValue); if (IsMixed(t)) { // keep the mixed attribute on the base class mixed = false; break; } } } if (items.Particle != null) { ImportGroup(items.Particle, identifier, members, membersScope, elementsScope, typeNs, mixed, ref needExplicitOrder, order, items.IsUnbounded, allowUnboundedElements); } for (int i = 0; i < items.Attributes.Count; i++) { object item = items.Attributes[i]; if (item is XmlSchemaAttribute) { ImportAttributeMember((XmlSchemaAttribute)item, identifier, members, membersScope, typeNs); } else if (item is XmlSchemaAttributeGroupRef) { XmlQualifiedName groupName = ((XmlSchemaAttributeGroupRef)item).RefName; ImportAttributeGroupMembers(FindAttributeGroup(groupName), identifier, members, membersScope, groupName.Namespace); } } if (items.AnyAttribute != null) { ImportAnyAttributeMember(items.AnyAttribute, members, membersScope); } if (items.baseSimpleType != null || (items.Particle == null && mixed)) { ImportTextMember(members, membersScope, mixed ? null : items.baseSimpleType); } ImportXmlnsDeclarationsMember(type, members, membersScope); MemberMapping[] typeMembers = (MemberMapping[])members.ToArray(typeof(MemberMapping)); return typeMembers; } internal static bool IsMixed(XmlSchemaType type) { if (!(type is XmlSchemaComplexType)) return false; XmlSchemaComplexType ct = (XmlSchemaComplexType)type; bool mixed = ct.IsMixed; // check the mixed attribute on the complexContent if (!mixed) { if (ct.ContentModel != null && ct.ContentModel is XmlSchemaComplexContent) { mixed = ((XmlSchemaComplexContent)ct.ContentModel).IsMixed; } } return mixed; } TypeItems GetTypeItems(XmlSchemaType type) { TypeItems items = new TypeItems(); if (type is XmlSchemaComplexType) { XmlSchemaParticle particle = null; XmlSchemaComplexType ct = (XmlSchemaComplexType)type; if (ct.ContentModel != null) { XmlSchemaContent content = ct.ContentModel.Content; if (content is XmlSchemaComplexContentExtension) { XmlSchemaComplexContentExtension extension = (XmlSchemaComplexContentExtension)content; items.Attributes = extension.Attributes; items.AnyAttribute = extension.AnyAttribute; particle = extension.Particle; } else if (content is XmlSchemaSimpleContentExtension) { XmlSchemaSimpleContentExtension extension = (XmlSchemaSimpleContentExtension)content; items.Attributes = extension.Attributes; items.AnyAttribute = extension.AnyAttribute; items.baseSimpleType = extension.BaseTypeName; } } else { items.Attributes = ct.Attributes; items.AnyAttribute = ct.AnyAttribute; particle = ct.Particle; } if (particle is XmlSchemaGroupRef) { XmlSchemaGroupRef refGroup = (XmlSchemaGroupRef)particle; items.Particle = FindGroup(refGroup.RefName).Particle; items.IsUnbounded = particle.IsMultipleOccurrence; } else if (particle is XmlSchemaGroupBase) { items.Particle = (XmlSchemaGroupBase)particle; items.IsUnbounded = particle.IsMultipleOccurrence; } } return items; } void ImportGroup(XmlSchemaGroupBase group, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, INameScope elementsScope, string ns, bool mixed, ref bool needExplicitOrder, bool allowDuplicates, bool groupRepeats, bool allowUnboundedElements) { if (group is XmlSchemaChoice) ImportChoiceGroup((XmlSchemaChoice)group, identifier, members, membersScope, elementsScope, ns, groupRepeats, ref needExplicitOrder, allowDuplicates); else ImportGroupMembers(group, identifier, members, membersScope, elementsScope, ns, groupRepeats, ref mixed, ref needExplicitOrder, allowDuplicates, allowUnboundedElements); if (mixed) { ImportTextMember(members, membersScope, null); } } MemberMapping ImportChoiceGroup(XmlSchemaGroupBase group, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, INameScope elementsScope, string ns, bool groupRepeats, ref bool needExplicitOrder, bool allowDuplicates) { NameTable choiceElements = new NameTable(); if (GatherGroupChoices(group, choiceElements, identifier, ns, ref needExplicitOrder, allowDuplicates)) groupRepeats = true; MemberMapping member = new MemberMapping(); member.Elements = (ElementAccessor[])choiceElements.ToArray(typeof(ElementAccessor)); Array.Sort(member.Elements, new ElementComparer()); AddScopeElements(elementsScope, member.Elements, ref needExplicitOrder, allowDuplicates); bool duplicateTypes = false; bool nullableMismatch = false; Hashtable uniqueTypeDescs = new Hashtable(member.Elements.Length); for (int i = 0; i < member.Elements.Length; i++) { ElementAccessor element = member.Elements[i]; string tdFullName = element.Mapping.TypeDesc.FullName; object val = uniqueTypeDescs[tdFullName]; if (val != null) { duplicateTypes = true; ElementAccessor existingElement = (ElementAccessor)val; if (!nullableMismatch && existingElement.IsNullable != element.IsNullable) nullableMismatch = true; } else { uniqueTypeDescs.Add(tdFullName, element); } ArrayMapping arrayMapping = element.Mapping as ArrayMapping; if (arrayMapping != null) { if (IsNeedXmlSerializationAttributes(arrayMapping)) { // we cannot use ArrayMapping in choice if additional custom // serialization attributes are needed to serialize it element.Mapping = arrayMapping.TopLevelMapping; element.Mapping.ReferencedByTopLevelElement = false; element.Mapping.ReferencedByElement = true; } } } if (nullableMismatch) member.TypeDesc = Scope.GetTypeDesc(typeof(object)); else { TypeDesc[] typeDescs = new TypeDesc[uniqueTypeDescs.Count]; IEnumerator enumerator = uniqueTypeDescs.Values.GetEnumerator(); for (int i = 0; i < typeDescs.Length; i++) { if (!enumerator.MoveNext()) break; typeDescs[i] = ((ElementAccessor)enumerator.Current).Mapping.TypeDesc; } member.TypeDesc = TypeDesc.FindCommonBaseTypeDesc(typeDescs); if (member.TypeDesc == null) member.TypeDesc = Scope.GetTypeDesc(typeof(object)); } if (groupRepeats) member.TypeDesc = member.TypeDesc.CreateArrayTypeDesc(); if (membersScope != null) { member.Name = membersScope.AddUnique(groupRepeats ? "Items" : "Item", member); if (members != null) { members.Add(member.Name, member); } } if (duplicateTypes) { member.ChoiceIdentifier = new ChoiceIdentifierAccessor(); member.ChoiceIdentifier.MemberName = member.Name + "ElementName"; // we need to create the EnumMapping to store all of the element names member.ChoiceIdentifier.Mapping = ImportEnumeratedChoice(member.Elements, ns, member.Name + "ChoiceType"); member.ChoiceIdentifier.MemberIds = new string[member.Elements.Length]; ConstantMapping[] constants = ((EnumMapping)member.ChoiceIdentifier.Mapping).Constants; for (int i = 0; i < member.Elements.Length; i++) { member.ChoiceIdentifier.MemberIds[i] = constants[i].Name; } MemberMapping choiceIdentifier = new MemberMapping(); choiceIdentifier.Ignore = true; choiceIdentifier.Name = member.ChoiceIdentifier.MemberName; if (groupRepeats) { choiceIdentifier.TypeDesc = member.ChoiceIdentifier.Mapping.TypeDesc.CreateArrayTypeDesc(); } else { choiceIdentifier.TypeDesc = member.ChoiceIdentifier.Mapping.TypeDesc; } // create element accessor for the choiceIdentifier ElementAccessor choiceAccessor = new ElementAccessor(); choiceAccessor.Name = choiceIdentifier.Name; choiceAccessor.Namespace = ns; choiceAccessor.Mapping = member.ChoiceIdentifier.Mapping; choiceIdentifier.Elements = new ElementAccessor[] {choiceAccessor}; if (membersScope != null) { choiceAccessor.Name = choiceIdentifier.Name = member.ChoiceIdentifier.MemberName = membersScope.AddUnique(member.ChoiceIdentifier.MemberName, choiceIdentifier); if (members != null) { members.Add(choiceAccessor.Name, choiceIdentifier); } } } return member; } bool IsNeedXmlSerializationAttributes(ArrayMapping arrayMapping) { if (arrayMapping.Elements.Length != 1) return true; ElementAccessor item = arrayMapping.Elements[0]; TypeMapping itemMapping = item.Mapping; if (item.Name != itemMapping.DefaultElementName) return true; if (item.Form != XmlSchemaForm.None && item.Form != XmlSchemaExporter.elementFormDefault) return true; if (item.Mapping.TypeDesc != null) { if (item.IsNullable != item.Mapping.TypeDesc.IsNullable) return true; if (item.Mapping.TypeDesc.IsAmbiguousDataType) return true; } return false; } bool GatherGroupChoices(XmlSchemaGroup group, NameTable choiceElements, string identifier, string ns, ref bool needExplicitOrder, bool allowDuplicates) { return GatherGroupChoices(group.Particle, choiceElements, identifier, ns, ref needExplicitOrder, allowDuplicates); } bool GatherGroupChoices(XmlSchemaParticle particle, NameTable choiceElements, string identifier, string ns, ref bool needExplicitOrder, bool allowDuplicates) { if (particle is XmlSchemaGroupRef) { XmlSchemaGroupRef refGroup = (XmlSchemaGroupRef)particle; if (!refGroup.RefName.IsEmpty) { AddReference(refGroup.RefName, GroupsInUse, Res.XmlCircularGroupReference); if (GatherGroupChoices(FindGroup(refGroup.RefName), choiceElements, identifier, refGroup.RefName.Namespace, ref needExplicitOrder, allowDuplicates)) { RemoveReference(refGroup.RefName, GroupsInUse); return true; } RemoveReference(refGroup.RefName, GroupsInUse); } } else if (particle is XmlSchemaGroupBase) { XmlSchemaGroupBase group = (XmlSchemaGroupBase)particle; bool groupRepeats = group.IsMultipleOccurrence; XmlSchemaAny any = null; bool duplicateElements = false; for (int i = 0; i < group.Items.Count; i++) { object item = group.Items[i]; if (item is XmlSchemaGroupBase || item is XmlSchemaGroupRef) { if (GatherGroupChoices((XmlSchemaParticle)item, choiceElements, identifier, ns, ref needExplicitOrder, allowDuplicates)) groupRepeats = true; } else if (item is XmlSchemaAny) { if (GenerateOrder) { AddScopeElements(choiceElements, ImportAny((XmlSchemaAny)item, true, ns), ref duplicateElements, allowDuplicates); } else { any = (XmlSchemaAny)item; } } else if (item is XmlSchemaElement) { XmlSchemaElement element = (XmlSchemaElement)item; XmlSchemaElement headElement = GetTopLevelElement(element); if (headElement != null) { XmlSchemaElement[] elements = GetEquivalentElements(headElement); for (int j = 0; j < elements.Length; j++) { if (elements[j].IsMultipleOccurrence) groupRepeats = true; AddScopeElement(choiceElements, ImportElement(elements[j], identifier, typeof(TypeMapping), null, elements[j].QualifiedName.Namespace, true), ref duplicateElements, allowDuplicates); } } if (element.IsMultipleOccurrence) groupRepeats = true; AddScopeElement(choiceElements, ImportElement(element, identifier, typeof(TypeMapping), null, element.QualifiedName.Namespace, false), ref duplicateElements, allowDuplicates); } } if (any != null) { AddScopeElements(choiceElements, ImportAny(any, true, ns), ref duplicateElements, allowDuplicates); } if (!groupRepeats && !(group is XmlSchemaChoice) && group.Items.Count > 1) { groupRepeats = true; } return groupRepeats; } return false; } void AddScopeElement(INameScope scope, ElementAccessor element, ref bool duplicateElements, bool allowDuplicates) { if (scope == null) return; ElementAccessor scopeElement = (ElementAccessor)scope[element.Name, element.Namespace]; if (scopeElement != null) { if (!allowDuplicates) { throw new InvalidOperationException(Res.GetString(Res.XmlDuplicateElementInScope, element.Name, element.Namespace)); } if (scopeElement.Mapping.TypeDesc != element.Mapping.TypeDesc) { throw new InvalidOperationException(Res.GetString(Res.XmlDuplicateElementInScope1, element.Name, element.Namespace)); } duplicateElements = true; } else { scope[element.Name, element.Namespace] = element; } } void AddScopeElements(INameScope scope, ElementAccessor[] elements, ref bool duplicateElements, bool allowDuplicates) { for (int i = 0; i < elements.Length; i++) { AddScopeElement(scope, elements[i], ref duplicateElements, allowDuplicates); } } void ImportGroupMembers(XmlSchemaParticle particle, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, INameScope elementsScope, string ns, bool groupRepeats, ref bool mixed, ref bool needExplicitOrder, bool allowDuplicates, bool allowUnboundedElements) { if (particle is XmlSchemaGroupRef) { XmlSchemaGroupRef refGroup = (XmlSchemaGroupRef)particle; if (!refGroup.RefName.IsEmpty) { AddReference(refGroup.RefName, GroupsInUse, Res.XmlCircularGroupReference); ImportGroupMembers(FindGroup(refGroup.RefName).Particle, identifier, members, membersScope, elementsScope, refGroup.RefName.Namespace, groupRepeats | refGroup.IsMultipleOccurrence, ref mixed, ref needExplicitOrder, allowDuplicates, allowUnboundedElements); RemoveReference(refGroup.RefName, GroupsInUse); } } else if (particle is XmlSchemaGroupBase) { XmlSchemaGroupBase group = (XmlSchemaGroupBase)particle; if (group.IsMultipleOccurrence) groupRepeats = true; if (GenerateOrder && groupRepeats && group.Items.Count > 1) { ImportChoiceGroup(group, identifier, members, membersScope, elementsScope, ns, groupRepeats, ref needExplicitOrder, allowDuplicates); } else { for (int i = 0; i < group.Items.Count; i++) { object item = group.Items[i]; if (item is XmlSchemaChoice) ImportChoiceGroup((XmlSchemaGroupBase)item, identifier, members, membersScope, elementsScope, ns, groupRepeats, ref needExplicitOrder, allowDuplicates); else if (item is XmlSchemaElement) ImportElementMember((XmlSchemaElement)item, identifier, members, membersScope, elementsScope, ns, groupRepeats, ref needExplicitOrder, allowDuplicates, allowUnboundedElements); else if (item is XmlSchemaAny) { ImportAnyMember((XmlSchemaAny)item, identifier, members, membersScope, elementsScope, ns, ref mixed, ref needExplicitOrder, allowDuplicates); } else if (item is XmlSchemaParticle) { ImportGroupMembers((XmlSchemaParticle)item, identifier, members, membersScope, elementsScope, ns, groupRepeats, ref mixed, ref needExplicitOrder, allowDuplicates, true); } } } } } XmlSchemaElement GetTopLevelElement(XmlSchemaElement element) { if (!element.RefName.IsEmpty) return FindElement(element.RefName); return null; } XmlSchemaElement[] GetEquivalentElements(XmlSchemaElement element) { ArrayList equivalentElements = new ArrayList(); foreach (XmlSchema schema in Schemas.SchemaSet.Schemas()) { for (int j = 0; j < schema.Items.Count; j++) { object item = schema.Items[j]; if (item is XmlSchemaElement) { XmlSchemaElement equivalentElement = (XmlSchemaElement)item; if (!equivalentElement.IsAbstract && equivalentElement.SubstitutionGroup.Namespace == schema.TargetNamespace && equivalentElement.SubstitutionGroup.Name == element.Name) { equivalentElements.Add(equivalentElement); } } } } return (XmlSchemaElement[])equivalentElements.ToArray(typeof(XmlSchemaElement)); } bool ImportSubstitutionGroupMember(XmlSchemaElement element, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, string ns, bool repeats, ref bool needExplicitOrder, bool allowDuplicates) { XmlSchemaElement[] elements = GetEquivalentElements(element); if (elements.Length == 0) return false; XmlSchemaChoice choice = new XmlSchemaChoice(); for (int i = 0; i < elements.Length; i++) choice.Items.Add(elements[i]); if (!element.IsAbstract) choice.Items.Add(element); if (identifier.Length == 0) identifier = CodeIdentifier.MakeValid(Accessor.UnescapeName(element.Name)); else identifier += CodeIdentifier.MakePascal(Accessor.UnescapeName(element.Name)); ImportChoiceGroup(choice, identifier, members, membersScope, null, ns, repeats, ref needExplicitOrder, allowDuplicates); return true; } void ImportTextMember(CodeIdentifiers members, CodeIdentifiers membersScope, XmlQualifiedName simpleContentType) { TypeMapping mapping; bool isMixed = false; if (simpleContentType != null) { // allow to use all primitive types mapping = ImportType(simpleContentType, typeof(TypeMapping), null, TypeFlags.CanBeElementValue | TypeFlags.CanBeTextValue, false); if (!(mapping is PrimitiveMapping || mapping.TypeDesc.CanBeTextValue)) { return; } } else { // this is a case of the mixed content type, just generate string typeDesc isMixed = true; mapping = GetDefaultMapping(TypeFlags.CanBeElementValue | TypeFlags.CanBeTextValue); } TextAccessor accessor = new TextAccessor(); accessor.Mapping = mapping; MemberMapping member = new MemberMapping(); member.Elements = new ElementAccessor[0]; member.Text = accessor; if (isMixed) { // just generate code for the standard mixed case (string[] text) member.TypeDesc = accessor.Mapping.TypeDesc.CreateArrayTypeDesc(); member.Name = members.MakeRightCase("Text"); } else { // import mapping for the simpleContent PrimitiveMapping pm = (PrimitiveMapping)accessor.Mapping; if (pm.IsList) { member.TypeDesc = accessor.Mapping.TypeDesc.CreateArrayTypeDesc(); member.Name = members.MakeRightCase("Text"); } else { member.TypeDesc = accessor.Mapping.TypeDesc; member.Name = members.MakeRightCase("Value"); } } member.Name = membersScope.AddUnique(member.Name, member); members.Add(member.Name, member); } MemberMapping ImportAnyMember(XmlSchemaAny any, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, INameScope elementsScope, string ns, ref bool mixed, ref bool needExplicitOrder, bool allowDuplicates) { ElementAccessor[] accessors = ImportAny(any, !mixed, ns); AddScopeElements(elementsScope, accessors, ref needExplicitOrder, allowDuplicates); MemberMapping member = new MemberMapping(); member.Elements = accessors; member.Name = membersScope.MakeRightCase("Any"); member.Name = membersScope.AddUnique(member.Name, member); members.Add(member.Name, member); member.TypeDesc = ((TypeMapping)accessors[0].Mapping).TypeDesc; bool repeats = any.IsMultipleOccurrence; if (mixed) { SpecialMapping textMapping = new SpecialMapping(); textMapping.TypeDesc = Scope.GetTypeDesc(typeof(XmlNode)); textMapping.TypeName = textMapping.TypeDesc.Name; member.TypeDesc = textMapping.TypeDesc; TextAccessor text = new TextAccessor(); text.Mapping = textMapping; member.Text = text; repeats = true; mixed = false; } if (repeats) { member.TypeDesc = member.TypeDesc.CreateArrayTypeDesc(); } return member; } ElementAccessor[] ImportAny(XmlSchemaAny any, bool makeElement, string targetNamespace) { SpecialMapping mapping = new SpecialMapping(); mapping.TypeDesc = Scope.GetTypeDesc(makeElement ? typeof(XmlElement) : typeof(XmlNode)); mapping.TypeName = mapping.TypeDesc.Name; TypeFlags flags = TypeFlags.CanBeElementValue; if (makeElement) flags |= TypeFlags.CanBeTextValue; // let the extensions to run RunSchemaExtensions(mapping, XmlQualifiedName.Empty, null, any, flags); if (GenerateOrder && any.Namespace != null) { NamespaceList list = new NamespaceList(any.Namespace, targetNamespace); if (list.Type == NamespaceList.ListType.Set) { ICollection namespaces = list.Enumerate; ElementAccessor[] accessors = new ElementAccessor[namespaces.Count == 0 ? 1 : namespaces.Count]; int count = 0; foreach(string ns in list.Enumerate) { ElementAccessor accessor = new ElementAccessor(); accessor.Mapping = mapping; accessor.Any = true; accessor.Namespace = ns; accessors[count++] = accessor; } if (count > 0) { return accessors; } } } ElementAccessor anyAccessor = new ElementAccessor(); anyAccessor.Mapping = mapping; anyAccessor.Any = true; return new ElementAccessor[] {anyAccessor}; } ElementAccessor ImportArray(XmlSchemaElement element, string identifier, string ns, bool repeats) { if (repeats) return null; if (element.SchemaType == null) return null; if (element.IsMultipleOccurrence) return null; XmlSchemaType type = element.SchemaType; ArrayMapping arrayMapping = ImportArrayMapping(type, identifier, ns, repeats); if (arrayMapping == null) return null; ElementAccessor arrayAccessor = new ElementAccessor(); arrayAccessor.Name = element.Name; arrayAccessor.Namespace = ns; arrayAccessor.Mapping = arrayMapping; if (arrayMapping.TypeDesc.IsNullable) arrayAccessor.IsNullable = element.IsNillable; arrayAccessor.Form = ElementForm(ns, element); return arrayAccessor; } ArrayMapping ImportArrayMapping(XmlSchemaType type, string identifier, string ns, bool repeats) { if (!(type is XmlSchemaComplexType)) return null; if (!type.DerivedFrom.IsEmpty) return null; if (IsMixed(type)) return null; Mapping previousMapping = (Mapping)ImportedMappings[type]; if (previousMapping != null) { if (previousMapping is ArrayMapping) return (ArrayMapping)previousMapping; else return null; } TypeItems items = GetTypeItems(type); if (items.Attributes != null && items.Attributes.Count > 0) return null; if (items.AnyAttribute != null) return null; if (items.Particle == null) return null; XmlSchemaGroupBase item = items.Particle; ArrayMapping arrayMapping = new ArrayMapping(); arrayMapping.TypeName = identifier; arrayMapping.Namespace = ns; if (item is XmlSchemaChoice) { XmlSchemaChoice choice = (XmlSchemaChoice)item; if (!choice.IsMultipleOccurrence) return null; bool needExplicitOrder = false; MemberMapping choiceMember = ImportChoiceGroup(choice, identifier, null, null, null, ns, true, ref needExplicitOrder, false); if (choiceMember.ChoiceIdentifier != null) return null; arrayMapping.TypeDesc = choiceMember.TypeDesc; arrayMapping.Elements = choiceMember.Elements; arrayMapping.TypeName = (type.Name == null || type.Name.Length == 0) ? "ArrayOf" + CodeIdentifier.MakePascal(arrayMapping.TypeDesc.Name) : type.Name; } else if (item is XmlSchemaAll || item is XmlSchemaSequence) { if (item.Items.Count != 1 || !(item.Items[0] is XmlSchemaElement)) return null; XmlSchemaElement itemElement = (XmlSchemaElement)item.Items[0]; if (!itemElement.IsMultipleOccurrence) return null; if (IsCyclicReferencedType(itemElement, new List(1){identifier})) { return null; } ElementAccessor itemAccessor = ImportElement(itemElement, identifier, typeof(TypeMapping), null, ns, false); if (itemAccessor.Any) return null; arrayMapping.Elements = new ElementAccessor[] { itemAccessor }; arrayMapping.TypeDesc = ((TypeMapping)itemAccessor.Mapping).TypeDesc.CreateArrayTypeDesc(); arrayMapping.TypeName = (type.Name == null || type.Name.Length == 0) ? "ArrayOf" + CodeIdentifier.MakePascal(itemAccessor.Mapping.TypeDesc.Name) : type.Name; } else { return null; } ImportedMappings[type] = arrayMapping; Scope.AddTypeMapping(arrayMapping); // for the array-like mappings we need to create a struct mapping for the case when it referenced by the top-level element arrayMapping.TopLevelMapping = ImportStructType(type, ns, identifier, null, true); arrayMapping.TopLevelMapping.ReferencedByTopLevelElement = true; if (type.Name != null && type.Name.Length != 0) ImportDerivedTypes(new XmlQualifiedName(identifier, ns)); return arrayMapping; } bool IsCyclicReferencedType(XmlSchemaElement element, List identifiers) { if (!element.RefName.IsEmpty) { XmlSchemaElement refElement = FindElement(element.RefName); string refElementIdentifier = CodeIdentifier.MakeValid(Accessor.UnescapeName(refElement.Name)); foreach (string identifier in identifiers) { if (refElementIdentifier == identifier) { return true; } } identifiers.Add(refElementIdentifier); XmlSchemaType refType = refElement.SchemaType; if (refType is XmlSchemaComplexType) { TypeItems items = GetTypeItems(refType); if ((items.Particle is XmlSchemaSequence || items.Particle is XmlSchemaAll) && items.Particle.Items.Count == 1 && items.Particle.Items[0] is XmlSchemaElement) { XmlSchemaElement innerRefElement = (XmlSchemaElement)items.Particle.Items[0]; if (innerRefElement.IsMultipleOccurrence) { return IsCyclicReferencedType(innerRefElement, identifiers); } } } } return false; } SpecialMapping ImportAnyMapping(XmlSchemaType type, string identifier, string ns, bool repeats) { if (type == null) return null; if (!type.DerivedFrom.IsEmpty) return null; bool mixed = IsMixed(type); TypeItems items = GetTypeItems(type); if (items.Particle == null) return null; if (!(items.Particle is XmlSchemaAll || items.Particle is XmlSchemaSequence)) return null; if (items.Attributes != null && items.Attributes.Count > 0) return null; XmlSchemaGroupBase group = (XmlSchemaGroupBase) items.Particle; if (group.Items.Count != 1 || !(group.Items[0] is XmlSchemaAny)) return null; XmlSchemaAny any = (XmlSchemaAny)group.Items[0]; SpecialMapping mapping = new SpecialMapping(); // check for special named any case if (items.AnyAttribute != null && any.IsMultipleOccurrence && mixed) { mapping.NamedAny = true; mapping.TypeDesc = Scope.GetTypeDesc(typeof(XmlElement)); } else if (items.AnyAttribute != null || any.IsMultipleOccurrence) { // these only work for named any case -- otherwise import as struct return null; } else { mapping.TypeDesc = Scope.GetTypeDesc(mixed ? typeof(XmlNode) : typeof(XmlElement)); } TypeFlags flags = TypeFlags.CanBeElementValue; if (items.AnyAttribute != null || mixed) flags |= TypeFlags.CanBeTextValue; // let the extensions to run RunSchemaExtensions(mapping, XmlQualifiedName.Empty, null, any, flags); mapping.TypeName = mapping.TypeDesc.Name; if (repeats) mapping.TypeDesc = mapping.TypeDesc.CreateArrayTypeDesc(); return mapping; } void ImportElementMember(XmlSchemaElement element, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, INameScope elementsScope, string ns, bool repeats, ref bool needExplicitOrder, bool allowDuplicates, bool allowUnboundedElements) { repeats = repeats | element.IsMultipleOccurrence; XmlSchemaElement headElement = GetTopLevelElement(element); if (headElement != null && ImportSubstitutionGroupMember(headElement, identifier, members, membersScope, ns, repeats, ref needExplicitOrder, allowDuplicates)) { return; } ElementAccessor accessor; if ((accessor = ImportArray(element, identifier, ns, repeats)) == null) { accessor = ImportElement(element, identifier, typeof(TypeMapping), null, ns, false); } MemberMapping member = new MemberMapping(); string name = CodeIdentifier.MakeValid(Accessor.UnescapeName(accessor.Name)); member.Name = membersScope.AddUnique(name, member); if (member.Name.EndsWith("Specified", StringComparison.Ordinal)) { name = member.Name; member.Name = membersScope.AddUnique(member.Name, member); membersScope.Remove(name); } members.Add(member.Name, member); // we do not support lists for elements if (accessor.Mapping.IsList) { accessor.Mapping = GetDefaultMapping(TypeFlags.CanBeElementValue | TypeFlags.CanBeTextValue); member.TypeDesc = accessor.Mapping.TypeDesc; } else { member.TypeDesc = accessor.Mapping.TypeDesc; } AddScopeElement(elementsScope, accessor, ref needExplicitOrder, allowDuplicates); member.Elements = new ElementAccessor[] { accessor }; if (element.IsMultipleOccurrence || repeats) { if (!allowUnboundedElements && accessor.Mapping is ArrayMapping) { accessor.Mapping = ((ArrayMapping)accessor.Mapping).TopLevelMapping; accessor.Mapping.ReferencedByTopLevelElement = false; accessor.Mapping.ReferencedByElement = true; } member.TypeDesc = accessor.Mapping.TypeDesc.CreateArrayTypeDesc(); } if (element.MinOccurs == 0 && member.TypeDesc.IsValueType && !element.HasDefault && !member.TypeDesc.HasIsEmpty) { member.CheckSpecified = SpecifiedAccessor.ReadWrite; } } void ImportAttributeMember(XmlSchemaAttribute attribute, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, string ns) { AttributeAccessor accessor = ImportAttribute(attribute, identifier, ns, attribute); if (accessor == null) return; MemberMapping member = new MemberMapping(); member.Elements = new ElementAccessor[0]; member.Attribute = accessor; member.Name = CodeIdentifier.MakeValid(Accessor.UnescapeName(accessor.Name)); member.Name = membersScope.AddUnique(member.Name, member); if (member.Name.EndsWith("Specified", StringComparison.Ordinal)) { string name = member.Name; member.Name = membersScope.AddUnique(member.Name, member); membersScope.Remove(name); } members.Add(member.Name, member); member.TypeDesc = accessor.IsList ? accessor.Mapping.TypeDesc.CreateArrayTypeDesc() : accessor.Mapping.TypeDesc; if ((attribute.Use == XmlSchemaUse.Optional || attribute.Use == XmlSchemaUse.None) && member.TypeDesc.IsValueType && !attribute.HasDefault && !member.TypeDesc.HasIsEmpty) { member.CheckSpecified = SpecifiedAccessor.ReadWrite; } } void ImportAnyAttributeMember(XmlSchemaAnyAttribute any, CodeIdentifiers members, CodeIdentifiers membersScope) { SpecialMapping mapping = new SpecialMapping(); mapping.TypeDesc = Scope.GetTypeDesc(typeof(XmlAttribute)); mapping.TypeName = mapping.TypeDesc.Name; AttributeAccessor accessor = new AttributeAccessor(); accessor.Any = true; accessor.Mapping = mapping; MemberMapping member = new MemberMapping(); member.Elements = new ElementAccessor[0]; member.Attribute = accessor; member.Name = membersScope.MakeRightCase("AnyAttr"); member.Name = membersScope.AddUnique(member.Name, member); members.Add(member.Name, member); member.TypeDesc = ((TypeMapping)accessor.Mapping).TypeDesc; member.TypeDesc = member.TypeDesc.CreateArrayTypeDesc(); } bool KeepXmlnsDeclarations(XmlSchemaType type, out string xmlnsMemberName) { xmlnsMemberName = null; if (type.Annotation == null) return false; if (type.Annotation.Items == null || type.Annotation.Items.Count == 0) return false; foreach(XmlSchemaObject o in type.Annotation.Items) { if (o is XmlSchemaAppInfo) { XmlNode[] nodes = ((XmlSchemaAppInfo)o).Markup; if (nodes != null && nodes.Length > 0) { foreach(XmlNode node in nodes) { if (node is XmlElement) { XmlElement e = (XmlElement)node; if (e.Name == "keepNamespaceDeclarations") { if (e.LastNode is XmlText) { xmlnsMemberName = (((XmlText)e.LastNode).Value).Trim(null); } return true; } } } } } } return false; } void ImportXmlnsDeclarationsMember(XmlSchemaType type, CodeIdentifiers members, CodeIdentifiers membersScope) { string xmlnsMemberName; if (!KeepXmlnsDeclarations(type, out xmlnsMemberName)) return; TypeDesc xmlnsTypeDesc = Scope.GetTypeDesc(typeof(XmlSerializerNamespaces)); StructMapping xmlnsMapping = new StructMapping(); xmlnsMapping.TypeDesc = xmlnsTypeDesc; xmlnsMapping.TypeName = xmlnsMapping.TypeDesc.Name; xmlnsMapping.Members = new MemberMapping[0]; xmlnsMapping.IncludeInSchema = false; xmlnsMapping.ReferencedByTopLevelElement = true; ElementAccessor xmlns = new ElementAccessor(); xmlns.Mapping = xmlnsMapping; MemberMapping member = new MemberMapping(); member.Elements = new ElementAccessor[] {xmlns}; member.Name = CodeIdentifier.MakeValid(xmlnsMemberName == null ? "Namespaces" : xmlnsMemberName); member.Name = membersScope.AddUnique(member.Name, member); members.Add(member.Name, member); member.TypeDesc = xmlnsTypeDesc; member.Xmlns = new XmlnsAccessor(); member.Ignore = true; } void ImportAttributeGroupMembers(XmlSchemaAttributeGroup group, string identifier, CodeIdentifiers members, CodeIdentifiers membersScope, string ns) { for (int i = 0; i < group.Attributes.Count; i++) { object item = group.Attributes[i]; if (item is XmlSchemaAttributeGroup) ImportAttributeGroupMembers((XmlSchemaAttributeGroup)item, identifier, members, membersScope, ns); else if (item is XmlSchemaAttribute) ImportAttributeMember((XmlSchemaAttribute)item, identifier, members, membersScope, ns); } if (group.AnyAttribute != null) ImportAnyAttributeMember(group.AnyAttribute, members, membersScope); } AttributeAccessor ImportSpecialAttribute(XmlQualifiedName name, string identifier) { PrimitiveMapping mapping = new PrimitiveMapping(); mapping.TypeDesc = Scope.GetTypeDesc(typeof(string)); mapping.TypeName = mapping.TypeDesc.DataType.Name; AttributeAccessor accessor = new AttributeAccessor(); accessor.Name = name.Name; accessor.Namespace = XmlReservedNs.NsXml; accessor.CheckSpecial(); accessor.Mapping = mapping; return accessor; } AttributeAccessor ImportAttribute(XmlSchemaAttribute attribute, string identifier, string ns, XmlSchemaAttribute defaultValueProvider) { if (attribute.Use == XmlSchemaUse.Prohibited) return null; if (!attribute.RefName.IsEmpty) { if (attribute.RefName.Namespace == XmlReservedNs.NsXml) return ImportSpecialAttribute(attribute.RefName, identifier); else return ImportAttribute(FindAttribute(attribute.RefName), identifier, attribute.RefName.Namespace, defaultValueProvider); } TypeMapping mapping; if (attribute.Name.Length == 0) throw new InvalidOperationException(Res.GetString(Res.XmlAttributeHasNoName)); if (identifier.Length == 0) identifier = CodeIdentifier.MakeValid(attribute.Name); else identifier += CodeIdentifier.MakePascal(attribute.Name); if (!attribute.SchemaTypeName.IsEmpty) mapping = (TypeMapping)ImportType(attribute.SchemaTypeName, typeof(TypeMapping), null, TypeFlags.CanBeAttributeValue, false); else if (attribute.SchemaType != null) mapping = ImportDataType((XmlSchemaSimpleType)attribute.SchemaType, ns, identifier, null, TypeFlags.CanBeAttributeValue, false); else { mapping = GetDefaultMapping(TypeFlags.CanBeAttributeValue); } // let the extensions to run if (mapping != null && !mapping.TypeDesc.IsMappedType) { RunSchemaExtensions(mapping, attribute.SchemaTypeName, attribute.SchemaType, attribute, TypeFlags.CanBeElementValue | TypeFlags.CanBeAttributeValue | TypeFlags.CanBeTextValue); } AttributeAccessor accessor = new AttributeAccessor(); accessor.Name = attribute.Name; accessor.Namespace = ns; accessor.Form = AttributeForm(ns, attribute); accessor.CheckSpecial(); accessor.Mapping = mapping; accessor.IsList = mapping.IsList; accessor.IsOptional = attribute.Use != XmlSchemaUse.Required; if (defaultValueProvider.DefaultValue != null) { accessor.Default = defaultValueProvider.DefaultValue; } else if (defaultValueProvider.FixedValue != null) { accessor.Default = defaultValueProvider.FixedValue; accessor.IsFixed = true; } else if (attribute != defaultValueProvider) { if (attribute.DefaultValue != null) { accessor.Default = attribute.DefaultValue; } else if (attribute.FixedValue != null) { accessor.Default = attribute.FixedValue; accessor.IsFixed = true; } } return accessor; } TypeMapping ImportDataType(XmlSchemaSimpleType dataType, string typeNs, string identifier, Type baseType, TypeFlags flags, bool isList) { if (baseType != null) return ImportStructDataType(dataType, typeNs, identifier, baseType); TypeMapping mapping = ImportNonXsdPrimitiveDataType(dataType, typeNs, flags); if (mapping != null) return mapping; if (dataType.Content is XmlSchemaSimpleTypeRestriction) { XmlSchemaSimpleTypeRestriction restriction = (XmlSchemaSimpleTypeRestriction)dataType.Content; foreach (object o in restriction.Facets) { if (o is XmlSchemaEnumerationFacet) { return ImportEnumeratedDataType(dataType, typeNs, identifier, flags, isList); } } if (restriction.BaseType != null) { return ImportDataType(restriction.BaseType, typeNs, identifier, null, flags, false); } else { AddReference(restriction.BaseTypeName, TypesInUse, Res.XmlCircularTypeReference); mapping = ImportDataType(FindDataType(restriction.BaseTypeName, flags), restriction.BaseTypeName.Namespace, identifier, null, flags, false); if (restriction.BaseTypeName.Namespace != XmlSchema.Namespace) RemoveReference(restriction.BaseTypeName, TypesInUse); return mapping; } } else if (dataType.Content is XmlSchemaSimpleTypeList || dataType.Content is XmlSchemaSimpleTypeUnion) { if (dataType.Content is XmlSchemaSimpleTypeList) { // check if we have enumeration list XmlSchemaSimpleTypeList list = (XmlSchemaSimpleTypeList)dataType.Content; if (list.ItemType != null) { mapping = ImportDataType(list.ItemType, typeNs, identifier, null, flags, true); if (mapping != null) { mapping.TypeName = dataType.Name; return mapping; } } else if (list.ItemTypeName != null && !list.ItemTypeName.IsEmpty) { mapping = ImportType(list.ItemTypeName, typeof(TypeMapping), null, TypeFlags.CanBeAttributeValue, true); if (mapping != null && mapping is PrimitiveMapping) { ((PrimitiveMapping)mapping).IsList = true; return mapping; } } } return GetDefaultMapping(flags); } return ImportPrimitiveDataType(dataType, flags); } TypeMapping ImportEnumeratedDataType(XmlSchemaSimpleType dataType, string typeNs, string identifier, TypeFlags flags, bool isList) { TypeMapping mapping = (TypeMapping)ImportedMappings[dataType]; if (mapping != null) return mapping; XmlSchemaType sourceType = dataType; while (!sourceType.DerivedFrom.IsEmpty) { sourceType = FindType(sourceType.DerivedFrom, TypeFlags.CanBeElementValue | TypeFlags.CanBeAttributeValue); } if (sourceType is XmlSchemaComplexType) return null; TypeDesc sourceTypeDesc = Scope.GetTypeDesc((XmlSchemaSimpleType)sourceType); if (sourceTypeDesc != null && sourceTypeDesc.FullName != typeof(string).FullName) return ImportPrimitiveDataType(dataType, flags); identifier = Accessor.UnescapeName(identifier); string typeName = GenerateUniqueTypeName(identifier); EnumMapping enumMapping = new EnumMapping(); enumMapping.IsReference = Schemas.IsReference(dataType); enumMapping.TypeDesc = new TypeDesc(typeName, typeName, TypeKind.Enum, null, 0); if (dataType.Name != null && dataType.Name.Length > 0) enumMapping.TypeName = identifier; enumMapping.Namespace = typeNs; enumMapping.IsFlags = isList; CodeIdentifiers constants = new CodeIdentifiers(); XmlSchemaSimpleTypeContent content = dataType.Content; if (content is XmlSchemaSimpleTypeRestriction) { XmlSchemaSimpleTypeRestriction restriction = (XmlSchemaSimpleTypeRestriction)content; for (int i = 0; i < restriction.Facets.Count; i++) { object facet = restriction.Facets[i]; if (!(facet is XmlSchemaEnumerationFacet)) continue; XmlSchemaEnumerationFacet enumeration = (XmlSchemaEnumerationFacet)facet; // validate the enumeration value if (sourceTypeDesc != null && sourceTypeDesc.HasCustomFormatter) { XmlCustomFormatter.ToDefaultValue(enumeration.Value, sourceTypeDesc.FormatterName); } ConstantMapping constant = new ConstantMapping(); string constantName = CodeIdentifier.MakeValid(enumeration.Value); constant.Name = constants.AddUnique(constantName, constant); constant.XmlName = enumeration.Value; constant.Value = i; } } enumMapping.Constants = (ConstantMapping[])constants.ToArray(typeof(ConstantMapping)); if (isList && enumMapping.Constants.Length > 63) { // if we have 64+ flag constants we cannot map the type to long enum, we will use string mapping instead. mapping = GetDefaultMapping(TypeFlags.CanBeElementValue | TypeFlags.CanBeTextValue | TypeFlags.CanBeAttributeValue); ImportedMappings.Add(dataType, mapping); return mapping; } ImportedMappings.Add(dataType, enumMapping); Scope.AddTypeMapping(enumMapping); return enumMapping; } internal class ElementComparer : IComparer { public int Compare(object o1, object o2) { ElementAccessor e1 = (ElementAccessor)o1; ElementAccessor e2 = (ElementAccessor)o2; return String.Compare(e1.ToString(string.Empty), e2.ToString(string.Empty), StringComparison.Ordinal); } } EnumMapping ImportEnumeratedChoice(ElementAccessor[] choice, string typeNs, string typeName) { typeName = GenerateUniqueTypeName(Accessor.UnescapeName(typeName), typeNs); EnumMapping enumMapping = new EnumMapping(); enumMapping.TypeDesc = new TypeDesc(typeName, typeName, TypeKind.Enum, null, 0); enumMapping.TypeName = typeName; enumMapping.Namespace = typeNs; enumMapping.IsFlags = false; enumMapping.IncludeInSchema = false; if (GenerateOrder) { Array.Sort(choice, new ElementComparer()); } CodeIdentifiers constants = new CodeIdentifiers(); for (int i = 0; i < choice.Length; i++) { ElementAccessor element = choice[i]; ConstantMapping constant = new ConstantMapping(); string constantName = CodeIdentifier.MakeValid(element.Name); constant.Name = constants.AddUnique(constantName, constant); constant.XmlName = element.ToString(typeNs); constant.Value = i; } enumMapping.Constants = (ConstantMapping[])constants.ToArray(typeof(ConstantMapping)); Scope.AddTypeMapping(enumMapping); return enumMapping; } PrimitiveMapping ImportPrimitiveDataType(XmlSchemaSimpleType dataType, TypeFlags flags) { TypeDesc sourceTypeDesc = GetDataTypeSource(dataType, flags); PrimitiveMapping mapping = new PrimitiveMapping(); mapping.TypeDesc = sourceTypeDesc; mapping.TypeName = sourceTypeDesc.DataType.Name; mapping.Namespace = mapping.TypeDesc.IsXsdType ? XmlSchema.Namespace : UrtTypes.Namespace; return mapping; } PrimitiveMapping ImportNonXsdPrimitiveDataType(XmlSchemaSimpleType dataType, string ns, TypeFlags flags) { PrimitiveMapping mapping = null; TypeDesc typeDesc = null; if (dataType.Name != null && dataType.Name.Length != 0) { typeDesc = Scope.GetTypeDesc(dataType.Name, ns, flags); if (typeDesc != null) { mapping = new PrimitiveMapping(); mapping.TypeDesc = typeDesc; mapping.TypeName = typeDesc.DataType.Name; mapping.Namespace = mapping.TypeDesc.IsXsdType ? XmlSchema.Namespace : ns; } } return mapping; } XmlSchemaGroup FindGroup(XmlQualifiedName name) { XmlSchemaGroup group = (XmlSchemaGroup)Schemas.Find(name, typeof(XmlSchemaGroup)); if (group == null) throw new InvalidOperationException(Res.GetString(Res.XmlMissingGroup, name.Name)); return group; } XmlSchemaAttributeGroup FindAttributeGroup(XmlQualifiedName name) { XmlSchemaAttributeGroup group = (XmlSchemaAttributeGroup)Schemas.Find(name, typeof(XmlSchemaAttributeGroup)); if (group == null) throw new InvalidOperationException(Res.GetString(Res.XmlMissingAttributeGroup, name.Name)); return group; } internal static XmlQualifiedName BaseTypeName(XmlSchemaSimpleType dataType) { XmlSchemaSimpleTypeContent content = dataType.Content; if (content is XmlSchemaSimpleTypeRestriction) { return ((XmlSchemaSimpleTypeRestriction)content).BaseTypeName; } else if (content is XmlSchemaSimpleTypeList) { XmlSchemaSimpleTypeList list = (XmlSchemaSimpleTypeList)content; if (list.ItemTypeName != null && !list.ItemTypeName.IsEmpty) return list.ItemTypeName; if (list.ItemType != null) { return BaseTypeName(list.ItemType); } } return new XmlQualifiedName("string", XmlSchema.Namespace); } TypeDesc GetDataTypeSource(XmlSchemaSimpleType dataType, TypeFlags flags) { TypeDesc typeDesc = null; if (dataType.Name != null && dataType.Name.Length != 0) { typeDesc = Scope.GetTypeDesc(dataType); if (typeDesc != null) return typeDesc; } XmlQualifiedName qname = BaseTypeName(dataType); AddReference(qname, TypesInUse, Res.XmlCircularTypeReference); typeDesc = GetDataTypeSource(FindDataType(qname, flags), flags); if (qname.Namespace != XmlSchema.Namespace) RemoveReference(qname, TypesInUse); return typeDesc; } XmlSchemaSimpleType FindDataType(XmlQualifiedName name, TypeFlags flags) { if (name == null || name.IsEmpty) { return (XmlSchemaSimpleType)Scope.GetTypeDesc(typeof(string)).DataType; } TypeDesc typeDesc = Scope.GetTypeDesc(name.Name, name.Namespace, flags); if (typeDesc != null && typeDesc.DataType is XmlSchemaSimpleType) return (XmlSchemaSimpleType)typeDesc.DataType; XmlSchemaSimpleType dataType = (XmlSchemaSimpleType)Schemas.Find(name, typeof(XmlSchemaSimpleType)); if (dataType != null) { return dataType; } if (name.Namespace == XmlSchema.Namespace) return (XmlSchemaSimpleType)Scope.GetTypeDesc("string", XmlSchema.Namespace, flags).DataType; else { if (name.Name == Soap.Array && name.Namespace == Soap.Encoding) { throw new InvalidOperationException(Res.GetString(Res.XmlInvalidEncoding, name.ToString())); } else { throw new InvalidOperationException(Res.GetString(Res.XmlMissingDataType, name.ToString())); } } } XmlSchemaType FindType(XmlQualifiedName name, TypeFlags flags) { if (name == null || name.IsEmpty) { return Scope.GetTypeDesc(typeof(string)).DataType; } object type = Schemas.Find(name, typeof(XmlSchemaComplexType)); if (type != null) { return (XmlSchemaComplexType)type; } return FindDataType(name, flags); } XmlSchemaElement FindElement(XmlQualifiedName name) { XmlSchemaElement element = (XmlSchemaElement)Schemas.Find(name, typeof(XmlSchemaElement)); if (element == null) throw new InvalidOperationException(Res.GetString(Res.XmlMissingElement, name.ToString())); return element; } XmlSchemaAttribute FindAttribute(XmlQualifiedName name) { XmlSchemaAttribute attribute = (XmlSchemaAttribute)Schemas.Find(name, typeof(XmlSchemaAttribute)); if (attribute == null) throw new InvalidOperationException(Res.GetString(Res.XmlMissingAttribute, name.Name)); return attribute; } XmlSchemaForm ElementForm(string ns, XmlSchemaElement element) { if (element.Form == XmlSchemaForm.None) { XmlSchemaObject parent = element; while (parent.Parent != null) { parent = parent.Parent; } XmlSchema schema = parent as XmlSchema; if (schema != null) { if (ns == null || ns.Length == 0) { return schema.ElementFormDefault == XmlSchemaForm.None ? XmlSchemaForm.Unqualified : schema.ElementFormDefault; } else { XmlSchemas.Preprocess(schema); return element.QualifiedName.Namespace == null || element.QualifiedName.Namespace.Length == 0 ? XmlSchemaForm.Unqualified : XmlSchemaForm.Qualified; } } return XmlSchemaForm.Qualified; } return element.Form; } internal string FindExtendedAnyElement(XmlSchemaAny any, bool mixed, CodeCompileUnit compileUnit, CodeNamespace mainNamespace, out SchemaImporterExtension extension) { extension = null; foreach (SchemaImporterExtension ex in Extensions) { string typeName = ex.ImportAnyElement(any, mixed, Schemas, this, compileUnit, mainNamespace, Options, CodeProvider); if (typeName != null && typeName.Length > 0) { extension = ex; return typeName; } } return null; } internal string FindExtendedType(string name, string ns, XmlSchemaObject context, CodeCompileUnit compileUnit, CodeNamespace mainNamespace, out SchemaImporterExtension extension ) { extension = null; foreach (SchemaImporterExtension ex in Extensions) { string typeName = ex.ImportSchemaType(name, ns, context, Schemas, this, compileUnit, mainNamespace, Options, CodeProvider); if (typeName != null && typeName.Length > 0) { extension = ex; return typeName; } } return null; } internal string FindExtendedType(XmlSchemaType type, XmlSchemaObject context, CodeCompileUnit compileUnit, CodeNamespace mainNamespace, out SchemaImporterExtension extension) { extension = null; foreach (SchemaImporterExtension ex in Extensions) { string typeName = ex.ImportSchemaType(type, context, Schemas, this, compileUnit, mainNamespace, Options, CodeProvider); if (typeName != null && typeName.Length > 0) { extension = ex; return typeName; } } return null; } XmlSchemaForm AttributeForm(string ns, XmlSchemaAttribute attribute) { if (attribute.Form == XmlSchemaForm.None) { XmlSchemaObject parent = attribute; while (parent.Parent != null) { parent = parent.Parent; } XmlSchema schema = parent as XmlSchema; if (schema != null) { if (ns == null || ns.Length == 0) { return schema.AttributeFormDefault == XmlSchemaForm.None ? XmlSchemaForm.Unqualified : schema.AttributeFormDefault; } else { XmlSchemas.Preprocess(schema); return attribute.QualifiedName.Namespace == null || attribute.QualifiedName.Namespace.Length == 0 ? XmlSchemaForm.Unqualified : XmlSchemaForm.Qualified; } } return XmlSchemaForm.Unqualified; } return attribute.Form; } } }