您的位置:首页 > 编程语言

基于 JFace Text Framework 构建全功能代码编辑器

2018-03-30 11:13 204 查看
转载自https://www.ibm.com/developerworks/cn/opensource/os-cn-ecljtf4/

Content Assistant

Content Assistant(内容提示)可以帮助程序员快速的完成代码,并且还有代码自动补全的附加功能。这对于一个代码编辑器来说是至关重要的,也是不少人喜欢用 IDE 编写代码的原因之一。但是这个功能背后却不是那么简单的,我们先来了解一下 JTF 中和 Content Assistant 相关的概念,下面是 Eclipse 中 Java 编辑器的内容提示的样子:
图1. Java 编辑器的内容提示

先来介绍一下图 1 中出现的三个概念:
Proposal(提议):Proposal 代表了一个可能的自动完成选项,程序员选择之后,代码会自动填入到编辑器里。
Proposal Popup(提议弹出列表):Proposal Popup 是用来显示自动完成列表的窗口
Additional Info(附加信息):每个提议都可以附带一些帮助信息,叫做 Additional Info,它会显示在弹出列表的旁边,并且当你选择某个 Proposal 的时候自动刷新。
提示:在弹出列表出现后,你可能会发现有些键盘事件被弹出列表处理了,比如你按上下箭头,它会改变当前被选择的 Proposal。这是因为在列表弹出之前,内容提示管理器向文本框添加了一个按键校验事件处理器,截获了这些按键。具体的代码可以参考 ContentAssistant 的内部类 InternalListener。
这三个部分都是可以定制的,只不过有的简单有点麻烦一点。比如我们看到弹出列表的下面有一行提示“Press ‘Alt+/’ to show Template Proposals”,这在标准的弹出列表里面是没有的,JDT 定制了这一部分。

为示例代码添加内容提示支持

我打算为本文的示例代码添加以下的内容提示支持:自动提示已经声明的变量名。比如下面的语句:
清单 1. 示例语句
那么当用户在激活内容提示时,我们将显示出 a 和 b 供它选择,也就是显示之前声明过的变量。所有的声明过的变量可以通过遍历语法树来得到,我们在 TreeHelper 里面有一个 getVariables,它会完成这样的功能,如果你生成的语法树不一样,调整这个方法就可以了。注意输入的时候语法必须是正确的,不然语法解析器识别不出这是一个声明语句,也就得不到变量了。

IContentAssistProcessor

第一步,我们要实现 IContentAssistProcessor 接口,它就是所有 Proposal 的来源。不过这个接口的方法比我们想象的要多一些:
清单 2. IContentAssistProcessor 接口
这些方法牵涉到了一些概念,我们来一一的解释它们:
computeCompletionProposals:这个就是所有 Proposal 的来源了,返回的类型是 ICompletionProposal 数组,ICompletionProposal 代表的就是单个的自动完成选项。
computeContextInformation;Context Information(上下文信息)是个新概念,它在这里表示你选择了某个 Proposal 之后,会有一个提示信息弹出来,那个就叫上下文信息。要注意它和上面提到过的 Additional Info 是不同的东西。
getCompletionProposalAutoActivationCharacters:这个方法引入了一个 Auto Activation(自动激活)的概念,所谓自动激活就是在某种条件下 Proposal Popup 自动弹出。这个“某种条件”指的是一些字符,比如最常用的应该是“.”号。
getContextInformationAutoActivationCharacters:上下文信息也有自动激活的功能
getErrorMessage:如果内容提示无法找到任何 Proposal,它可以返回一个错误信息给用户
getContextInformationValidator:上下文信息是可以进行校验的,如果失败,上下文信息不会被显示
computeCompletionProposals 方法显然是必须实现的,我添加了一个 ExprContentAssistProcessor 类,下面是它的实现方式:
清单 3. ExprContentAssistProcessor 实现了 IContentAssistProcessor 接口
我们遍历语法树得到了所有的变量,你可以看到整个实现代码在 ANTLR 以及一些工具类的帮助下显得非常简洁。注意我们为每个变量创建了一个 CompletionProposal,它的构造函数参数非常多,最后一个就是 Additional Info,我这里只是填了一些无用的信息作为演示之用。其它的参数涉及自动完成需要的所有信息,比如插入的字符串,在哪里插入,图标等等。

