WEB开发网
开发学院软件开发Java 基于 JFace Text Framework 构建全功能代码编辑器... 阅读

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

 2010-03-18 00:00:00 来源:WEB开发网   
核心提示:HyperlinkHyperlink(超链接)在 Java 编辑器中用来进行快速的代码定位,当你按住 Ctrl 键并把鼠标指向一个函数名的时候,基于 JFace Text Framework 构建全功能代码编辑器: 第 8 部分:Hyperlink,函数名会显示为超链接,点击之后代码会跳转到函数的声明处,比如如果一个变

Hyperlink

Hyperlink(超链接)在 Java 编辑器中用来进行快速的代码定位,当你按住 Ctrl 键并把鼠标指向一个函数名的时候,函数名会显示为超链接,点击之后代码会跳转到函数的声明处。这个功能使得 Eclipse 浏览代码很方便,这次我就来介绍如何在自己的编辑器中添加超链接功能。

超链接的定位

编辑器不会知道哪块区域应该显示为超链接,这是通过 IHyperlinkDetector 接口实现的。这里牵涉到语义方面的内容,因为你必须要能知道鼠标下面到底是个什么。在解析器那一层需要实现这样的支持。

超链接的渲染

JTF 是如何显示超链接的?可能你会想到标注,缺省的实现不是这样。这里要介绍另外一个接口:IHyperlinkPresenter。JTF 缺省的时候是用 StyledText 的 StyleRange 实现的,其实就是把超链接的那块文字置为蓝色且带下划线。但是因为有了这么一个接口,你可以把超链接弄成任何样子。

实现超链接

本文要实现的超链接功能是:点击某个变量名,编辑器会选中声明该变量的那条语句。

底层支持

底层需要支持两个功能:判断某个位置是一个变量,以及得到变量声明的语句范围。由于我的例子很简单,判断是不是变量也非常简单,只要符号类型是一个 ID 类型就行了。得到变量声明的语句范围需要检查语法树,因为变量声明的子树以等号为根节点,所以找到对应的根节点就行了。然后从等号开始得到子树的最左和最右节点,从而计算出整个子树的字符范围。这些代码已经添加到了 TreeHelper 中,具体请参看 getVariableDeclaration 和 getTreeRange 方法。

实现 IHyperlink

我实现了一个 VariableHyperlink 来封装超链接信息,它最重要的方法是 open(),因为它会在你点击超链接后被调用:

清单1. VariableHyperlink 的 open 方法

  
public void open() { 
 // get doc 
 IDocument doc = viewer.getDocument(); 
 
 // get tree 
 Tree tree = TreeManager.getTree(doc); 
 
 // get variable declaration range 
 Point range = TreeHelper.getVariableDeclaration(tree, variable); 
 
 // select text 
 if(range != null) { 
 viewer.setSelectedRange(range.x, range.y); 
 viewer.revealRange(range.x, range.y); 
 } 
} 

我用到了刚才提到的 TreeHelper 中的方法来得到声明语句的范围,剩下的事情就比较直接了,选择这个范围并让确保其在编辑器中可见。

实现 IHyperlinkDetector

ExprHyperlinkDetector 完成了发现变量的功能,然后把变量信息包装在 VariableHyperlink 中。在之前的文章中,我已经不止一次的使用了 TokenList 来得到某个偏移处的符号,ExprHyperlinkDetector 也依赖于 TokenList,所以这里不详细解释了。

配置

覆盖 ExprConfiguration 的 getHyperlinkDetectors,让它返回 ExprHyperlinkDetector。它的返回值是一个数组,所以你可以安装多个 IHyperlinkDetector 实例。对于这个简单的小例子,并无必要使用多个 IHyperlinkDetector 实例。

结束语

超链接是 JTF 中相对简单的一个功能了,但是不要忘了,我是没有实现全部的功能的:

是不是一定需要按住 Ctrl 键才能激活超链接功能呢?不是,请看看 SourceViewerConfiguration 的 getHyperlinkStateMask 方法。

试试看使用 IHyperlinkPresenter 实现一个自定义的超链接样式,这里牵涉到 HyperlinkManager 这个类的使用。这个功能较为复杂一些,需要这样做的时候也不多,没有时间的读者不妨看看 DefaultHyperlinkPresenter 类的实现。

例子还是有 bug 的,比如如果一个变量声明了两次,则只会选择第一条声明语句。可以考虑显示一个浮动窗口来选择跳转到哪个,你想到了什么解决方案呢?

本文示例源代码或素材下载

Tags:基于 JFace Text

编辑录入:爽爽 [复制链接] [打 印]
赞助商链接