/* * (c) Copyright Ascensio System SIA 2010-2017 * * This program is a free software product. You can redistribute it and/or * modify it under the terms of the GNU Affero General Public License (AGPL) * version 3 as published by the Free Software Foundation. In accordance with * Section 7(a) of the GNU AGPL its Section 15 shall be amended to the effect * that Ascensio System SIA expressly excludes the warranty of non-infringement * of any third-party rights. * * This program is distributed WITHOUT ANY WARRANTY; without even the implied * warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. For * details, see the GNU AGPL at: http://www.gnu.org/licenses/agpl-3.0.html * * You can contact Ascensio System SIA at Lubanas st. 125a-25, Riga, Latvia, * EU, LV-1021. * * The interactive user interfaces in modified source and object code versions * of the Program must display Appropriate Legal Notices, as required under * Section 5 of the GNU AGPL version 3. * * Pursuant to Section 7(b) of the License you must retain the original Product * logo when distributing the program. Pursuant to Section 7(e) we decline to * grant you any rights under trademark law for use of our trademarks. * * All the Product's GUI elements, including illustrations and icon sets, as * well as technical writing content are licensed under the terms of the * Creative Commons Attribution-ShareAlike 4.0 International. See the License * terms at http://creativecommons.org/licenses/by-sa/4.0/legalcode * */ using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Xml; using System.Xml.Schema; using System.Xml.Serialization; using System.IO; using System.Collections; using System.CodeDom; using System.CodeDom.Compiler; namespace codegen { public class GenMemberPivot { public string sName; public string sNamespace; public string sType; public Type oSystemType; public bool? bIsAttribute; public string sDefAttribute; public bool? bQualified;//нужно ли при записи в xml писать prefix public int? nArrayRank; public List aArrayTypes; public bool? bIsArrayTypesHidden; public GenMemberPivot(string _sName) { sName = _sName; sNamespace = null; sType = null; oSystemType = null; bIsAttribute = null; sDefAttribute = null; bQualified = null; nArrayRank = null; bIsArrayTypesHidden = null; aArrayTypes = null; } public void merge(GenMemberPivot val) { if (string.IsNullOrEmpty(this.sName)) sName = val.sName; if (string.IsNullOrEmpty(this.sNamespace)) sNamespace = val.sNamespace; if (string.IsNullOrEmpty(this.sType)) sType = val.sType; if (null == this.oSystemType) oSystemType = val.oSystemType; if (!this.bIsAttribute.HasValue) bIsAttribute = val.bIsAttribute; if (string.IsNullOrEmpty(this.sDefAttribute)) sDefAttribute = val.sDefAttribute; if (!this.bQualified.HasValue) bQualified = val.bQualified; if (!this.nArrayRank.HasValue) nArrayRank = val.nArrayRank; if (!this.bIsArrayTypesHidden.HasValue) bIsArrayTypesHidden = val.bIsArrayTypesHidden; } public void completeDefaults() { if (!bIsAttribute.HasValue) bIsAttribute = false; if (!bQualified.HasValue) bQualified = true; } } //[System.Xml.Serialization.XmlElementAttribute("c")] //public CT_CalcCell[] c { //nArrayRank=0;aArrayTypes=[CT_CalcCell];bIsArrayTypesHidden=true //[System.Xml.Serialization.XmlElementAttribute("b", typeof(CT_Boolean))] //[System.Xml.Serialization.XmlElementAttribute("d", typeof(CT_DateTime))] //[System.Xml.Serialization.XmlElementAttribute("e", typeof(CT_Error))] //[System.Xml.Serialization.XmlElementAttribute("m", typeof(CT_Missing))] //[System.Xml.Serialization.XmlElementAttribute("n", typeof(CT_Number))] //[System.Xml.Serialization.XmlElementAttribute("s", typeof(CT_String))] //public object[] Items { //nArrayRank=0;aArrayTypes=[CT_Boolean, CT_DateTime, CT_Error, CT_Missing, CT_Number, CT_String];bIsArrayTypesHidden=true //[System.Xml.Serialization.XmlElementAttribute("consolidation", typeof(CT_Consolidation))] //[System.Xml.Serialization.XmlElementAttribute("extLst", typeof(CT_ExtensionList))] //[System.Xml.Serialization.XmlElementAttribute("worksheetSource", typeof(CT_WorksheetSource))] //public object Item //{ //nArrayRank=null;aArrayTypes=[CT_Consolidation, CT_ExtensionList, CT_WorksheetSource];bIsArrayTypesHidden=true //[System.Xml.Serialization.XmlArrayItemAttribute("author", IsNullable=false)] //public string[] authors { //nArrayRank=0;aArrayTypes=[string];bIsArrayTypesHidden=false //[System.Xml.Serialization.XmlArrayItemAttribute("b", typeof(CT_Boolean), IsNullable=false)] //[System.Xml.Serialization.XmlArrayItemAttribute("d", typeof(CT_DateTime), IsNullable=false)] //[System.Xml.Serialization.XmlArrayItemAttribute("e", typeof(CT_Error), IsNullable=false)] //[System.Xml.Serialization.XmlArrayItemAttribute("m", typeof(CT_Missing), IsNullable=false)] //[System.Xml.Serialization.XmlArrayItemAttribute("n", typeof(CT_Number), IsNullable=false)] //[System.Xml.Serialization.XmlArrayItemAttribute("s", typeof(CT_String), IsNullable=false)] //[System.Xml.Serialization.XmlArrayItemAttribute("x", typeof(CT_Index), IsNullable=false)] //public object[][] r { //nArrayRank=1;aArrayTypes=[CT_Boolean, CT_DateTime, CT_Error, CT_Missing, CT_Number, CT_String, CT_Index];bIsArrayTypesHidden=false public class GenClassPivot { public string sName; public string sNamespace; public bool bIsEnum; public List aMembers = new List(); public string sRootName; public GenClassPivot(string _sName, string _sNamespace) { sName = _sName; sNamespace = _sNamespace; bIsEnum = false; sRootName = ""; } public bool isRoot() { return !string.IsNullOrEmpty(this.sRootName); } } class CodeGenPivot { Dictionary m_mapGeneratedClasses = new Dictionary(); int m_nItemsChoiceTypeCount = 0; public void Start(string sDirIn, string sDirCppXmlOut, string sDirCppBinOut, string sDirJsBinOut, ValidationEventHandler oValidationEventHandler) { string sChartNamespace = "http://purl.oclc.org/ooxml/spreadsheetml/main"; string[] aFiles = Directory.GetFiles(sDirIn); XmlSchemaSet schemaSet = new XmlSchemaSet(); schemaSet.ValidationEventHandler += oValidationEventHandler; for (int i = 0; i < aFiles.Length; i++) { string sFile = aFiles[i]; if (".xsd" == Path.GetExtension(sFile)) schemaSet.Add(null, sFile); } schemaSet.Compile(); XmlSchema chartSchema = null; XmlSchemas schemas = new XmlSchemas(); foreach (XmlSchema schema in schemaSet.Schemas()) { if (schema.TargetNamespace == sChartNamespace) { chartSchema = schema; schemas.Add(schema); } } if (null != chartSchema) { CodeNamespace ns = new CodeNamespace(); XmlCodeExporter exporter = new XmlCodeExporter(ns); CodeGenerationOptions generationOptions = CodeGenerationOptions.GenerateProperties; XmlSchemaImporter importer = new XmlSchemaImporter(schemas, generationOptions, new ImportContext(new CodeIdentifiers(), false)); foreach (XmlSchemaElement element in chartSchema.Elements.Values) { XmlTypeMapping mapping = importer.ImportTypeMapping(element.QualifiedName); exporter.ExportTypeMapping(mapping); } CodeGenerator.ValidateIdentifiers(ns); ////Microsoft.CSharp.CSharpCodeProvider oProvider; //// output the C# code //Microsoft.CSharp.CSharpCodeProvider codeProvider = new Microsoft.CSharp.CSharpCodeProvider(); //using (StringWriter writer = new StringWriter()) //{ // codeProvider.GenerateCodeFromNamespace(ns, writer, new CodeGeneratorOptions()); // string sCode = writer.GetStringBuilder().ToString(); //} List aGenClasses = PreProcess(ns, chartSchema); aGenClasses = FilterClassesPivot(aGenClasses); //(new CodegenCPP()).Process(sDirCppXmlOut, sDirCppBinOut, aGenClasses); //(new CodegenJS()).Process(sDirJsBinOut, aGenClasses); (new CodegenJSXml()).Process(sDirJsBinOut, aGenClasses); } } List FilterClassesPivot(List aInput) { Queue aTemp = new Queue(); List aRes = new List(); string[] aTargetTypes = new string[] { "CT_PivotCacheDefinition", "CT_PivotCacheRecords", "CT_pivotTableDefinition" }; //string[] aTargetTypes = new string[] { "CT_Workbook" }; //string[] aTargetTypes = new string[] { "CT_Comments" }; Dictionary mapTargetSubTypes = new Dictionary(); Dictionary namspaces = new Dictionary(); Dictionary mapAllClasses = new Dictionary(); for (int i = 0; i < aInput.Count; ++i) { GenClassPivot oGenClass = aInput[i]; if (0 == aTargetTypes.Length || -1 != Array.IndexOf(aTargetTypes, oGenClass.sName)) aTemp.Enqueue(oGenClass); mapAllClasses[oGenClass.sName] = oGenClass; } while (aTemp.Count > 0) { GenClassPivot oGenClass = aTemp.Dequeue(); if (!mapTargetSubTypes.ContainsKey(oGenClass.sName)) { mapTargetSubTypes.Add(oGenClass.sName, true); aRes.Add(oGenClass); for (int j = 0; j < oGenClass.aMembers.Count; ++j) { GenMemberPivot oGenMember = oGenClass.aMembers[j]; GenClassPivot oTempClass; if (null != oGenMember.sType && mapAllClasses.TryGetValue(oGenMember.sType, out oTempClass)) { namspaces[oTempClass.sNamespace] = true; aTemp.Enqueue(oTempClass); } if (null != oGenMember.aArrayTypes) { for (int k = 0; k < oGenMember.aArrayTypes.Count; ++k) { GenMemberPivot oGenMemberArrayTypes = oGenMember.aArrayTypes[k]; if (null != oGenMemberArrayTypes.sType && mapAllClasses.TryGetValue(oGenMemberArrayTypes.sType, out oTempClass)) { aTemp.Enqueue(oTempClass); } } } } } } return aRes; } List PreProcess(CodeNamespace code, XmlSchema schema) { List aGenClasses = new List(); for (int i = 0; i < code.Types.Count; ++i) { GenClassPivot oNewClass = PreProcessClass(aGenClasses, code.Types[i]); if (null != oNewClass) aGenClasses.Add(oNewClass); } return aGenClasses; } GenClassPivot PreProcessClass(List aGenClasses, CodeTypeDeclaration type) { GenClassPivot oGenClass = null; //получаем xml namespace string sNamespace = null; bool bIncludeInSchema = true; string sRootName = ""; for (int i = 0; i < type.CustomAttributes.Count; i++) { CodeAttributeDeclaration attribute = type.CustomAttributes[i]; if (attribute.Name == "System.Xml.Serialization.XmlTypeAttribute") { foreach (CodeAttributeArgument argument in attribute.Arguments) { if (argument.Name == "Namespace") sNamespace = ((CodePrimitiveExpression)argument.Value).Value.ToString(); else if (argument.Name == "IncludeInSchema") bIncludeInSchema = Convert.ToBoolean(((CodePrimitiveExpression)argument.Value).Value); //todo argument.Name == "TypeName" } } else if (attribute.Name == "System.Xml.Serialization.XmlRootAttribute") { foreach (CodeAttributeArgument argument in attribute.Arguments) { if (argument.Name == "Namespace") sNamespace = ((CodePrimitiveExpression)argument.Value).Value.ToString(); else if ("" == argument.Name) sRootName = ((CodePrimitiveExpression)argument.Value).Value.ToString(); } } } if (bIncludeInSchema) { oGenClass = new GenClassPivot(type.Name, sNamespace); oGenClass.sRootName = sRootName; int nItemsElementName = 0; if (type.IsEnum) { oGenClass.bIsEnum = true; for (int i = 0; i < type.Members.Count; ++i) { CodeTypeMember member = type.Members[i]; GenMemberPivot oGenMember = new GenMemberPivot(member.Name); for (int j = 0; j < member.CustomAttributes.Count; j++) { CodeAttributeDeclaration attribute = member.CustomAttributes[j]; if (attribute.Name == "System.Xml.Serialization.XmlEnumAttribute") ParseArguments(attribute.Arguments, oGenMember); } oGenClass.aMembers.Add(oGenMember); } } else { for (int i = 0; i < type.Members.Count; ++i) { CodeTypeMember member = type.Members[i]; //CodeMemberField пропускаем CodeMemberProperty codeMemberProperty = member as CodeMemberProperty; if (codeMemberProperty != null) { GenMemberPivot oNewGenMember = PreProcessProperty(aGenClasses, codeMemberProperty, oGenClass, ref nItemsElementName); if (null != oNewGenMember) oGenClass.aMembers.Add(oNewGenMember); } } } } return oGenClass; } GenMemberPivot PreProcessProperty(List aGenClasses, CodeMemberProperty codeMemberProperty, GenClassPivot oGenClass, ref int nItemsElementName) { GenMemberPivot oGenMember = new GenMemberPivot(codeMemberProperty.Name); bool bIgnore = false; InitMemberType(oGenMember, codeMemberProperty.Type.BaseType); if (null != codeMemberProperty.Type.ArrayElementType) { oGenMember.nArrayRank = codeMemberProperty.Type.ArrayElementType.ArrayRank; } bool bXmlElementAttribute = false; List aTempMemebers = new List(); for (int i = 0; i < codeMemberProperty.CustomAttributes.Count; i++) { CodeAttributeDeclaration attribute = codeMemberProperty.CustomAttributes[i]; if (attribute.Name == "System.Xml.Serialization.XmlAttributeAttribute") { oGenMember.bQualified = false; oGenMember.bIsAttribute = true; ParseArguments(attribute.Arguments, oGenMember); //todo могут быть повторы имен атрибутов и child nodes. } else if (attribute.Name == "System.ComponentModel.DefaultValueAttribute") { if (attribute.Arguments.Count > 0) { CodeExpression oCodeExpression = attribute.Arguments[attribute.Arguments.Count - 1].Value; //todo other if (oCodeExpression is CodePrimitiveExpression) oGenMember.sDefAttribute = ((oCodeExpression as CodePrimitiveExpression)).Value.ToString(); else if (oCodeExpression is CodeFieldReferenceExpression) oGenMember.sDefAttribute = ((oCodeExpression as CodeFieldReferenceExpression)).FieldName; } } else if (attribute.Name == "System.Xml.Serialization.XmlIgnoreAttribute") bIgnore = true; else if (attribute.Name == "System.Xml.Serialization.XmlElementAttribute") { bXmlElementAttribute = true; GenMemberPivot oTemp = new GenMemberPivot(null); ParseArguments(attribute.Arguments, oTemp); aTempMemebers.Add(oTemp); } else if (attribute.Name == "System.Xml.Serialization.XmlArrayItemAttribute") { GenMemberPivot oTemp = new GenMemberPivot(null); ParseArguments(attribute.Arguments, oTemp); aTempMemebers.Add(oTemp); } else if (attribute.Name == "System.Xml.Serialization.XmlArrayAttribute") { ParseArguments(attribute.Arguments, oGenMember); } //todo не всегда прописан //else if (attribute.Name == "System.Xml.Serialization.XmlChoiceIdentifierAttribute") //{ // if (attribute.Arguments.Count > 0) // { // CodePrimitiveExpression oPrimitiveExpression = attribute.Arguments[0].Value as CodePrimitiveExpression; // oGenMember.sChoiceIdentifier = oPrimitiveExpression.Value.ToString(); // } //} } if (bIgnore) return null; else { if (aTempMemebers.Count > 0) { if (1 == aTempMemebers.Count) { GenMemberPivot TempMember = aTempMemebers[0]; TempMember.merge(oGenMember); TempMember.nArrayRank = null; } oGenMember.aArrayTypes = aTempMemebers; if (bXmlElementAttribute) oGenMember.bIsArrayTypesHidden = true; else oGenMember.bIsArrayTypesHidden = false; } else { oGenMember.nArrayRank = null; } oGenMember.completeDefaults(); return oGenMember; } } void ParseArguments(CodeAttributeArgumentCollection oArguments, GenMemberPivot oGenMember) { CodePrimitiveExpression oPrimitiveExpression = null; CodeTypeOfExpression oTypeOfExpression = null; string sNamespace = null; bool? bForm = null; foreach (CodeAttributeArgument argument in oArguments) { if ("" == argument.Name) { if (argument.Value is CodePrimitiveExpression) oPrimitiveExpression = argument.Value as CodePrimitiveExpression; else if (argument.Value is CodeTypeOfExpression) oTypeOfExpression = argument.Value as CodeTypeOfExpression; } else if ("Namespace" == argument.Name) sNamespace = ((CodePrimitiveExpression)argument.Value).Value.ToString(); else if ("Form" == argument.Name) { string sValue = ((CodeFieldReferenceExpression)argument.Value).FieldName; if ("Qualified" == sValue) bForm = true; else if ("Unqualified" == sValue) bForm = false; } } if (null != oPrimitiveExpression) oGenMember.sName = oPrimitiveExpression.Value.ToString(); if (null != oTypeOfExpression) InitMemberType(oGenMember, oTypeOfExpression.Type.BaseType); if (null != sNamespace) oGenMember.sNamespace = sNamespace; if (bForm.HasValue) oGenMember.bQualified = bForm.Value; } void InitMemberType(GenMemberPivot oGenMember, string sBaseType) { if (-1 != sBaseType.IndexOf("System.")) { oGenMember.oSystemType = Type.GetType(sBaseType); if (null == oGenMember.oSystemType) oGenMember.oSystemType = typeof(string); } else oGenMember.sType = sBaseType; } } }