用T4生成多个文件(转)

初次认识并尝试使用T4生成代码的时候,相关学习资料似乎比较少。不过现在VS2010 的MSDN里已有相关章节,可参看《代码生成和文本模板》章节。可以用C#的语法写模板,实在舒服很多。

很快就发现T4难以生成多个文件的缺陷,微软似乎也不着急改进这一点。通过搜索,从InfoQ找到一篇文章《T4生成多个文件》,链接到一篇文章,Damien Guard的扩展可以方便的生成多个文件。原文是英文,能看懂,然而如果翻译则斟酌字词太辛苦。

首先,保存以下代码为一个模板文件(例如保存文件名为Manager.ttinclude):

<#@ assembly="" name="System.Core" #="">
<#@ assembly="" name="System.Data.Linq" #="">
<#@ assembly="" name="EnvDTE" #="">
<#@ assembly="" name="System.Xml" #="">
<#@ assembly="" name="System.Xml.Linq" #="">
<#@ import="" namespace="System" #="">
<#@ import="" namespace="System.CodeDom" #="">
<#@ import="" namespace="System.CodeDom.Compiler" #="">
<#@ import="" namespace="System.Collections.Generic" #="">
<#@ import="" namespace="System.Data.Linq" #="">
<#@ import="" namespace="System.Data.Linq.Mapping" #="">
<#@ import="" namespace="System.IO" #="">
<#@ import="" namespace="System.Linq" #="">
<#@ import="" namespace="System.Reflection" #="">
<#@ import="" namespace="System.Text" #="">
<#@ import="" namespace="System.Xml.Linq" #="">
<#@ import="" namespace="Microsoft.VisualStudio.TextTemplating" #="">
<#+ manager="" class="" records="" the="" various="" blocks="" so="" it="" can="" split="" them="" up="" class="" manager="" {="" private="" class="" block="" {="" public="" string="" name;="" public="" int="" start,="" length;="" }="" private="" block="" currentblock;="" private=""> files = new List();
    private Block footer = new Block();
    private Block header = new Block();
    private ITextTemplatingEngineHost host;
    private StringBuilder template;
    protected List generatedFileNames = new List();

    public static Manager Create(ITextTemplatingEngineHost host, StringBuilder template) {
        return (host is IServiceProvider) ? new VSManager(host, template) : new Manager(host, template);
    }

    public void StartNewFile(String name) {
        if (name == null)
            throw new ArgumentNullException("name");
        CurrentBlock = new Block { Name = name };
    }

    public void StartFooter() {
        CurrentBlock = footer;
    }

    public void StartHeader() {
        CurrentBlock = header;
    }

    public void EndBlock() {
        if (CurrentBlock == null)
            return;
        CurrentBlock.Length = template.Length - CurrentBlock.Start;
        if (CurrentBlock != header && CurrentBlock != footer)
            files.Add(CurrentBlock);
        currentBlock = null;
    }

    public virtual void Process(bool split) {
        if (split) {
            EndBlock();
            String headerText = template.ToString(header.Start, header.Length);
            String footerText = template.ToString(footer.Start, footer.Length);
            String outputPath = Path.GetDirectoryName(host.TemplateFile);
            files.Reverse();
            foreach(Block block in files) {
                String fileName = Path.Combine(outputPath, block.Name);
                String content = headerText + template.ToString(block.Start, block.Length) + footerText;
                generatedFileNames.Add(fileName);
                CreateFile(fileName, content);
                template.Remove(block.Start, block.Length);
            }
        }
    }

    protected virtual void CreateFile(String fileName, String content) {
        if (IsFileContentDifferent(fileName, content))
            File.WriteAllText(fileName, content);
    }

    public virtual String GetCustomToolNamespace(String fileName) {
        return null;
    }

    public virtual String DefaultProjectNamespace {
        get { return null; }
    }

    protected bool IsFileContentDifferent(String fileName, String newContent) {
        return !(File.Exists(fileName) && File.ReadAllText(fileName) == newContent);
    }

    private Manager(ITextTemplatingEngineHost host, StringBuilder template) {
        this.host = host;
        this.template = template;
    }

    private Block CurrentBlock {
        get { return currentBlock; }
        set {
            if (CurrentBlock != null)
                EndBlock();
            if (value != null)
                value.Start = template.Length;
            currentBlock = value;
        }
    }

    private class VSManager: Manager {
        private EnvDTE.ProjectItem templateProjectItem;
        private EnvDTE.DTE dte;
        private Action checkOutAction;
        private Action<>> projectSyncAction;

        public override String DefaultProjectNamespace {
            get {
                return templateProjectItem.ContainingProject.Properties.Item("DefaultNamespace").Value.ToString();
            }
        }

        public override String GetCustomToolNamespace(string fileName) {
            return dte.Solution.FindProjectItem(fileName).Properties.Item("CustomToolNamespace").Value.ToString();
        }

        public override void Process(bool split) {
            if (templateProjectItem.ProjectItems == null)
                return;
            base.Process(split);
            projectSyncAction.EndInvoke(projectSyncAction.BeginInvoke(generatedFileNames, null, null));
        }

        protected override void CreateFile(String fileName, String content) {
            if (IsFileContentDifferent(fileName, content)) {
                CheckoutFileIfRequired(fileName);
                File.WriteAllText(fileName, content);
            }
        }

        internal VSManager(ITextTemplatingEngineHost host, StringBuilder template)
            : base(host, template) {
            var hostServiceProvider = (IServiceProvider) host;
            if (hostServiceProvider == null)
                throw new ArgumentNullException("Could not obtain IServiceProvider");
            dte = (EnvDTE.DTE) hostServiceProvider.GetService(typeof(EnvDTE.DTE));
            if (dte == null)
                throw new ArgumentNullException("Could not obtain DTE from host");
            templateProjectItem = dte.Solution.FindProjectItem(host.TemplateFile);
            checkOutAction = (String fileName) => dte.SourceControl.CheckOutItem(fileName);
            projectSyncAction = (IEnumerable keepFileNames) => ProjectSync(templateProjectItem, keepFileNames);
        }

        private static void ProjectSync(EnvDTE.ProjectItem templateProjectItem, IEnumerable keepFileNames) {
            var keepFileNameSet = new HashSet(keepFileNames);
            var projectFiles = new Dictionary();
            var originalFilePrefix = Path.GetFileNameWithoutExtension(templateProjectItem.get_FileNames(0)) + ".";
            foreach(EnvDTE.ProjectItem projectItem in templateProjectItem.ProjectItems)
                projectFiles.Add(projectItem.get_FileNames(0), projectItem);

            // Remove unused items from the project
            foreach(var pair in projectFiles)
                if (!keepFileNames.Contains(pair.Key) && !(Path.GetFileNameWithoutExtension(pair.Key) + ".").StartsWith(originalFilePrefix))
                    pair.Value.Delete();

            // Add missing files to the project
            foreach(String fileName in keepFileNameSet)
                if (!projectFiles.ContainsKey(fileName))
                    templateProjectItem.ProjectItems.AddFromFile(fileName);
        }

        private void CheckoutFileIfRequired(String fileName) {
            var sc = dte.SourceControl;
            if (sc != null && sc.IsItemUnderSCC(fileName) && !sc.IsItemCheckedOut(fileName))
                checkOutAction.EndInvoke(checkOutAction.BeginInvoke(fileName, null, null));
        }
    }
} #>

接着,在T4模板文件里引用这个模板,并声明一个Manager类实例:

<#@ template="" language="C#" hostspecific="True" #="">
<#@include file="Manager.ttinclude" #="">
<# var="" manager="Manager.Create(Host," generationenvironment);="" #="">

使用两行代码可使代码输出到单独文件,你要输出的代码可写在这两个语句中间,StartNewFile的参数就是输出的文件名:

<# manager.startnewfile("employee.generated.cs");="" #="">
<# manager.endblock();="" #="">

比如可以这样写:

<# manager.startnewfile("employee.generated.cs");="" #="">
public class Employee {  }
<# manager.endblock();="" #="">

还可以为每个输出文件输出同样的头部或顶部,只需要相应的语句:

<# manager.startheader();="" #="">
// Code generated by a template
using System;
<# manager.endblock();="" #="">

<# manager.startfooter();="" #="">
// It's the end
<# manager.endblock();="" #="">

最后使用这句来执行输出多个文件:

<# manager.process(true);="" #="">
原文地址:https://www.cnblogs.com/marslin/p/3777511.html