Draw2d中如何控制ToolTip的持续时间
软件版本:Eclipse3.2, draw2d_3.2.0.v20060626
在Draw2d中,Figure都支持ToolTip(通过调用setToolTip方法),当鼠标停留在其上面时,显示一个标签,以便提供有用的信息给使用者。这个显示的持续时间为5秒。那么对于ToolTip内容比较长的情况,5秒时间不够用,怎么办?这是一个典型的中国式需求 :-P
话说从头。相传SWT与Draw2d连接的桥梁叫做LightweightSystem,而在LightweightSystem中,负责分发SWT事件的,叫做SWTEventDispatcher,这个SWTEventDispatcher呢,使用一个ToolTipHelper的对象来控制ToolTip的显示,而在ToolTipHelper中,那个5秒的持续时间竟然是写死的!——必须承认,做eclipse插件开发,很多时候你都是在阅读他们的源码并且不时摇头叹息——那么,我们的解决办法自然就是:将这个ToolTipHelper的行为替换掉。OK,继承一个:
public class ToolTipHelperX extends ToolTipHelper {
private Timer timer;
private IFigure currentTipSource;
private long duration = 5000;
/**
* Constructs a ToolTipHelper to be associated with Control <i>c</i>.
*
* @param c the control
* @since 2.0
*/
public ToolTipHelperX(Control c) {
super(c);
}
/*
* Calculates the location where the tooltip will be painted. Returns this as a Point.
* Tooltip will be painted directly below the cursor if possible, otherwise it will be
* painted directly above cursor.
*/
private Point computeWindowLocation(IFigure tip, int eventX, int eventY) {
org.eclipse.swt.graphics.Rectangle clientArea = control.getDisplay().getClientArea();
Point preferredLocation = new Point(eventX, eventY + 26);
Dimension tipSize = getLightweightSystem()
.getRootFigure()
.getPreferredSize()
.getExpanded(getShellTrimSize());
// Adjust location if tip is going to fall outside display
if (preferredLocation.y + tipSize.height > clientArea.height)
preferredLocation.y = eventY - tipSize.height;
if (preferredLocation.x + tipSize.width > clientArea.width)
preferredLocation.x -= (preferredLocation.x + tipSize.width) - clientArea.width;
return preferredLocation;
}
/**
* Sets the LightWeightSystem's contents to the passed tooltip, and displays the tip. The
* tip will be displayed only if the tip source is different than the previously viewed
* tip source. (i.e. The cursor has moved off of the previous tooltip source figure.)
* <p>
* The tooltip will be painted directly below the cursor if possible, otherwise it will be
* painted directly above cursor.
*
* @param hoverSource the figure over which the hover event was fired
* @param tip the tooltip to be displayed
* @param eventX the x coordinate of the hover event
* @param eventY the y coordinate of the hover event
* @since 2.0
*/
public void displayToolTipNear(IFigure hoverSource, IFigure tip, int eventX, int eventY) {
if (tip != null && hoverSource != currentTipSource) {
getLightweightSystem().setContents(tip);
Point displayPoint = computeWindowLocation(tip, eventX, eventY);
Dimension shellSize = getLightweightSystem().getRootFigure()
.getPreferredSize().getExpanded(getShellTrimSize());
setShellBounds(displayPoint.x, displayPoint.y, shellSize.width, shellSize.height);
show();
currentTipSource = hoverSource;
timer = new Timer(true);
timer.schedule(new TimerTask() {
public void run() {
Display.getDefault().syncExec(new Runnable() {
public void run() {
hide();
timer.cancel();
}
});
}
}, duration);
}
}
/**
* Disposes of the tooltip's shell and kills the timer.
*
* @see PopUpHelper#dispose()
*/
public void dispose() {
if (isShowing()) {
timer.cancel();
hide();
}
getShell().dispose();
}
/**
* @see PopUpHelper#hookShellListeners()
*/
protected void hookShellListeners() {
// Close the tooltip window if the mouse enters the tooltip
getShell().addMouseTrackListener(new MouseTrackAdapter() {
public void mouseEnter(org.eclipse.swt.events.MouseEvent e) {
hide();
currentTipSource = null;
timer.cancel();
}
});
}
/**
* Displays the hover source's tooltip if a tooltip of another source is currently being
* displayed.
*
* @param figureUnderMouse the figure over which the cursor was when called
* @param tip the tooltip to be displayed
* @param eventX the x coordinate of the cursor
* @param eventY the y coordinate of the cursor
* @since 2.0
*/
public void updateToolTip(IFigure figureUnderMouse, IFigure tip, int eventX, int eventY) {
/* If the cursor is not on any Figures, it has been moved
off of the control. Hide the tool tip. */
if (figureUnderMouse == null) {
if (isShowing()) {
hide();
timer.cancel();
}
}
// Makes tooltip appear without a hover event if a tip is currently being displayed
if (isShowing() && figureUnderMouse != currentTipSource) {
hide();
timer.cancel();
displayToolTipNear(figureUnderMouse, tip, eventX, eventY);
} else if (!isShowing() && figureUnderMouse != currentTipSource)
currentTipSource = null;
}
public void setDuration(long duration) {
this.duration = duration;
}
}
这段代码不短,但它其实可以很短,原因是作者没考虑到会有人想要继承它,以致我们无法直接覆盖displayToolTipNear方法了事,于是我干脆就把ToolTipHelper的代码全拷过来贴上。
然后,再来看SWTEventDispatcher,发现,它的这个toolTipHelper成员无法注入(没有setter),用的时候是直接在getter里new出来的;不过还好,这个getter是protected,也就是说,在设计上考虑了继承的可能性,这样代码就会简单一些了。来来来,再继承一个:
public class SWTEventDispatcherX extends SWTEventDispatcher {
private ToolTipHelperX toolTipHelperX;
private long tooltipDuration;
public SWTEventDispatcherX(long tooltipDuration) {
super();
this.tooltipDuration = tooltipDuration;
}
protected ToolTipHelper getToolTipHelper() {
if (toolTipHelperX == null) {
toolTipHelperX = new ToolTipHelperX(control);
toolTipHelperX.setDuration(tooltipDuration);
}
return toolTipHelperX;
}
}
OK了,还剩下最后一个,传说中的LightweightSystem。这个东西好,他直接提供了SWTEventDispatcher的注入器(就是setter了,说注入器显得唬人一些),就不需要我们再做什么了,小功告成!
譬如说,现在要在一个名叫parent的Composite上放置一个FigureCanvas,做为其它Figure的容器,就可以这样写:
LightweightSystem lws = new LightweightSystem();
lws.setEventDispatcher(new SWTEventDispatcherX(1800000L));//半小时
FigureCanvas canvas = new FigureCanvas(parent, SWT.DOUBLE_BUFFERED, lws);
canvas.setContents(yourTopMostFigure);
...
如此一来,这个FigureCanvas上的Figure的ToolTip都会持续显示半小时。