简介: 本文是GEF进阶的第四部分,主要描述了Locator的概念和使用方法。Locator是 一个图形定位器,用来动态的决定某个图形相对于另外一个图形的位置,因此可以用来构造一些 复杂的图形或者实现一些比较有趣的功能。由于Eclipse 3.3已经发布,本文的示例代码是在 Eclipse 3.3, GEF 3.3运行调试的。
本文是GEF进阶的第四部分,主要描述了Locator的概念和使用方法。Locator是一个图形定 位器,用来动态的决定某个图形相对于另外一个图形的位置,因此可以用来构造一些复杂的图形或者 实现一些比较有趣的功能。由于Eclipse 3.3已经发布,本文的示例代码是在Eclipse 3.3, GEF 3.3 运行调试的。
Locator,顾名思义,是一个定位器。我们先来看看这个接口:
public interface Locator {
void relocate(IFigure target);
}
这个接口非常简单,只有一个方法,参数是一个IFigure,所以首先可以明确的是:Locator是一个 图形定位器。那么一个Locator用在什么地方,又为什么要用Locator呢?一般来讲,如果你希望一个图形 能跟着另外一个图形移动,那么Locator就很有用了。因此Locator是相对定位的有利工具。我们下面来介 绍Locator的几个典型应用。
我们在运行GEF shapes的例子时,可以看到图形之间可以有连线,但是连线上没有文字说明,如果有 文字说明的话,会有如下图所示的效果:
从图1看到连线上有了一个“Label”的字样,你可以把它叫做Connection Label(连线标签)。总之, 这个标签是附着在连线上的。不光如此,如果你拖动连线,这个标签也会跟着移动,这就是Locator 的功劳。我们先来修改shapes例子的代码,给连线加上这样的标签。要修改的地方在ConnectionEditPart 的createFigure方法里,修改后如下所示:
protected IFigure createFigure() {
final PolylineConnection connection = (PolylineConnection) super.createFigure();
connection.setTargetDecoration(new PolygonDecoration()); // arrow at target endpoint
connection.setLineStyle(getCastedModel().getLineStyle()); // line drawing style
// add a label
final Label label = new Label("Label");
label.setOpaque(true);
connection.add(label, new MidpointLocator(connection, 0));
return connection;
}
加这么一个label确实很简单,我们先创建一个Label对象,再将它加到连线里面,所以本质上Label是连线的 一个孩子。在添加的时候,我们使用了MidpointLocator, 这是GEF缺省带的一个Locator,作用是把图形定位 到连线的中点。所以我们看到Label的中点始终和连线的中点相同。而连线的中点又叫做MidpointLocator的 reference point(参考点), 我们在本系列的第一部分里面介绍过Anchor(锚点),这个参考点的概念与其是类似的。
随之而来我们会发现一个局限性:这个Label永远只能在连线的中心,无法移动到其它的地方。在有些时候这 确实是个问题:比如某个图特别复杂,连线特别多,以致于连线上的标签都重叠在了一起,可是由于它不能拖动,所 以看上去很不美观。如果我们希望这个Label能被拖动该怎么办呢?有两个方案:
- 为这个Label创建一个EditPart,负责控制这个label的移动,改变大小,等等。
- 使用一个自定义的Locator,并处理Label的鼠标事件,随时刷新它的位置
第一种方法需要的代码稍多且不符合本文主题。我们来看看第二种方法该如何实施。
首要的问题是我们的Locator如何工作呢?我们已经使用了MidpointLocator,它可以随时定位到连线的中点, 如果我们给它加一个偏移,不就可以实现拖动到任何位置的功能了吗?这个概念如图2所示:
策略定下来之后我们来实现一个MidpointOffsetLocator,它直接继承自MidpointLocator,但是我们给它添 加一个offset的成员变量,这样我们就可以控制标签中点的位置了。代码如下所示:
清单3. MidpointOffsetLocator的实现 public class MidpointOffsetLocator extends MidpointLocator {
private Point offset;
public MidpointOffsetLocator(Connection c, int i) {
super(c, i);
offset = new Point(0, 0);
}
@Override
protected Point getReferencePoint() {
Point point = super.getReferencePoint();
return point.getTranslated(offset);
}
public Point getOffset() {
return offset;
}
public void setOffset(Point offset) {
this.offset = offset;
}
}
关键的方法在于我们覆盖了getReferencePoint(),让它返回之前加上我们的偏移量。就这么简单,我们自己的Locator诞生了!
Locator有了,下面的问题就是如何才能在适当的时机修改这个偏移量呢?第一想到的就是鼠标事件监听器。由于Figure本身 已经支持添加各种各样的监听器,所以这一步也非常简单。我们继续修改ConnectionEditPart的createFigure方法,给我们的Label 加上鼠标事件处理方法,如下所示:
清单4. 添加鼠标事件监听器
protected IFigure createFigure() {
// 省略其它无关代码
......
// add a label
final Label label = new Label("Label");
label.setOpaque(true);
connection.add(label, new MidpointOffsetLocator(connection, 0));
label.addMouseListener(new MouseListener() {
public void mouseDoubleClicked(MouseEvent me) {
}
public void mousePressed(MouseEvent me) {
anchorX = me.x;
anchorY = me.y;
me.consume();
}
public void mouseReleased(MouseEvent me) {
me.consume();
}
});
label.addMouseMotionListener(new MouseMotionListener() {
public void mouseDragged(MouseEvent me) {
dx += me.x - anchorX;
dy += me.y - anchorY;
anchorX = me.x;
anchorY = me.y;
Object constraint = connection.getLayoutManager().getConstraint(label);
if(constraint instanceof MidpointOffsetLocator) {
((MidpointOffsetLocator)constraint).setOffset(new Point(dx, dy));
label.revalidate();
}
me.consume();
}
// 省略其它无关代码
......
});
return connection;
}
要注意的有三点,第一我把MidpointLocator替换成了我们自己的Locator,第二我在ConnectionEditPart加了四个成员变量(dx, dy, anchorX, anchorY)来跟踪鼠标位置,第三我在鼠标拖动事件中修改偏移量并刷新它,但是为了安全起见,我先判断了constraint类型。
我们现在完成了Label的拖动功能,不过你可以发现一些可以继续提高的地方,最明显的莫过于Label的位置不能保存。我们应该把这个偏移量保存到Connection模型中去。这个问题和本文无关,留给读者做个练习。本小节的示例代码在org.eclipse.gef.examples.shapes_locator_step1.zip中,大家可以看看实际的效果。
Locator另一个典型的应用是用在handle中,所谓handle,就是你选择一个图形之后,在它的边框周围出现的一些辅助性的图形,比如一些小方块。如下图所示:
因为Handle也使用了Locator,所以我们在拖动图形的时候,Handle的位置也会随之更新。回头看看上一小节中的Label,可以发现一个美中不足的是:Label的周围没有handle,这样的话用户可能会不知道我们的Label是可以拖动的,对用户不太直观,如果能在连线被选择的时候把Label的周围也加上handle,那么用户可以容易的发现这个Label原来也支持拖动,这有利于提高用户友好度。本小节的例子就来实现这个功能。
MyConnectionEndpointEditPolicy
追究handle的发源地,会发现连线上的handle是在ConnectionEndpointEditPolicy中创建的, 我们要添加handle,自然就是要实现一个自己的EditPolicy了。关于EditPolicy的相关概念,这里不做赘述。观察ConnectionEndpointEditPolicy的实现,可以看到一个有趣的方法createSelectionHandles,那么我们来覆盖它,如下:
清单5. 为Label添加Handle public class MyConnectionEndpointEditPolicy extends ConnectionEndpointEditPolicy {
@Override
protected List createSelectionHandles() {
List handles = super.createSelectionHandles();
List<IFigure> children = (List<IFigure>)getHostFigure().getChildren();
for(IFigure figure : children) {
if(figure instanceof Label)
handles.add(new MoveHandle((GraphicalEditPart)getHost(),
new MoveHandleLocator(figure)));
}
return handles;
}
}
我们遍历了连线的所有孩子,如果它是一个Label,就为它创建一个MoveHandle,MoveHandle使用了MoveHandleLocator,这些类都是GEF自带的,不需要费什么功夫。
有了自定义的EditPolicy,别忘了在ConnectionEditPart中替换原来的EditPolicy,这部分代码过于简单,省略不提。
到这里为止,我们的Label就有了handle了,如下图所示:
MoveHandle的效果是在Label周围画了一个矩形框, 表明这个图形可以移动。不过我们还有一个小问题:点击Label的时候MoveHandle不出来,要解决这个问题也很简单,在Label的鼠标点击事件中增加一行getViewer().appendSelection(ConnectionEditPart.this)就可以了。
本小节的示例代码在org.eclipse.gef.examples.shapes_locator_step2.zip中。我们仍然有很多可以提高的地方,首先,我们用的是缺省的MoveHandle,如果不喜欢Label周围这个矩形或者矩形的位置,可以自定义一个Handle,使用自定义的Locator. 其次,我们的Label只能拖动,不能改变大小,这牵涉到为Label添加Resize Handle并使用相应的Locator,这些内容都留给有兴趣的读者作为练习。
GEF中的几乎一切东西都可以定制,本文介绍的是Locator的定制,Locator是相对布局的关键接口。灵活使用Locator,可以实现一些很有趣的功能。