摘自:努力学习的熊 codesmith基础
CodeSmith基础(一)
请大家耐心看完所有的基础文章,前两篇网上发表的比较多,是CodeSmith英文帮助文档的第一篇,我后面写的基础是将其他所有的英文帮助全部翻译出来了,全部为本人手写翻译,希望对大家有所帮助 创建好一个模板后第一步要指明这是一个C#语言的模板。
< %@ CodeTemplate Language ="C#" TargetLanguage ="C#" Description ="Generates a class including a special informational header" % > 第二步,我们要指明模板在生成代码时的属性,即生成代码需要的输入值变量。
< %@ Property Name ="NameSpace" Type ="String" Category ="Context" Description ="The namespace to use for this class" % > 如上边所示,在进行代码生成时,在CodeSmith Explorer中选择模板后生成代码的窗口中,变量的名称为NameSpace,类型是String,类别是Context,当用户选中这个属性时对于属性的描述Description。 我们可以按照C#语言的语法去使用定义的变量,例如: 例如下面这个例子模板使用了上面介绍的知识。Test.cst
/// // File: <%=ClassName%>.cs
< %@ CodeTemplate Language ="C#" TargetLanguage ="C#" Description ="Generates a class including a special informational header" % > < %@ Property Name ="NameSpace" Type ="String" Category ="Context" Description ="The namespace to use for this class" % > < %@ Property Name ="ClassName" Type ="String" Category ="Context" Description ="The name of the class to generate" % > < %@ Property Name ="DevelopersName" Type ="String" Category ="Context" Description ="The name to include in the comment header" % > /// // File: <%=ClassName%>.cs // Description: Enter summary here after generation. // --------------------- // Copyright © <%= DateTime.Now.Year %> Our Client // --------------------- // History // <%= DateTime.Now.ToShortDateString() %> <%= DevelopersName%> Original Version /// using System; namespace <%= NameSpace %> { /// <summary> /// Summary description for <%=ClassName %> . /// </summary> public class <%= ClassName %> { public <%= ClassName %> () { // // TODO: Add constructor logic here // } }} 然后我们在CodeSmith Explorer中双击这个模板就会看到相应的属性界面,这里的属性均是我们在前边定义的属性。
按下Generate按钮生成,即可实现一个简单的类代码的生成。
1 /**/ /// 2 // File: MyClass.cs 3 // Description: Enter summary here after generation. 4 // --------------------- 5 // Copyright © 2003 Our Client 6 // --------------------- 7 // History 8 // 12/2/2003 Mr. Smith Original Version 9 /**/ /// 10 11 using System; 12 13 namespace MyNameSpace 14 { 15 /**//// <summary>16 /// Summary description for MyClass.17 /// </summary>18 public class MyClass19 { 20 public MyClass()21 { 22 //23 // TODO: Add constructor logic here24 //25 }26 }27} 生成后的代码即可放入Visual Studio .NET中使用,我们使用CodeSmith的目的就是为了快速高效的开发。
CodeSmith基础(二)
本文将介绍CodeSmith与数据库进行交互生成相应的存储过程,本例使用的数据库为SQL Server 2000。 在与数据库进行交互时,我们使用到了一个CodeSmith自带的组件SchemaExplorer,利用这个组件我们可以访问数据库的数据表、存储过程、视图等,并可以得到相应的数据类型、标识列、列的(字段)名称等信息。 下面这个例子是教我们如何生成一个存储过程,虽然网上有很多一样的例子,但是我是从CodeSmith中的英文帮助中自己翻译出来的:) 使用的是SQL Server 2000自带的Northwind数据库,生成一个关于Orders订单表的更新存储过程。 第一步还是指明模板使用的语言和生成的目标语言。
CodeSmith基础(三)
这里写的东东都是从CodeSmith自带的帮助文档中FAQ里学到的东东 1.如何在模板中添加注释 CodeSmith: <%-- Comments --%> VB.NET: <%-- 'Comments --%> C#: <%-- // Comments --%> <%-- /* Comments */ --%> 2.创建一个可以下拉选择的属性 首先定义一个枚举类型的变量,然后将属性的类型设置为枚举型 CS_Identity_Example.cst文件源代码
1 <% @ Property Name = " CollectionType " Type = " CollectionTypeEnum " Category = " Collection " Description = " Type of collection " %> 2 3 < script runat = " tempate " > 4 public enum CollectionTypeEnum 5 { 6 Vector, 7 HashTable, 8 SortedList 9 } 10 </ script > 3.解决ASP.NET中标签<%重复问题
先将ASP.NET中使用的这个重复标签写成<%%,避免在生成代码时由于是标签重复引起的编译错误或生成错误。
4.如何声明一个常量
< script runat = " template " > private const string MY_CONST = " example " ; </ script > 5.如何对模板进行调试
如果要调试一个模板,首先要在代码模板里进行声明,然后在你想要进行调试的地方用Debugger.Break()语句设置断点即可。
<% @ CodeTemplate Language = " C# " TargetLanguage = " T-SQL " Description = " Debugging your template " Debug = " true " %> <% Debugger.Break(); %> 6.如何将属性设置成选择一个文件夹的路径
[Editor( typeof (System.Windows.Forms.Design.FolderNameEditor), typeof (System.Drawing.Design.UITypeEditor))] public string OutputDirectory{ get { return _outputDirectory;} set {_outputDirectory = value;}} 7.怎样调用子模板
1 <% 2 foreach (TableSchema table in SourceDatabase.Tables) 3 { 4 OutputSubTemplate(table); 5} 6 %> 7 < script runat = " template " > 8 private CodeTemplate _mySubTemplate; 9 10 [Browsable( false )] 11 public CodeTemplate MySubTemplate 12 { 13 get 14 { 15 if (_mySubTemplate == null) 16 { 17 CodeTemplateCompiler compiler = new CodeTemplateCompiler(this.CodeTemplateInfo.DirectoryName + "MySubTemplate.cst"); 18 compiler.Compile(); 19 if (compiler.Errors.Count == 0) 20 { 21 _mySubTemplate = compiler.CreateInstance(); 22 } 23 else 24 { 25 for (int i = 0; i < compiler.Errors.Count; i++) 26 { 27 Response.WriteLine(compiler.Errors[ i].ToString()); 28 } 29 } 30 } 31 return _mySubTemplate; 32 } 33} 34 35 public void OutputSubTemplate(TableSchema table) 36 { 37 MySubTemplate.SetProperty("SourceTable", table); 38 MySubTemplate.SetProperty("IncludeDrop", false); 39 MySubTemplate.SetProperty("InsertPrefix", "Insert"); 40 MySubTemplate.Render(Response); 41} 42 </ script > FAQ中给出的例子为生成一个数据库中所有表的更新Update存储过程
SubTemplatesExample.cst文件源代码
1 <% @ CodeTemplate Language = " C# " TargetLanguage = " T-SQL " 2 Description = " Generates a update stored procedure. " %> 3 4 <% @ Property Name = " SourceDatabase " Type = " SchemaExplorer.DatabaseSchema " 5 Category = " Context " 6 Description = " Database " %> 7 8 <% @ Assembly Name = " SchemaExplorer " %> 9 10 <% @ Import Namespace = " SchemaExplorer " %> 11 12 <% 13 foreach (TableSchema table in SourceDatabase.Tables) 14 { 15 OutputSubTemplate(table); 16} 17 %> 18 19 < script runat = " template " > 20 private CodeTemplate _mySubTemplate; 21 22 23 24 25 [Browsable( false )] 26 public CodeTemplate MySubTemplate 27 { 28 get 29 { 30 if (_mySubTemplate == null) 31 { 32 CodeTemplateCompiler compiler = new CodeTemplateCompiler(this.CodeTemplateInfo.DirectoryName + "MySubTemplate.cst"); 33 compiler.Compile(); 34 35 if (compiler.Errors.Count == 0) 36 { 37 _mySubTemplate = compiler.CreateInstance(); 38 } 39 else 40 { 41 for (int i = 0; i < compiler.Errors.Count; i++) 42 { 43 Response.WriteLine(compiler.Errors[ i].ToString()); 44 } 45 } 46 } 47 48 return _mySubTemplate; 49 } 50} 51 52 public void OutputSubTemplate(TableSchema table) 53 { 54 MySubTemplate.SetProperty("SourceTable", table); 55 MySubTemplate.SetProperty("IncludeDrop", false); 56 MySubTemplate.SetProperty("InsertPrefix", "Insert"); 57 58 MySubTemplate.Render(Response); 59} 60 </ script > MySubTemplate.cst文件源代码
1 <% @ CodeTemplate Language = " C# " TargetLanguage = " T-SQL " 2 Description = " Generates a update stored procedure. " %> 3 4 <% @ Property Name = " SourceTable " Type = " SchemaExplorer.TableSchema " 5 Category = " Context " 6 Description = " Table that the stored procedures should be based on. " %> 7 8 <% @ Assembly Name = " SchemaExplorer " %> 9 10 <% @ Import Namespace = " SchemaExplorer " %> 11 12 13 < script runat = " template " > 14 public string GetSqlParameterStatement(ColumnSchema column) 15 { 16 string param = "@" + column.Name + " " + column.NativeType;1718 switch (column.DataType)19 { 20 case DbType.Decimal:21 { 22 param += "(" + column.Precision + ", " + column.Scale + ")";23 break;24 }25 default:26 { 27 if (column.Size > 0)28 { 29 param += "(" + column.Size + ")";30 }31 break;32 }33 }3435 return param;36} 37 </ script > 38 39 ----------------------------------------------------------------- 40 -- Date Created: <%= DateTime.Now.ToLongDateString() %> 41 -- Created By: Generated by CodeSmith 42 ----------------------------------------------------------------- 43 44 CREATE PROCEDURE dbo.Update <%= SourceTable.Name %> 45 <% for ( int i = 0 ; i < SourceTable.Columns.Count; i ++ ) { %>46 <%= GetSqlParameterStatement(SourceTable.Columns[i]) %><% if (i < SourceTable.Columns.Count - 1) { %>,<% } %>47 <% } %> 48 AS 49 50 UPDATE [ <%= SourceTable.Name %> ] SET 51 <% for ( int i = 0 ; i < SourceTable.NonPrimaryKeyColumns.Count; i ++ ) { %>52 [<%= SourceTable.NonPrimaryKeyColumns[i].Name %>] = @<%= SourceTable.NonPrimaryKeyColumns[i].Name %><% if (i < SourceTable.NonPrimaryKeyColumns.Count - 1) { %>,<% } %>53 <% } %> 54 WHERE 55 <% for ( int i = 0 ; i < SourceTable.PrimaryKey.MemberColumns.Count; i ++ ) { %>56 <% if (i > 0) { %>AND <% } %>57 [<%= SourceTable.PrimaryKey.MemberColumns[i].Name %>] = @<%= SourceTable.PrimaryKey.MemberColumns[i].Name %>58 <% } %> 8.在加载模板时默认加载的命名空间Namespaces和组件Assemblies
组件:mscorlib, System, System.Xml, System.Data, System.Drawing, Microsoft.VisualBasic, System.Windows.Forms, CodeSmith.Engine
命名空间:System, System.Data, System.Diagnostics, System.ComponentModel, Microsoft.VisualBasic, CodeSmith.Engine
9.使用SchemaExplorer能否确定一个字段(Field)是标识字段(主键,Identity Field)
在字段的扩展属性集合中包含一个叫“CS_IsIdentity”的属性,如果这个属性的值为true,则表名当前字段为一个标识字段
1 Identity Field = <% foreach (ColumnSchema cs in SourceTable.Columns) { 2 if ( (( bool )cs.ExtendedProperties[ " CS_IsIdentity " ].Value) == true ) 3 { 4 Response.Write(cs.Name); 5 } 6 } 7 %>
1 <% @ CodeTemplate Language = " C# " TargetLanguage = " T-SQL " 2 Description = " Identifies the identity field of a table " %> 3 4 <% @ Property Name = " SourceTable " Type = " SchemaExplorer.TableSchema " 5 Category = " Context " 6 Description = " Table to target. " %> 7 8 <% @ Assembly Name = " SchemaExplorer " %> 9 10 <% @ Import Namespace = " SchemaExplorer " %> 11 12 13 14 Identity Field = <% foreach (ColumnSchema cs in SourceTable.Columns) { 15 if ( (( bool )cs.ExtendedProperties[ " CS_IsIdentity " ].Value) == true ) 16 { 17 Response.Write(cs.Name); 18 } 19 } 20 %> 10.如何确定一个字段的默认值(各人认为翻译成如何知道一个字段有默认值并且默认值是什么)
在字段的扩展属性集合中包含一个叫“CS_Default”的属性
1 <% 2 foreach (ColumnSchema cs in SourceTable.Columns) { 3 if (cs.ExtendedProperties[ " CS_Default " ] != null ) 4 { 5 Response.WriteLine(cs.ExtendedProperties[ " CS_Default " ].Value); 6 } 7 } 8 %> 11.如何使用SchemaExplorer得到存储过程的输入输出参数
使用CodeSmith提供的CommandSchema对象,它包含需要的输入输出参数集合
1 Input Parameters: 2 <% foreach (ParameterSchema ps in SourceProcedure.AllInputParameters) 3 { 4 Response.Write(ps.Name); 5 Response.Write( " /n " ); 6 } 7 %> 8 9 10 Output Parameters: 11 <% foreach (ParameterSchema ps in SourceProcedure.AllOutputParameters) 12 { 13 Response.Write(ps.Name); 14 Response.Write( " /n " ); 15 } 16 %> InputOutputParameterExample.cst文件源代码
1 <% @ CodeTemplate Language = " C# " TargetLanguage = " T-SQL " 2 Description = " Generates a update stored procedure. " %> 3 4 <% @ Property Name = " SourceProcedure " Type = " SchemaExplorer.CommandSchema " 5 Category = " Context " 6 Description = " The stored procedure to examine " %> 7 8 <% @ Assembly Name = " SchemaExplorer " %> 9 10 <% @ Import Namespace = " SchemaExplorer " %> 11 12 Input Parameters: 13 <% foreach (ParameterSchema ps in SourceProcedure.AllInputParameters) 14 { 15 Response.Write(ps.Name); 16 Response.Write( " /n " ); 17 } 18 %> 19 20 21 Output Parameters: 22 <% foreach (ParameterSchema ps in SourceProcedure.AllOutputParameters) 23 { 24 Response.Write(ps.Name); 25 Response.Write( " /n " ); 26 } 27 %>
CodeSmith基础(四)
本文是翻译的第四篇,内容为在CodeSmith中使用的语法和标签的参考。CodeSmith模板语法参考 本文的目的是在编写一个CodeSmith模板时遇到的各种类型的变量和对象提供参考。本文的目的不是要介绍CodeSmith,如果您想快速了解CodeSmith请查看我翻译的CodeSmith基础(一)和CodeSmith基础(二)。标签 标签一般出现在模板的头部,被用做设置许多不同的属性。代码模板的声明(CodeTemplate Directive) 这个是模板中唯一必须的声明,包含一些模板特殊的属性,包含模板使用的语言、生成的语言和一些对于模板的描述。 例:
参数的介绍:
Language:在开发编写模板时使用的语言,例如
C#,
VB.NET,
Jscript等。
TargetLanguage:只是对模板代码的一个分类,不会影响生成的代码语言。是模板的一个属性,说明模板要基于那种语言生成相应的代码。例如你可以用
CodeSmith从任何一种语言生成
C#代码。
Description:对于模板的一些说明信息,在
CodeSmith Explorer中选中该模板时会显示这里的信息。
Inherits:所有
CodeSmith模板默认继承自
CodeSmith.Engine.CodeTemplate,这个类提供模板使用的一些基本功能,像
ASP.NET页面的
Page类,这些被继承的类的属性可以被修改,但是这些新的类也必须继承
CodeSmith.Engine.CodeTemplate。
CodeSmith也同样可以找到这个类,当然你要引入一个组件包含这个类。
Src:在某些方面
Src和继承
Inherits比较相似,它们都允许你从其他的类包含一些功能进模板。这两个属性的区别是,
Src可以让类与你的模板被动态编译,而
Inherits仅允许你提供一个已经编译好的类或组件。
Debug:可以确定是否在模板中可以包含调试符号。如果将这个属性设置为
True,则可以使用
System.Diagnostics.Debugger.Break()方法来设置断点。
LinePragmas:设置为
True,模板的错误将被指向到模板的源代码。设置为
False,模板的错误将被指向到编译的源代码。
属性的声明(Property Directive) 属性被用做在模板运行时声明一个使用的参数,例:
< %@ CodeTemplate Language ="C#" TargetLanguage ="C#" Description ="Generates a class." % >
< %@ Property Name ="ClassName" Type ="String" Default ="Class1" Category ="Context" Description ="The name of the class to generate" Optional ="true" % > 属性参数的介绍:
Name:模版使用的参数的名称。 Type:参数类型可以是任何.NET有效的数据类型,例如简单的String类型或者是CodeSmith的SchemaExplorer.DatabaseSchema类型。注意,类型必须是基类库的类型,例如用String或者Int32代替string和int。 Default:设置默认值。 Category:用来说明这个属性在CodeSmith Explorer的属性面板中显示成什么类型,例如下拉选择、直接输入等。 Description:在属性面板中对于这个属性的描述。 Optional:设置这个属性是否是必须的,设置为True表明这个参数值可有可无,设置为False则这个参数必须有值。 Editor:表明在属性面板中输入这个属性的值时使用何种GUI(图形界面编辑器)编辑器。 EditorBase:编辑器使用的基本类型,如果没有被说明,UITypeEditor为默认编辑器。 Serializer:这块我的水平不太会犯疑:)The serializer parameter specifies the IPropertySerializer type to use when serializing the properties values. This is equivalent to using a [PropertySerializerAttribute].
XML属性声明(XmlProperty Directive)例:属性的参数:
Name:名称。
Schema:这个参数用来指定一个
XSD文件,创建一个强类型对象模型。如果这个计划被指定,编译器会尝试分析这个
XSD文件并为这个计划生成一个强类型对象模型,这样可以在模版中使用强类型和智能与
XML协同工作。如果这个计划没有被设定,这个参数将为
XmlDocument类型并且将使用
XML DOM去导航到一个
XML内容并生成代码。
Category:在
CodeSmith属性面板中的类别。
Description:描述。
Optional:这个参数是否是必须的,如果设置为
True,则参数不是必须的,反之
False则为必须的。在设置为
False时,如果用户没有提供参数则
CodeSmith不能继续运行。
注册的声明(Register Directive) 这个属性通常被用作引入另一个模版文件并与当前的模版文件同时被编译。这是一种使用子模版的交互方法。 例:模版一旦被注册,就可以建立一个模版的实例,然后象这样设置它的属性:注册的参数: Name:代表被引入的模版的名称。它可以被用作创建一个模版的实例。 Template:被引入模版文件的相对路径,它可以与当前的模版一起被动态的编译。 MergeProperties:设置成True时,所有被引用的面板的属性将被动态的添加到当前模版中。 ExcludePorperties:当使用MergeProperties时,你可能不需要某些属性被添加到当前模版中。将不需要的属性以逗号分隔放在这里,*号可以被用作通配符使用。组件的声明(Assembly Directive) 用作在模版中引用一个外部部组件,或者包含一个编译好的源文件。例:或 CodeSmith自动加载一些不同的组件:System, System.Diagnostics, System.ComponentModel, Microsoft.VisualBasic, CodeSmith.Engine组件的参数: Name:需要引用组件的名称,组建必须存在于Global Assembly Cache,与CodeSmith在同一路径下或与模版文件在同一路径下。 Src:要包含文件的相对路径。引入的声明(Import Directive) 在模版中引入一个命名空间,这个与VB.NET中的Imports和C#中的using相同。 例:引入的参数: NameSpace:被引入的命名空间的名字。记住同时必须要加载包含这个命名空间的相应组件,除非这个组件是被默认加载的。 < %@ XmlProperty Name ="EntityMap" Schema ="EntityMap.xsd" Optional ="False" Category ="Context" Description ="EntityMap XML file to base the output on." % > XML
< %@ Register Name ="MySubTemplate" Template ="MySubTemplate.cst" MergeProperties ="True" ExcludeProperties ="SomeExcludedPropertyName,SomeProperties*" % >
1 < script runat = " template " > 2 public void OutputSubTemplate() 3 { 4 MySubTemplate mySubTemplate = new MySubTemplate(); 5 6 // set an individual properties value. 7 mySubTemplate.SomeExcludedPropertyName = " SomeValue " ; 8 9 // copy all properties with matching name and type to the sub template instance. 10 this .CopyPropertiesTo(mySubTemplate); 11 12 // render the template to the current templates Response object. 13 mySubTemplate.Render( this .Response); 14 15 // render the template to a file. 16 mySubTemplate.RenderToFile( " C:/SomeFile.txt " ); 17 } 18 </ script >
< %@ Assembly Name ="SchemaExplorer" % >
< %@ Assembly Src ="MySourceFile.cs" % >
< %@ Import Namespace ="SchemaExplorer" % >
CodeSmith基础(五)
篇将介绍CodeSmith的模版中的语法。代码标签 <% %>标签 可以放置任意数量的代码在其中,但并不能直接输出到模版中。<%= %>标签
CodeSmith基础(六)
本文主要介绍CodeSmith对象。
CodeSmith Object CodeSimth中有许多对象可以在编写模板的时候使用,这里将介绍这些对象的一些公用方法和属性以及怎么使用它们。
代码模板对象(CodeTemplate Object)在模板中,“this”(或者“Me”在VB.NET中)在当前模板中代码代码模板对象。
代码模板的方法(CodeTemplate Methods)1.public virtual void GetFileName()可以重载这个方法设置模板输出到文件的名称。否则CodeSmith将基于模板名称和TargetLanguage设置它的文件名。
2.public void CopyPropertiesTo(CodeTemplate target)这个方法可以实现从一个模板中将其所有属性的值拷贝到另一个模板所有对应属性中,并按照相应的属性值类型进行匹配。
3.public object GetProperty(string propertyName)这个方法将返回一个给定名称的属性的值。
4.public void SetProperty(string propertyName, object value)此方法可以根据给定名称的属性设置其值。
5.public string SavePropertiesToXml ()这个方法将现有的属性值保存成一个XML的属性字符串。
6.public void SavePropertiesToXmlFile (string fileName)这个方法将当前属性值保存成一个XML的属性文件。
7.public void RestorePropertiesFromXml(string propertySetXml, string baseDirectory)从保存在XML文件中的属性字符串,将模板的属性值恢复。
8.public void RestorePropertiesFromXmlFile(string fileName)从保存在XML文件中的属性文件,将模板的属性值恢复。
代码模板的属性(CodeTemplate Properties)Response:此属性可以访问当前的TextWriter对象,这个对象是用来输出模板用的。
CodeTemplateInfo:这个属性用来访问当前的CodeTemplateInfo对象,这个对象包含当前模板的一些信息。
Progress:这个属性用来报告当前模板的执行过程。
Response Object这个对象提供直接写输出模板的方法。与ASP.NET的response对象很相似。下面是一个利用Response的Write方法在模板上输出一段文字的例子。
<% Response.Write("This will appear in the template") %>
IndentLevel (Int32)当使用Response对象时输出文本的缩进级别。
Indent() Method将输出缩进一个级别。
Unindent() Method将输出少缩进一个级别。
AddTextWriter(TextWriter writer) Method为Response对象增加一个TextWriter。这样可以使在同一时间用多个TextWriter输出模板。
CodeTemplateInfo Object此对象包含一些当前模板的信息。下面是一些CodeTemplateInfo可用的属性。
DateCreated (DateTime)返回一个date类型值,是模板创建的时间。
DateModified (DateTime)返回模板最后一次被修改的时间。
Description (string)返回模板声明时对模版的描述信息。
DirectoryName (string)返回当前模板文件所在的路径。
FileName (string)返回当前模版文件的文件名称。
FullPath (string)返回当前模板的完整路径,路径名+文件名。
Language (string)返回代码模版声明时使用的语言。
TargetLanguage (string)返回代码模版声明时生成的目标语言。
Progress Object
这个属性用来报告当前模板的执行过程。下面是一些Progress可用的成员。
MaximumValue (Int32)模版progress允许的最大值。
MinimumValue (Int32)模版progress允许的最小值。
Step (Int32)模版每执行一不progress的增长值。
Value (Int32)Progress的当前值。
PerformStep() Method按照指定好的progress的增加值执行一步。(原文:Perform a progress step incrementing the progress value by the amount specified in the Step property.)
Increment(Int32 amount) Method指定progress的增加值。(原文:Increment the progress value by the specified amount.)
OnProgress (ProgressEventHandler) Event这个事件用来报告模版的执行过程。(原文:This event can be used to be notified of template execution progress.)
CodeSmith基础(七)
本文翻译的内容为CodeSmith控制台指南。
很多人仅仅知道CodeSmith像一个图形应用程序,或者可能是一个Visual Studio的附件,但是通过CodeSmith的控制台应用程序还有好多其他的使用方法。控制台应用程序是很有价值的,因为可以通过它去生成脚本,或者其他一些自动工具。这篇文档的目的就是要告诉你怎样使用它的控制台应用程序并且如何去定义变量和参数。
Basic Usage
大多数情况下是用控制台应用程序来创建一个模板,一个属性文件,然后保存输出的文件。这有一个很好的例子介绍将合并模版的处理过程放到一个过程中,就像使用NAnt工具。
首先我们要确定完成一个什么样的模版,为这个模板创建一个什么样的XML属性文件。XML属性文件提供在执行模版是需要的各个属性。生成一个属性文件最简单的方法是在CodeSmith Explorer中打开一个模版,填写属性,点击生成按钮generate,然后再点击Save Property Set XML按钮。这个按钮会在点击完生成按钮后找到,在Save Output和Copy Output按钮旁边。然后系统提示输入保存XML属性文件的文件名,下面看一个ArrayList.cst模版创建的XML属性文件。CodeSmith Explorer会更简便。
1 <? xml version="1.0" encoding="us-ascii" ?> 2 < codeSmith > 3 < propertySet > 4 < property name ="Accessibility" > Public </ property > 5 < property name ="ClassName" > PersonArray </ property > 6 < property name ="ItemType" > Person </ property > 7 < property name ="ItemValueType" > False </ property > 8 < property name ="ItemCustomSearch" > False </ property > 9 < property name ="KeyName" > PersonID </ property > 10 < property name ="KeyType" > int </ property > 11 < property name ="IncludeInterfaces" > True </ property > 12 < property name ="IncludeNamespaces" > False </ property > 13 </ propertySet > 14 </ codeSmith > 就像看到的一样,也可以手动创建这个文件,但是使用
现在我们有了这个XML文件,我们继续看一下如何去执行这个模版并是用控制台工具保存结果。首先我们需要是用/template参数去声明我们要是用的模版,像这样:
C:/Program Files/CodeSmith/v3.0>cs /template:Samples/Collections/ArrayList.cst
在这个例子中我们使用了ArrayList.cst模版,它存储在本地的Samples/Collections文件夹下。下一步我们要去声明我们在最后一步需要创建的XML文件,我们是用/propertyset参数去实现。
C:/Program Files/CodeSmith/v3.0>cs /template:Samples/Collections/ArrayList.cst /propertyset:PersonArray.xml
这个/property参数用来指定我们的XML属性文件。最后一个我们需要用的参数是/output参数,用来指定输出怎样被保存。
C:/Program Files/CodeSmith/v3.0>cs /template:Samples/Collections/ArrayList.cst /propertyset:PersonArray.xml /out:test.cs
使用/out参数指定将结果输出到一个叫test.cs文件中保存。执行这个命令后,模板将开始运行,使用属性文件将结果输出到test.cs文件保存。
这是大多数情况下有效使用控制台。
Merging Output
在各种代码生成中最大的挑战就是将生成的代码和开发人员编写或修改的代码区分开。控制台对这个问题提供了一个有效的独特的解决方案,使用一个指定的参数在当前已存在的代码文件中需要将模板生成的代码添加的地方指定一块区域。
下面是一个简单的代码文件,包含了我们要添加生成代码的区域。我们的目标是将DatabaseSchema/BusinessObject.cst模版生成的代码添加到类文件的GeneratedOrderEntity区域中。和上一个例子一样,使用CodeSmith console控制台应用程序执行这个模版,但是这次要使用另一个参数merge。
1 using System; 2 3 namespace Entities 4 { 5 GeneratedOrderEntity#region GeneratedOrderEntity6 78 #endregion 9}
C:/Program Files/CodeSmith/v3.0>cs /template:Samples/DatabaseSchema/BusinessObject.cst /propertyset:OrderEntity.xml /out:OrderEntity.cs /merge:InsertRegion="RegionName=Sample Generated Region;Language=C#;"
使用merge参数我们可以指定区域的名称,在这个例子中是GeneratedOrderEntity,然后控制台应用程序将执行模版,并将结果添加到这个区域中。我们来看一下执行完这个指令后生成的代码。就像看到的一样,Order类被添加到了我们指定的区域中。在代码文件中使用merge参数生成的内容在其他部分被修改或手写后很容易重新再次生成而不会产生影响。
1 using System; 2 3 namespace Infozerk.AuthServices.UnitTestSuite 4 { 5 GeneratedOrderEntity#region GeneratedOrderEntity 6 7 8 Order#region Order 9 /**//// <summary>10 /// This object represents the properties and methods of a Order.11 /// </summary>12 public class Order13 { 14 protected int _id;15 protected string _customerID = String.Empty;16 protected int _employeeID;17 protected DateTime _orderDate;18 protected DateTime _requiredDate;19 protected DateTime _shippedDate;20 protected int _shipVia;2122--为了简短省略了类的其他部分
参数介绍Parameter Reference
Specifying Output
/out:<file>
指定从模版创建的输出文件的名称。
/out:default
指定这个文件被默认保存成模版是用的名称。
/merge:<mergetype>=<init>
指定模版输出的区域。可以简写为/m
Specifying Input
/template:<file>
选择要执行的模版,简写为/t
/propertyset:<file>
生成代码时需要使用的XML属性文件。简写为/p
Compiler Options
/debug[+|-]
指定模版需要包含的调试信息。(允许在运行模版时进行调试)
/tempfiles[+|-]
指定保留临时文件。(如果在临时文件上调试也可以)
Miscellaneous
/help
显示帮助信息。
/nologo
禁止生成器版权信息。
CodeSmith基础(八)
编写CodeSmith自定义属性的编辑器(Writing Custom Property Editors)
当你开始编写自定义的CodeSmith模板时,很可能对于使用它的strings或integers属性很满意,但有时你会发现需要创建一个不同类型的属性,可能是一个自定义的类型或者是.NET framework中但是在属性面板中没有提供的类型。在模板中去作这些很简单,但是怎样指定一个类型在运行模板时显示在属性面板中呢?例如创建了一个Person类并且具有很多不同的属性,但是却没有办法让用户去组装这个类……除非创建一个自定义属性编辑器。
属性面板提供了方法去编写自定义的属性编辑器,当用户在面板上选择一个属性时可以激发相应的方法,同时也可以通过编写代码实现提示用户输入一个必要的值。下面我们举个例子,创建一个接受组建的属性并且是用影射循环贯串所有的类,是所有的类都可以使用它和它的方法,去创建一个NUnit测试基础。(这句翻译的不好,原文:As an example we are going to build a template which accepts an assembly as a property and then using reflection loops through all of the classes, and the methods of those classes, to build NUnit test stubs.)
首先,我们来关注一下模板的组件属性,暂且不看自定义编写的代码。模板的第一部分是一些声明定义和属性。将属性放在脚本标签中代替使用属性声明,在下一部分将看到这样做的必要。
1 <% @ CodeTemplate Language = " C# " TargetLanguage = " C# " Description = " Builds a class for each class in the assembly, and a test stub for every method. " %> 2 3 <% @ Import NameSpace = " System.Reflection " %> 4 5 < script runat = " template " > 6 7 private Assembly assembly; 8 9 public Assembly AssemblyToLoad 10 { 11 get{ return assembly;}12 set{assembly = value;}13} 14 15 </ script >
然后我们为组建assembly中的每一个类创建一个类,为每一个类创建他的方法。然后直接将模板的输出内容放入Visual Studio.NET,然后在编写组件的单元测试时使用向导。
1 using System; 2 using NUnit.Framework; 3 4 <% 5 foreach (Type T in AssemblyToLoad.GetTypes()) 6 { 7 if(T.IsClass) 8 { 9 %>1011 [TestFixture]12 public class <%=T.Name%>Tests13 { 14 <%15 MethodInfo[] methods = T.GetMethods ( BindingFlags.Public | BindingFlags.Instance | BindingFlags.Static );16 foreach(MethodInfo M in methods)17 { 18 %>1920 [Test]21 public void <%=M.Name%>Test22 { 23 //TODO Write this test24 } 25 <%26 }2728 %>}<%29 }30 } 31 %>
首先我们需要创建一个继承UITypeEditor的类。
1 public class AssemblyFilePicker : UITypeEditor 2 { 3 public AssemblyFilePicker(): base()4 { 5 }6}
关于UITypeEditor的说明请大家参看MSDN或Visual Studio.NET自带帮助中的说明,其中有详细的例子。
然后我们需要重载UITypeEditor类的两个不同的方法。第一个需要重载点的方法是GetEditStyle,这个方法是告诉属性面板对于当前类型是用什么类型的编辑器,在这个例子中我们设置编辑类型为Modal。这样大家可以在该属性格子的右边看到一个小按钮,它将引发一个对话框等模式的对话(trigger a modal dialog)。这是我们的GetEditStyle方法:
1 public override UITypeEditorEditStyle GetEditStyle(ITypeDescriptorContext context) 2 { 3 return UITypeEditorEditStyle.Modal;4}
其中的Modal为显示一个省略号按钮。
首先我们要从当前的服务和控件中得到一个参考,有了控件的参考我们可以通过它转到ShowDialog方法。(原文:First we need to get a reference to the current service and control, we need the reference to the control so we can pass it to the ShowDialog method.) 然后我们创建一个openFileDialog类并填入适合的属性。 然后我们通过控件的参考(reference)将对话框显示给用户。 下一步我们检查用户是否点击了OK按钮,如果点击了,通过文件选择对话框选择文件后使用LoadForm方法加载这个组件,最后返回这个值。 这个值将被放在属性面板中并可以被模板读取,但是需要注意,在我们作这个之前要将组件import引入到模板中,并在模板中用一对属性声明。
1 IWindowsFormsEditorService editorService = (IWindowsFormsEditorService)provider.GetService( typeof (IWindowsFormsEditorService)); 2 Control editorControl = editorService as Control; 3 4 if (editorControl != null ) 5 {
1 OpenFileDialog openFileDialog = new OpenFileDialog(); 2 3 openFileDialog.CheckFileExists = true ; 4 openFileDialog.DefaultExt = " .dll " ; 5 openFileDialog.Multiselect = false ; 6 openFileDialog.Title = " Select an Assembly: " ; 7 openFileDialog.Filter = " Assembly Files | *.dll " ;
1 DialogResult result = openFileDialog.ShowDialog(editorControl);
1 if (result == DialogResult.OK) 2 { 3Assembly assembly = Assembly.LoadFrom( openFileDialog.FileName ) ; 4 value = assembly; 5 } 6 else 7 { 8 value = null; 9 } 10 } 11 } 12 13 return value; 14 }
加载这个模板我们仅需将这个组件assembly与模板放在同一目录下,然后再模板中加入下面两行代码。
1 < %@ Assembly Name ="AssemblyHelper" % > 2 < %@ Import NameSpace ="AssemblyHelper" % >
需要重载的另一个方法是EditValue方法,当用户电击属性时会调用这个方法。按照我们需要加载的组件类型需要创建一个打开文件对话框(open file dialog)然后捕获这个对话框,在属性格子中返回对话框的结果。
1 public override object EditValue(ITypeDescriptorContext context, IServiceProvider provider, object value) 2 { 34if (provider != null) 5{
这个模板仅仅可以编译通过,但是由于我们编写显示了一个类型属性面板并不知道如何去操作它,所以我们没有办法自定义指定组件在加载时想要加载的组件。
我们需要创建一个UITypeEditor,这是一个建立属性面板是用的特殊属性的类。UITypeEditor需要创建在一个和模板分离的组件中,我们是用Visual Studio创建这个类。
<% foreach (ColumnSchema column in SourceTable.Columns) { %> <%= column.Name %> <% } %>
在模版中输出一个字符串。上例中的<%=column.Name%>
脚本标签
在这个标签中可以包含一段代码,但是他不直接影响输出的模版。可以放置一些比较有帮助的方法在其中,然后在模版的各个地方可以调用它。在脚本标签中必须包含这个参数runat=”template”,否则他会被处理成普通文本。
例:
1 < script runat = " template " > 2 private string GetColumnName(ColumnSchema cs) 3 { 4 return cs.Name; 5 } 6 </ script > 7 8 <% foreach (ColumnSchema cs in SourceTable.Columns) { %> 9 <%= GetColumnName(cs) %> 10 <% } %>
使用标签可以大量减少代码,并使模版更加的易读和一管理。
Include标签
和ASP.NET一样,可以在模版中包含一些文本文件,但同ASP.NET一样它也不是总能达到你的目标。
例:
<!-- #include file="myfile.inc" -->
有时在多个模版中引用一个组件中的功能,调用其中的方法,这时我们引用组件。但有些情况下,适用Include标签可以得到更好的效果。
Comment标签
注释标签,在前边已经做过介绍。
例:
<% -- This is a comment -- %>
< %@ CodeTemplate Language ="C#" TargetLanguage ="T-SQL" Description ="Generates a update stored procedure." % > 第二步就是我们要加载使用访问数据库的组件
SchemaExplorer,并声明其使用的命名空间。 < %@ Assembly Name ="SchemaExplorer" % > < %@ Import Namespace ="SchemaExplorer" % > 因为是针对表去生成存储过程,则首先要定义一个存储表名称使用的变量,然后指明这个变量类型为数据库中的表,这样我们可以通过这个数据表类型的变量得到相应的表的信息。
< %@ Property Name ="SourceTable" Type ="SchemaExplorer.TableSchema" Category ="Context" Description ="Table that the stored procedures should be based on." % > 如果想访问视图的话,则将变量类型Type中的SchemaExplorer.TableSchema修改为SchemaExplorer.ViewSchema即可。
得到表名的方法
CREATE PROCEDURE dbo. Update <%= SourceTable.Name %> 下面利用循环语句遍历表的各个列,拼出存储过程需要传递的参数。
<% for ( int i = 0 ; i < SourceTable.Columns.Count; i ++ ) { %> <%= GetSqlParameterStatement(SourceTable.Columns[i]) %><% if (i < SourceTable.Columns.Count - 1 ) { %> , <% } %> <% } %> 调用的GetSqlParameterStatement方法是用来生成参数的字符串,例如生成“
@CustomerID nchar(5)”,后边紧跟的if判断是用来生成参数之间相隔使用的逗号的。
生成参数字符串的方法,参数为SchemaExplorer.ColumnSchema列类型
1 < script runat = " template " > 2 public string GetSqlParameterStatement(ColumnSchema column) 3 { 4 string param = " @ " + column.Name + " " + column.NativeType; 5 6 switch (column.DataType) 7 { 8 case DbType.Decimal: 9 { 10 param += " ( " + column.Precision + " , " + column.Scale + " ) " ; 11 break ; 12 } 13 default : 14 { 15 if (column.Size > 0 ) 16 { 17 param += " ( " + column.Size + " ) " ; 18 } 19 break ; 20 } 21 } 22 23 return param; 24 } 25 </ script > 下面来生成需要更新的字段,更新时仅能更新非主键字段的值,在SchemaExplorer中支持这种区别,使用
SourceTable.NonPrimaryKeyColumns即可得到非主键字段的集合。 1 UPDATE [ <%= SourceTable.Name %> ] SET 2 <% for ( int i = 0 ; i < SourceTable.NonPrimaryKeyColumns. Count ; i ++ ) { %> 3 [ <%= SourceTable.NonPrimaryKeyColumns[i ] .Name %> ] = @ <%= SourceTable.NonPrimaryKeyColumns [ i ] .Name %><% if (i < SourceTable.NonPrimaryKeyColumns. Count - 1 ) { %> , <% } %> 4 <% } %> 然后再使用SourceTable.PrimaryKey.MemberColumns得到数据表中的主键集合,生成更新条件
1 WHERE 2 <% for ( int i = 0 ; i < SourceTable.PrimaryKey.MemberColumns. Count ; i ++ ) { %> 3 <% if (i > 0 ) { %>AND <% } %> 4 [ <%= SourceTable.PrimaryKey.MemberColumns[i ] .Name %> ] = @ <%= SourceTable.PrimaryKey.MemberColumns [ i ] .Name %> 5 <% } %> 以下为整体的代码结构
1 < %@ CodeTemplate Language ="C#" TargetLanguage ="T-SQL" 2 Description ="Generates a update stored procedure." % > 3 4 < %@ Property Name ="SourceTable" Type ="SchemaExplorer.TableSchema" 5 Category ="Context" 6 Description ="Table that the stored procedures should be based on." % > 7 8 < %@ Assembly Name ="SchemaExplorer" % > 9 10 < %@ Import Namespace ="SchemaExplorer" % > 11 12 < script runat ="template" > 13 public string GetSqlParameterStatement(ColumnSchema column) 14 { 15 string param = "@" + column.Name + " " + column.NativeType; 16 17 switch (column.DataType) 18 { 19 case DbType.Decimal: 20 { 21 param += "(" + column.Precision + ", " + column.Scale + ")"; 22 break; 23 } 24 default: 25 { 26 if (column.Size > 0) 27 { 28 param += "(" + column.Size + ")"; 29 } 30 break; 31 } 32 } 33 34 return param; 35 } 36 </ script > 37 38 ----------------------------------------------------------------- 39 -- Date Created: < %= DateTime .Now.ToLongDateString() % > 40 -- Created By: Generated by CodeSmith 41 ----------------------------------------------------------------- 42 43 CREATE PROCEDURE dbo.Update < %= SourceTable .Name % > 44 < % for (int i = 0; i < SourceTable.Columns.Count; i++) { % > 45 < %= GetSqlParameterStatement (SourceTable.Columns[i]) % >< % if (i < SourceTable.Columns.Count - 1) { % > , < % } % > 46 < % } % > 47 AS 48 49 UPDATE [ < %= SourceTable .Name % > ] SET 50 < % for (int i = 0; i < SourceTable.NonPrimaryKeyColumns.Count; i++) { % > 51 [ < %= SourceTable .NonPrimaryKeyColumns[i].Name % > ] = @ < %= SourceTable .NonPrimaryKeyColumns[i].Name % >< % if (i < SourceTable.NonPrimaryKeyColumns.Count - 1) { % > , < % } % > 52 < % } % > 53 WHERE 54 < % for (int i = 0; i < SourceTable.PrimaryKey.MemberColumns.Count; i++) { % > 55 < % if (i > 0) { %>AND < % } % > 56 [ < %= SourceTable .PrimaryKey.MemberColumns[i].Name % > ] = @ < %= SourceTable .PrimaryKey.MemberColumns[i].Name % > 57 < % } % > 58
然后我们要在组建属性中添加Editor属性,并且指定为UITypeEditor编辑器。 当属性面板读取到这些属性时,它将使用我们自定义的UITypeEditor编辑器并允许用户从打开文件对话框中选择一个组件。
1 [Editor( typeof (AssemblyHelper.AssemblyFilePicker), typeof (System.Drawing.Design.UITypeEditor))] 2 public Assembly AssemblyToLoad 3 { 4 get{ return assembly;}5 set{assembly = value;}6}