`
hqs7636
  • 浏览: 222734 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

基于 JFace Text Framework 构建全功能代码编辑器: 第 4 部分

阅读更多
Content Assistant
developerWorks


文档选项
将打印机的版面设置成横向打印模式

打印本页
将此页作为电子邮件发送

将此页作为电子邮件发送


样例代码

级别: 中级

马 若劼 (maruojie@cn.ibm.com), 软件工程师, IBM 中国软件开发中心

2008 年 3 月 27 日

    Content Assistant(内容提示)可以用来帮助程序员尽快完成源代码。可以说这个功能是一个代码编辑器必须具备的功能,不然一个编辑器也无法称为代码编辑器了。本文探讨内容提示的相关概念并给出具体示例。

Content Assistant

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

图1. Java 编辑器的内容提示
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 = 3;
b = 4;


那么当用户在激活内容提示时,我们将显示出 a 和 b 供它选择,也就是显示之前声明过的变量。所有的声明过的变量可以通过遍历语法树来得到,我们在 TreeHelper 里面有一个 getVariables,它会完成这样的功能,如果你生成的语法树不一样,调整这个方法就可以了。注意输入的时候语法必须是正确的,不然语法解析器识别不出这是一个声明语句,也就得不到变量了。

IContentAssistProcessor

第一步,我们要实现 IContentAssistProcessor 接口,它就是所有 Proposal 的来源。不过这个接口的方法比我们想象的要多一些:

清单 2. IContentAssistProcessor 接口

                public interface IContentAssistProcessor {
ICompletionProposal[] computeCompletionProposals(ITextViewer viewer, int offset);
IContextInformation[] computeContextInformation(ITextViewer viewer, int offset);
char[] getCompletionProposalAutoActivationCharacters();
char[] getContextInformationAutoActivationCharacters();
String getErrorMessage();
IContextInformationValidator getContextInformationValidator();
}


这些方法牵涉到了一些概念,我们来一一的解释它们:

    * computeCompletionProposals:这个就是所有 Proposal 的来源了,返回的类型是 ICompletionProposal 数组,ICompletionProposal 代表的就是单个的自动完成选项。
    * computeContextInformation;Context Information(上下文信息)是个新概念,它在这里表示你选择了某个 Proposal 之后,会有一个提示信息弹出来,那个就叫上下文信息。要注意它和上面提到过的 Additional Info 是不同的东西。
    * getCompletionProposalAutoActivationCharacters:这个方法引入了一个 Auto Activation(自动激活)的概念,所谓自动激活就是在某种条件下 Proposal Popup 自动弹出。这个“某种条件”指的是一些字符,比如最常用的应该是“.”号。
    * getContextInformationAutoActivationCharacters:上下文信息也有自动激活的功能
    * getErrorMessage:如果内容提示无法找到任何 Proposal,它可以返回一个错误信息给用户
    * getContextInformationValidator:上下文信息是可以进行校验的,如果失败,上下文信息不会被显示

computeCompletionProposals 方法显然是必须实现的,我添加了一个 ExprContentAssistProcessor 类,下面是它的实现方式:

清单 3. ExprContentAssistProcessor 实现了 IContentAssistProcessor 接口

               
public ICompletionProposal[] computeCompletionProposals(ITextViewer viewer, int offset) {
// get document
IDocument doc = viewer.getDocument();

// get tree
Tree tree = TreeManager.getTree(doc);
if(tree == null)
return null;

// get current selected range
Point range = viewer.getSelectedRange();

// get all declared variables
List<String> variables = TreeHelper.getVariables(tree);

// create proposals
List<ICompletionProposal> proposals = new ArrayList<ICompletionProposal>();
  for(String var : variables) {
proposals.add(new CompletionProposal(
  var, range.x, range.y, var.length(), null, var, null, "Add your info here"));
}
return proposals.toArray(new ICompletionProposal[proposals.size()]);
}


我们遍历语法树得到了所有的变量,你可以看到整个实现代码在 ANTLR 以及一些工具类的帮助下显得非常简洁。注意我们为每个变量创建了一个 CompletionProposal,它的构造函数参数非常多,最后一个就是 Additional Info,我这里只是填了一些无用的信息作为演示之用。其它的参数涉及自动完成需要的所有信息,比如插入的字符串,在哪里插入,图标等等。

配置

又到了将我们的实现和 JTF 连接起来的时间,还是修改 ExprConfiguration, 要覆盖的方法变成了 getContentAssistant:

清单 4. 让 JTF 知道我们的内容提示实现

               
public IContentAssistant getContentAssistant(ISourceViewer sourceViewer)
{
      ContentAssistant assistant = new ContentAssistant();
      assistant.setInformationControlCreator(new IInformationControlCreator() {
      public IInformationControl createInformationControl(Shell parent)
      {
      DefaultInformationControl control = new DefaultInformationControl(parent);
      return control;
      }
      });

      // add assist processor
      IContentAssistProcessor processor = new ExprContentAssistProcessor();
      assistant.setContentAssistProcessor(processor, IDocument.DEFAULT_CONTENT_TYPE);
     
      return assistant;
}


注意我们第一步实现的只是一个 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 的立场。
分享到:
评论

相关推荐

    JFace Text Framework教程

    总结来说,IBM的JFace Text Framework教程为开发者提供了一个强大且灵活的工具,用于构建功能丰富的文本编辑器。通过学习和实践,开发者可以掌握如何利用这个框架来提升开发效率,创建出专业级的IDE(集成开发环境)...

    eclipse与MyEclipse代码提示上屏修改后jar包(org.eclipse.jface.text_3.8.101.v20130802-1147)

    标题提及的"org.eclipse.jface.text_3.8.101.v20130802-1147"是Eclipse中的一个关键组件,它负责文本编辑器的相关功能,包括代码提示。 `org.eclipse.jface.text`是Eclipse JFace库的一部分,JFace是构建用户界面的...

    jface.text 3.12.1.zip

    本文将围绕"jface.text 3.12.1.zip"这一特定的Eclipse插件进行深入探讨,该插件是Eclipse文本编辑器的重要组成部分,对于提升Java代码补全体验有着至关重要的作用。 首先,"jface.text"是Eclipse的Java Face (JFace...

    org.eclipse.jface.text_3.12.2.v20180112-1341.zip

    标题中的"org.eclipse.jface.text_3.12.2.v20180112-1341.zip"是Eclipse插件的一个版本,它专注于文本编辑功能。Eclipse是一个开源的集成开发环境(IDE),而JFace是Eclipse的一部分,提供了面向用户的界面组件。...

    org.eclipse.jface.text_3.12.2.v20180112-1341.jar

    Eclipse JFace Text是Eclipse IDE中的一个核心组件,它提供了高级文本编辑功能,广泛用于构建用户界面中的文本编辑器。这个特定的版本号表明它是该组件的一次更新迭代,可能包含错误修复、性能提升或其他新特性。 ...

    org.eclipse.jface.text_3.11.2.v20170220-1911最新修改版

    JFace Text是Eclipse中的一个API,它是构建文本编辑器的基础,提供了文本编辑、搜索、替换、高亮等基本功能。在这个特定的版本中,开发者们对JFace Text进行了优化,特别是对于用户输入行为的处理,比如取消了空格和...

    基于WTP开发自定义的JSP编辑器

    了解和利用WTP中与插件开发相关的其他关键概念,如JFace Text Framework和Eclipse Editor Framework,可以帮助进一步优化编辑器的性能和用户体验。 #### 五、总结 基于WTP开发自定义的JSP编辑器是一个综合性的工程...

    org.eclipse.jface.text_3.8.2.v20121126-164145源码

    深入研究这部分源码,不仅可以提升对Eclipse JFace的理解,也有助于提升开发者在构建文本编辑器组件时的定制能力。同时,这也提醒我们,源码的修改应遵循良好的编程实践,确保软件的稳定性和可维护性。

    org.eclipse.jface.text_3.7.2.v20111213-1208.jar

    《Eclipse JFace Text库与MyEclipse 10.0.0的编辑器功能优化》 在IT行业中,开发工具的效率与便利性对于程序员的工作至关重要。Eclipse作为一款广受欢迎的开源集成开发环境(IDE),其强大的扩展性和自定义能力一直...

    Eclipse SWT/JFace 核心应用的全部源代码

    JFace包括数据绑定、视图管理、对话框和服务等组件,它将许多复杂的SWT操作进行了封装,例如表格和树的数据模型、表单编辑器等。通过研究JFace的源代码,开发者可以学习到如何优雅地组织用户界面,提高代码的可读性...

    org.eclipse.jface.text_3.9.2.v20141003-1326.zip

    这个插件是JFace Text组件的一部分,JFace是Eclipse工作台(Workbench)框架的一个子项目,它提供了基于SWT(Standard Widget Toolkit)的高级用户界面构建工具。版本号"3.9.2"表明这是该插件的第三个主要版本的第九...

    MyEclipse10.7.1代码补全vs风格 jface.text修正版

    因为10.7.1已经完美破解(修正了war文件到处问题),故放上该版本修改过的org.eclipse.jface.text_3.7.2.v20111213-1208.jar 代码补全空格不上屏,适合习惯使用vs等工具的用户 详细描述参见10.0时的插件 ...

    jface源代码不容错过

    7. **视图和编辑器的实现**:Eclipse RCP中的视图和编辑器是通过JFace实现的,源代码能帮助我们理解这两者如何在RCP环境中协同工作。 总之,研究JFace的源代码不仅可以帮助开发者深入理解Eclipse RCP的工作原理,还...

    JFace教程

    **JFace** 是 Eclipse 平台的一个重要组成部分,它提供了一套丰富的 API 来帮助开发者构建基于 SWT(Standard Widget Toolkit)的用户界面。JFace 的核心目标是简化 GUI 开发流程,使开发者能够更加专注于应用程序...

    Eclipse SWT/JFACE 核心应用 配套源代码

    4. **源代码分析**:通过研究这些源代码,开发者可以学习到如何初始化SWT和JFace工作台,如何创建窗口和视图,如何添加控件并处理用户交互,以及如何组织和管理应用程序的数据。此外,代码中可能还包括错误处理和...

    Eclipse SWT&JFace开发实战精解 源代码

    Eclipse SWT(Standard Widget Toolkit)和JFace是Java编程中用于构建桌面应用程序的两个关键库。它们都是Eclipse IDE的一部分,提供了丰富的用户界面组件和高级的UI设计工具。本篇文章将深入探讨这两个库的核心概念...

    SWT,JFACE核心应用源代码,第二部分.rar

    在"SWT,JFACE核心应用源代码,第二部分.rar"这个压缩包中,我们可以预期找到一系列关于SWT和JFace实际应用的源代码示例。这些示例可能涵盖了如何使用SWT创建基本和复杂的用户界面元素,如何利用JFace进行数据绑定,...

    swt JFace的API文档

    7. **Views and Editors**:JFace 提供了用于构建工作台视图和编辑器的框架,支持多窗口和多页编辑环境。 8. **Control Adapters**:这些适配器类可以添加额外的功能,例如在控件上添加右键菜单。 9. **Bindings ...

    org.eclipse.jface.text_3.12.0.v20170523-1043.jar

    总的来说,Eclipse JFace Text模块是Eclipse IDE中不可或缺的一部分,它提供了丰富的文本编辑功能,尤其是自动补全特性,大大提升了开发效率。版本3.12.0.v20170523-1043的JFace Text可能包含了一些优化和改进,使得...

Global site tag (gtag.js) - Google Analytics