通常,如果不做特殊处理的话,在表头(树头,以下统称表头)的上点击右键和在表体里点击右键,显示的菜单结果一样的。
有时候我希望能显示不一样的菜单,例如在Windows的Explorer上,如果在表头点击菜单,显示就是可显示的列项:
在SWT中,缺省是不支持对表头添加菜单的,不过我们可以自己实现这个功能。
在SWT中有一个MenuDetect事件,可以查看源码,在Widget类的showMenu(int x, int y):
boolean showMenu (int x, int y) { Event event = new Event (); event.x = x; event.y = y; sendEvent (SWT.MenuDetect, event); // widget could be disposed at this point if (isDisposed ()) return false; if (!event.doit) return true; Menu menu = getMenu (); if (menu != null && !menu.isDisposed ()) { if (x != event.x || y != event.y) { menu.setLocation (event.x, event.y); } menu.setVisible (true); return true; } return false; }
可以看到,每次菜单显示之前,会先发出一个SWT.MenuDetect事件,之后才是显示具体的Menu。所以如果我们想对表头和表体显示不同的菜单,可以在每次显示菜单前,根据点击位置的不同,设置不同的菜单,最后让它显示出来。
根据以上源码,显然,我们可以增加一个SWT.MenuDetect事件监听,来设置菜单,这个菜单在之后就会被显示出来。
看以下代码:
final Menu bodyMenu = ...; final Menu headerMenu = ... ; table.addListener(SWT.MenuDetect, new Listener() { public void handleEvent(Event event) { Table t = (Table) event.widget; Point pt = t.getDisplay().map(null, t, event.x, event.y); Rectangle clientArea = t.getClientArea(); boolean isHeader = ((pt.y - clientArea.y) <= t .getHeaderHeight()); t.setMenu(isHeader ? headerMenu : bodynMenu); } });
每次,当Menu显示之前,我们把event中的x,y,映射到表中的具体位置,然后用表头的高度与它的大小关系,来判断当前点击处是否在表头处,如果在表头处,则显示表头菜单,否则显示表体菜单。
这里都假设表头是可见的,如果表头不可见。这篇文章就没意义了。