title author date CreateTime categories
VisualStudio 开发文件自定义工具单文件生成工具
lindexi
2019-04-26 10:49:32 +0800
2019-4-26 9:8:18 +0800
VisualStudio

在 VisualStudio 右击任意的文件夹,点击属性,可以看到属性里面有一个选项是自定义工具。如果创建的是 WPF 项目,可以看到页面或用户控件的自定义工具是 XamlIntelliSenseFileGenerator 可以用来在设计时转换文件,将转换的输出放在另一个文件中。通过这个方法可以自己定义一些内容,在修改内容文件的时候自动生成代码

我最近在做优化,我想将一个自定义的 yml 文件编译为一个类,这样我可以提高软件的运行速度。我想做到和编译 xaml 一样,在 xaml 修改的时候就自动将 yml 编译为一个类,于是我就找到了自定义工具

本文的方法将是在 VisualStudio 2019 使用的方法,在 VisualStudio 2019 的特性 就提供了新的插件 Nuget 包使用方法十分简单

在安装 Visual Studio 2019 的时候,可以选择插件开发,这时可以在新的欢迎界面快速创建插件

现在创建的插件默认给的是 AsyncPackage 而不是之前的 Package 这个类不需要修改内容

创建一个新的类,这个类继承 IVsSingleFileGenerator 接口,这样就可以用来创建文件了。在继承这个接口之后,可以发现需要重写两个方法,第一个方法是 DefaultExtension 方法,这个方法需要返回转换创建的文件的后缀名,同时返回后缀名的字符长度

        public int DefaultExtension(out string defaultExtension)
{
defaultExtension = ".generated.cs";
return defaultExtension.Length;
}

在自定义工具可以自动创建一个和设置自定义工具的文件的相同命名的文件,只是这个文件的后缀名修改为这个方法返回的后缀名。因为这个方法可能是会被一些不清真的代码使用,所以需要返回当前的字符串的长度

第二个方法是 Generate 这是一个核心的方法,在设置使用自定义工具的文件被更改的时候,将会自动调用这个方法。在这个方法的输出里面相对有一些坑,我想要输出林德熙是逗比这个字符串到文件,我需要使用下面代码

        public int Generate(string wszInputFilePath, string bstrInputFileContents, string wszDefaultNamespace, IntPtr[] rgbOutputFileContents, out uint pcbOutput, IVsGeneratorProgress pGenerateProgress)
{
var bytes = Encoding.UTF8.GetBytes("林德熙是逗比");
var length = bytes.Length; rgbOutputFileContents[0] = Marshal.AllocCoTaskMem(length);
Marshal.Copy(bytes, 0, rgbOutputFileContents[0], length);
pcbOutput = (uint) length; return VSConstants.S_OK;
}

这个方法的参数含义

  • wszInputFilePath 文件的绝对路径
  • bstrInputFileContents 文件的内容,可能出现编码问题
  • wszDefaultNamespace 建议的命名空间
  • rgbOutputFileContents 输出的内容
  • pcbOutput 输出的长度

需要将输出的字符串先转 byte 数组,然后赋值给输出的内容返回输出内容的长度。还需要在赋值之前先申请一段内存

当然为了让 VisualStudio 能发现这个类,需要添加一些特性

    [ComVisible(true)]
[Guid(FilePluralizationGeneratorId)]
[CodeGeneratorRegistrationAttribute(typeof(FilePluralizationGenerator), nameof(FilePluralizationGenerator), VSConstants.UICONTEXT.CSharpProject_string, GeneratesDesignTimeSource = true)]
[ProvideObject(typeof(FilePluralizationGenerator))]
public sealed class FilePluralizationGenerator : IVsSingleFileGenerator

这里的 ProvideObject 是做注入使用的,添加这个特性将会注入到 VisualStudio 插件,但是如果有看过之前的博客,就会发现默认需要自己修改注册表的方法才能做到使用自定义工具,现在可以创建 CodeGeneratorRegistrationAttribute 这个类,继承 RegistrationAttribute 做到修改注册表,请看代码

    [AttributeUsage(AttributeTargets.Class, AllowMultiple = true)]
public sealed class CodeGeneratorRegistrationAttribute : RegistrationAttribute
{
public CodeGeneratorRegistrationAttribute(Type generatorType, string generatorName, string contextGuid)
{ } /// <summary>
/// Called to register this attribute with the given context. The context
/// contains the location where the registration information should be placed.
/// It also contains other information such as the type being registered and path information.
/// </summary>
public override void Register(RegistrationContext context)
{
// 写注册表
} /// <summary>
/// Unregister this file extension.
/// </summary>
/// <param name="context"></param>
public override void Unregister(RegistrationContext context)
{ }
}

此时在 FilePluralizationGenerator 上进行断点,然后运行插件,在调试的 VisualStudio 创建一个项目,添加一个 Foo.txt 文本,右击属性,修改自定义工具

此时就可以看到代码运行,输出的林德熙是逗比将会写入到 Foo.txt 下面的 Foo.generated.cs 文件里面

修改 Foo.txt 文件,点击保存会继续进入函数,在自定义工具可以在文件修改的时候自动调用

所有代码放在github

更多插件开发请看

Developing Custom Tool (aka Single File Generators) for Visual Studio 2012

IVsSingleFileGenerator Interface (Microsoft.VisualStudio.Shell.Interop)

RegistrationAttribute Class (Microsoft.VisualStudio.Shell)

