通常,如果不做特殊处理的话,在表头(树头,以下统称表头)的上点击右键和在表体里点击右键,显示的菜单结果一样的。
有时候我希望能显示不一样的菜单,例如在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,映射到表中的具体位置,然后用表头的高度与它的大小关系,来判断当前点击处是否在表头处,如果在表头处,则显示表头菜单,否则显示表体菜单。
这里都假设表头是可见的,如果表头不可见。这篇文章就没意义了。