在了解了菜单和命令之后,我们接下来的几篇文章将以自定义编辑器为主题。在开发程序的时候,我们可以用文本编辑器来编写程序代码,并且实际上我们可以用文本编辑器完成所有的开发工作,但我们通常不这么做,因为在visual studio中有很多可以提高我们效率的编辑器,例如winforms编辑器和asp.net的页面编辑器。
Visual Studio IDE允许我们创建自己的编辑器。但创建一个自定义编辑器要比创建一个Tool window复杂多了。
利用VSPackage向导,可以帮助我们创建一个自定义编辑器,但我不打算利用VSPackage向导。这是因为向导生成的代码太长了:光编辑器就有差不多有五千行的代码,但实际上并不需要这么多代码。另外一个原因是:向导生成的代码是有bug的,经常会报出“The operation could not be completed”的异常。不过,vs2008 sdk的例子(C# Example.EditorWithToolbox) 是比较好的,大家可以从这里入手。
这篇文章主要讲述编辑器的基本结构,并且给出一个我做的例子。
众所周知,Visual Studio里有文本编辑器、表单编辑器等等,它们都是内部的编辑器,因为它们运行在Visual Studio的进程里。通过工具|外部工具(Tools|External Tools ),我们可以添加外部的编辑器,但它们实际上是运行在独立的进程里的,是独立的exe文件。
当然,也有一些外部编辑器看起来像是运行在Visual Studio里面,例如VSTO项目中的Microsoft Word 2007或者Excel 2007。但它们实际上是作为ActiveX控件嵌入在VS的Document Window里的,实现这种编辑器要做的工作太多了。
我们只讨论运行在devenv.exe这个进程中的编辑器。这种编辑器有多种类型:
编辑器的结构符合MVC结构,下图可以帮助我们了解它的主要结构:
和工具窗(Tool Window)一样,自定义编辑器也是从属于VSPackage的。package可以用由vs shell提供的SVsRegisterEditors服务来注册编辑器,实际上注册的是一个实现了IVsEditorFactory的编辑器工厂,这个工厂负责初始化编辑器。
一个编辑器要实现很多功能,例如保存、剪切、复制、粘贴,还要负责显示编辑器的界面,维护编辑器要编辑的文件,等等等等。实际上有两类东西需要编辑器管理:
文档视图(Document View)和文档数据(Document Data)不一定非得用两个类来实现,用一个类就已经足够了。要实现编辑器,还需要实现一些接口,如下表:
接口 | 说明 |
IVsEditorFactory | 这个接口用来创建和初始化编辑器,以及在关闭编辑器时去清理资源。该接口对于实现编辑器是必须的。 |
IOleCommandTarget | Document view和Document data都必须识别由vs或其他package传过来的命令。例如,document view要识别和执行“复制”和“粘贴”命令,document data要识别“加载”和“保存”命令。IOleCommandTarget接口就是负责这个的。 (在第13章的时候,我们曾经提到过Command target,在以后的文章里我们还会多次见到它。) |
IVsWindowPane | 文档视图实现了IVsWindowPane接口之后,就可以像vs ide中的其他窗口一样,可以移动、停靠。 (我们第4篇文章曾经用这个接口创建了一个Tool window) |
IVsPersistDocData | 这个接口用来管理文档数据(例如把数据加载到内存里,或者把它存在某个地方),它大概有10个方法,比较复杂:
|
IPersistFileFormat | 通常情况下,设计器对应的文档被持久化到文件里。一个文档可以有多种不同的文件格式。这个接口就是用来处理文件格式的。 |
编辑器拥有文档数据和一个或多个文档视图。在编辑器还没有被打开的情况下,文档数据只是被存放在文件或数据库(或其他地方)里,但是一旦打开了编辑器,就意味着至少有一个视图正在处理数据,如果编辑器有多个视图的话,还需要在多视图之间同步数据。
例如数据库表的设计器有Grid视图和DDL视图,当我们修改了其中一个视图里的数据之后,数据会同步到另外一个视图。再比如Windows Forms设计器,当我们在设计视图做了些修改之后,会把代码同步到代码视图。
Visual Studio利用Running Document Table(RDT)来管理打开的文档。当一个文档的数据改变之后,它可以判断哪些视图和哪些文件(或其他的持久介质,例如数据库的表、存储过程等等)被修改了。当我们关掉一个文件或者关掉解决方案的时候,RTD就会告诉Visual Studio,从而弹出一个询问我们是否要保存文件的对话框:
在这个对话框里的每一项都代表RDT里的一个文档。文档是作为一个独立的单元来持久化的。例如,假设你打算把你的solution都存放在一个文件里的话,你就只有一个文档;假设你的解决方案存放在两个单独的文件里,那么就有两个文档。
RDT利用所谓的“编辑锁”来协调已经打开的文档。当一个文档视图打开的时候,这个文档就被加到了RDT里,RDT给这个文档加了一个“编辑锁”,如果再打开这个文档的另一个视图,RDT又会给这个文档加一个新的“编辑锁”。所以当打开第二个视图的时候,锁的个数变为2。
当一个文档对应的锁的个数变为0的时候,VS就会提示我们去保存文档。
那么,RDT里到底存放了些什么东西呢?
编辑器是和文件的扩展名挂钩的,当注册编辑器的时候,需要指定优先级。优先级很重要,因为VS可以利用优先级来给某个特定扩展名的文件选用最佳的编辑器。
当VS打开一个文件的时候,它通过该文件的扩展名找到对应的编辑器(不止一个),然后根据优先级来决定该使用哪一个编辑器。
VS怎样才能确保至少有一个编辑器可以打开文件呢?这个难题由Visual Studio内置的编辑器来解决:VS内置了一些编辑器(例如二进制编辑器和XML编辑器),这些编辑器和“.*”文件挂了钩。所以,如果一个文件并没有特定的编辑器的话,就会用这些内置的编辑器打开它们。
说了这么多,终于该看一看怎样做一个自定义编辑器了。我的编辑器叫做BlogItemEditor,界面如下图:
这个编辑器用来编辑简单的博客,有标题、分类和内容。用xml文件来存储(我并没有实现上传到博客引擎的功能)。分类是一个列表,可以有多个分类(在界面上用分号隔开),博客内容存为CDATA元素格式:
<BlogItem xmlns=”...”>
<Title>Sample Blog Item</Title>
<Categories>
<Category>VSX Sample</Category>
<Category>Visual Studio 2008</Category>
<Category>LearnVSXNow!</Category>
</Categories>
<Body>
<![CDATA[
After dealing with menus and commands I take a break to show you some new
topics related to custom editors. As we develop applications we use programming
languages with text editors to define the code to be compiled into our product.
...
Where we are?
...
]]>
</Body>
</BlogItem>
项目结构
我的编辑器用了4个核心的类,如下图:
BlogItemEditorFactory是编辑器工厂。我把文档视图和文档数据分隔到3个类里面。BlogItemEditorPane 负责管理文档视图和文档数据,BlogItemEditorControl 是编辑器的界面,BlogItemEditorData 是数据的载体。
我把创建一个简单的编辑器的代码封装了一下,放到了VsxLibrary里:
类型 | 作用 |
SimpleEditorFactory<TEditorPane> | 编辑器工厂,负责创建编辑器 |
SimpleEditorPane<TFactory, TUIControl> | 负责管理文档视图和数据,TFactory是编辑器工厂,TUIControl是界面。 |
ICommonCommandSupport | 指示界面是否支持“全选”、“复制”、“粘贴”等功能。 |
IXmlPersistable | 定义用于读取和保存XElement的方法。 |
BlogItemEditorFactory
BlogItemEditorFactory 继承自SimpleEditorFactory<>泛型类,由于基类里已经做了创建编辑器的逻辑,所以这个子类就很简单了:
[Guid(GuidList.guidBlogEditorFactoryString)]
public sealed class BlogItemEditorFactory:
SimpleEditorFactory<BlogItemEditorPane>
{
}
它用BlogItemEditorPane 作为编辑器的视图和数据。
BlogItemEditorData
这个类用于处理数据:
public sealed class BlogItemEditorData : IXmlPersistable
{
public BlogItemEditorData(string title, string categories, string body)
{ ... }
public string Title { get; }
public string Categories { get; }
public string Body { get; }
public void SaveTo(string fileName) { ... }
public void ReadFrom(string fileName) { ... }
// --- IXmlPersistable implementation
public void SaveTo(XElement targetElement) { ... }
public void ReadFrom(XElement sourceElement) { ... }
}
Title、Categories和Body属性在构造函数里初始化;SaveTo(XElement) 和 ReadFrom(XElement)是IXmlPersistable接口的方法实现;另外两个带有字符串参数的SaveTo和ReadFrom方法负责保存和读取把BlogItemData。
BlogItemEditorControl
这个用户控件就是我们的编辑器的界面了。它实现了ICommonCommandSupport接口:
public partial class BlogItemEditorControl :
UserControl,
ICommonCommandSupport
{
public BlogItemEditorControl()
{
InitializeComponent();
}
// ...
// --- ICommonCommandSupport implementation
// ...
}
ICommonCommandSupport
这个接口指示界面是否支持“全选”、“复制”、“粘贴”等功能,它的定义如下:
public interface ICommonCommandSupport
{
// --- Support flags
bool SupportsSelectAll { get; }
bool SupportsCopy { get; }
bool SupportsCut { get; }
bool SupportsPaste { get; }
bool SupportsRedo { get; }
bool SupportsUndo { get; }
// --- Command execution methods
void DoSelectAll();
void DoCopy();
void DoCut();
void DoPaste();
void DoRedo();
void DoUndo();
}
以Supports为前缀的属性表示是否支持相应的命令,如果支持的话,就会调用相应的以Do为前缀的方法。
BlogItemEditorPane
我们的编辑器的主要工作是由BlogItemEditorPane 来完成的,不过,它的代码是很简单的:
public sealed class BlogItemEditorPane:
SimpleEditorPane<BlogItemEditorFactory, BlogItemEditorControl>
{
public BlogItemEditorPane() { ... }
protected override string GetFileExtension() { ... }
protected override Guid GetCommandSetGuid() { ... }
protected override void LoadFile(string fileName) { ... }
protected override void SaveFile(string fileName) { ... }
e) { ... }
}
基类SimpleEditorPane接受两个类型参数,第一个是编辑器工厂类,第二个是表示用户界面的用户控件,而我们的子类只需要重写下面几个虚方法:
方法名 | 描述 |
GetFileExtension | 定义我们的编辑器支持的文件扩展名 |
GetCommandSetGuid | 定义我们的编辑器可以处理的CommandSetGuid |
LoadFile | 读取数据 |
SaveFile | 保存数据 |
现在让我们来看一下基类SimpleEditorPane的类声明:(迟些我会仔细说明一下这个类)
public abstract class SimpleEditorPane<TFactory, TUIControl> :
WindowPane,
IOleCommandTarget,
IVsPersistDocData,
IPersistFileFormat
where TFactory: IVsEditorFactory
where TUIControl: Control, ICommonCommandSupport, new()
{
// ...
}
SimpleEditorPane 实现了自定义编辑器需要的所有的关键的接口。继承了WindowPane,我们就实现了IVsWindowPane接口;实现了IOleCommandTarget 接口,就可以处理命令;IVsPersistDocData 和IPersistFileFormat 用于持久化。TFactory是实现了IVsEditorFactory 的类,TUIControl 是实现了ICommonCommandSupport 接口的控件。
本篇文章我们开始创建一个自定义编辑器。首先我们了解了VS编辑器的基本架构。编辑器包含文档数据和文档视图,一个编辑器可以有多个视图,所有的视图都为同一个数据工作。
VS IDE用Running Document Table来管理打开的文档,编辑器是有优先级的。
本篇文章只是BlogItemEditor 示例的开篇,下一篇我们将从编辑器工厂类开始。
源码下载:
http://files.cnblogs.com/default/LearnVSXNow-8528.zip
原文链接:
http://dotneteers.net/blogs/divedeeper/archive/2008/03/12/LearnVSXNowPart15.aspx