2019-4-26-VisualStudio-开发文件自定义工具单文件生成工具的更多相关文章

  1. vue第七单元(vue的单文件组件形式-单文件组件的加载原理-vue-cli构建的开发环境以及生命周期)

    第七单元(vue的单文件组件形式-单文件组件的加载原理-vue-cli构建的开发环境以及生命周期) #课程目标 掌握安装 vue-cli 命令行工具的方法,掌握使用命令行在本地搭建开发环境,使用命令行 ...

  2. 文档生成工具doxygen+图像生成工具GraphViz

    文档生成工具doxygen+图像生成工具GraphViz 虽然jdk自带的javadoc也很好用,不过使用doxygen+GraphViz 的组合可以生成许多强大的图(类图.协作图.文件包含/被包含图 ...

  3. 开源作品-PHP写的Redis管理工具(单文件绿色版)-SuRedisAdmin_PHP_1_0

    前言:项目开发用到了Redis,但是在调试Redis数据的时候,没有一款通用的可视化管理工具.在网络找了一些,但是感觉功能上都不尽人意,于是决定抽出一点时间,开发一个用起来顺手的Redis管理工具.秉 ...

  4. 开源作品-ThinkPHP在线分析工具(单文件绿色版)-TPLogAnalysis_PHP_1_0

    TPLogAnalysis_PHP_1_0 前言:项目开发基于ThinkPHP框架,但是在调试程序的时候,没有一款日志可视化分析工具.在网络也找不到任何相关的TP日志分析工具.求人不如求己,于是决定抽 ...

  5. 开源作品-PHP写的在线文件管理工具(单文件绿色版)-SuExplorer_PHP_3_0

    前言:项目开发过程中,网站一般部署到远程服务器,所以文件管理就不能和本机操作一样方便.通常文件管理是用ftp下载到本地,修改后再上传,或者远程登录到服务器进行修改.但是这些操作都依赖于复杂的第三方软件 ...

  6. 开源作品-PHP写的JS和CSS文件压缩利器(单文件绿色版)-SuMinify_PHP_1_5

    前言: 网站项目需要引用外部文件以减小加载流量,而且第一次加载外部资源文件后,其他同域名的页面如果引用相同的地址,可以利用浏览器缓存直接读取本地缓存资源文件,而不需要每个页面都下载相同的外部资源文件. ...

  7. Spring MVC - MultipartFile实现文件上传(单文件与多文件上传)

    准备工作: 需要先搭建一个spirngmvc的maven项目 1.加入jar包 <dependency> <groupId>commons-fileupload</gro ...

  8. Struts2之文件上传(单文件/多文件)

    <一>简述: Struts2的文件上传其实也是通过拦截器来实现的,只是该拦截器定义为默认拦截器了,所以不用自己去手工配置,<interceptor name="fileUp ...

  9. TypeScript类型定义文件(*.d.ts)生成工具

    在开发ts时,有时会遇到没有d.ts文件的库,同时在老项目迁移到ts项目时也会遇到一些文件需要自己编写声明文件,但是在需要的声明文件比较多的情况,就需要自动生产声明文件.用过几个库.今天简单记录一下. ...

随机推荐

  1. 在同一个项目中灵活运用application/json 和application/x-www-form-urlencoded 两种传输格式(配合axios,同时配置loading)

    'use strict' import axios from 'axios' // import qs from 'qs' import { Notification} from 'element-u ...

  2. pytest-参数化2

    import pytesttest_user_data=['linda','sai','tom']@pytest.fixture(scope='module')def login(request): ...

  3. 【目录】Asp.NETCore轻松学系列

    随笔分类 - Asp.NETCore轻松学系列 Asp.NETCore轻松学系列阅读指引目录 摘要: 耗时两个多月,坚持写这个入门系列文章,就是想给后来者更好更快的上手体验,这个系列可以说是从入门到进 ...

  4. Spring学习笔记(8)——依赖注入

    spring依赖注入使用构造器注入使用属性setter方法注入使用Field注入(用于注解方式) 注入依赖对象可以采用手工装配或自动装配,在实际应用中建议使用手工装配,因为自动装配会产生未知情况,开发 ...

  5. linux sed删除^M换行符以及^[[转义字符

    1. 删除文档中的蓝色转义字符^M 注意:^M 不能从键盘输入^和M,也不能复制.而是需要按Ctrl+v 然后再按Ctrl+M 按Ctrl+v的时候屏幕不会输出,再按下Ctrl+M的时候即会出现^M  ...

  6. elasticsearch启动常见问题

    原文:https://blog.csdn.net/qq_22211217/article/details/80740873 一.Exception in thread "main" ...

  7. 第一章 笔记本电脑安装Linux系统(Centos7)

    目标:通过[Linux+Docke+Nginx+Jenkins+k8s(Kubernetes)+CICD(自动化)]进行项目部署 内容:根据个人进度实时分章节记录自己所遇到的问题 一.准备工作 1.下 ...

  8. 请问如何实现字符串UTF8->BIG5,BIG5->UTF8。保证送分。-Java/JavaSE

    请问如何实现字符串UTF8-> BIG5,BIG5-> UTF8. ------回答--------- ------其他回答(100分)--------- public String BI ...

  9. Codeforces542E Playing on Graph 思维+DFS+BFS

    解法参考https://www.cnblogs.com/BearChild/p/7683114.html这位大佬的,这位大佬讲得很好了. 这道题还是有一定的思维的. 直接贴代码: #include&l ...

  10. 新手创建多module mvn工程

    1.创建工程 创建一个mvn工程有两种方式,一种是通过命令创建,一种是通过idea去一步一步配置. 1.1 命令模式 mvn archetype:generate -DgroupId={groupId ...