配置

又到了将我们的实现和 JTF 连接起来的时间,还是修改 ExprConfiguration, 要覆盖的方法变成了 getContentAssistant:
清单 4. 让 JTF 知道我们的内容提示实现
注意我们第一步实现的只是一个 Processor,还不是真正的内容提示管理器,幸运的是 JTF 为我们提供了 ContentAssistant,我们只要新建一个就可以了。第二行看上去有些不解,稍后我会解释。请注意最后一段,大家可以发现内容提示也是和文本类型绑定到一起的。

快捷键

用过 Java 编辑器的应该知道,内容提示可以用热键进行呼出,这个热键可以在 Eclipse 的设置里找到,以 Eclipse 3.3 为例,我们在设置中找到 General->Keys,然后在 filter 中输入 Content Assist 即可找到。为了能够让快捷键对我们的编辑器也有效果,需要安装一个 Handle 来处理它。这部分内容超出了本文的范围,所以我就不详细解释了。大家可以发现 ExprViewer 中多了一些成员和方法,比如 createHandlers 方法,它们都是为了处理快捷键而准备的。

效果

到这里为止,一个很基本的内容提示就完成了,下图是它的效果:
图 2. 内容提示效果图

Information Control

回过头来看看上一节中我卖的关子:ContentAssistant 设置了一个 IInformationControlCreator。从字面上很好理解,Information Control(信息控件)就是用来显示信息的一个控件,而 IInformationControlCreator 就是创建控件的工厂了。信息控件可以用来显示任何信息,在内容提示的情况下,显示的就是 Additional Info。这个控件可以使用任何形式,那么里面的内容也就根据控件的能力可以有不同的变化。比如,你可以用一个浏览器控件来显示信息,这样的话,你的信息可以用HTML来写。在例子中,我们用的是 JTF 的缺省实现:DefaultInformationControl,它内部使用的是 StyledText 控件。它虽然用的不是浏览器,但是它内部提供了一个信息渲染接口:IInformationPresenter。如果你使用 HTMLTextPresenter,它可以支持你在信息中嵌入 HTML 标签。
由于信息控件是一个通用的部件,它被广泛的用在其它需要显示信息的地方,比如我们以后会提到的Text Hover(文本悬浮帮助)。同时由于JTF使用了一系列的接口来抽象信息控件的功能,因此可以很方便的实现自己的信息控件。

结束语

正如我所说,本文的例子是很基本的,有很多可以提高的地方,这些高级的功能留给有兴趣的读者完成。这里给出一些我能想到的问题以供参考:
内容提示只是显示所有的变量,它不会根据用户已经输入的内容来提示。比如有两个变量 test 和 haha,如果用户输入了“te”再激活内容提示,那么我们应该只提示 test。这个并非难事,我们有 TokenList 来帮助我们得到符号信息。
列出的 Proposal 没有图标,只有文字,这是一个小问题。学习了本文之后,你能立刻想起来要加个图标应该修改哪里吗?
对于 Proposal Popup:我们没有定制,可以尝试像 Java 编辑器那样给它底部加上些提示
对于信息控件,用的是缺省实现。可以尝试使用浏览器,然后使用 HTML 显示帮助信息,看上去效果会更好。
对于 IContentAssistProcessor,我们没有实现其它方法,比如上下文信息,自动激活。
要使内容提示功能达到和 Java 编辑器一样的高度,还是要花一些精力的。我一向提倡先了解基本概念,再深入具体细节。希望本文可以作为大家的起点,最终构造出一个专业的内容提示模块。

声明

本文仅代表作者的个人观点,不代表 IBM 的立场。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签:  Eclipse