閑話稍說,直接上bug視頻:
Bug視頻
大致現象:在一個可編輯tableviewer的單元格中,雙擊,對應的編輯對話框會不時彈出多次。我這邊嚴重的時候,會連續彈出將近20次,那就只能修改了~_~
Bug分析:
調試了一把,發現是SWT底層將同一鼠標事件(通過event.time可以確認)連續發送了多次。而JFace中的TableEditorImpl將這事件轉發給了tableviewer,tableviewer會進一步調用對應的double click listener。開發人員做的double click listener實現就是彈出一個編輯對話框。問題就出現了,當彈出的對話框關閉之后,馬上有彈出了,上面說過,重復事件又被廣播了~_~
修改方法:
時間較緊,就直接在TableEditorImpl這邊修改了一把。將org.eclipse.jface插件以源碼工程方式導入,將TableEditorImpl.java類做了一定的修改,加了一個雙擊事件的時間戳判斷。
【TableEditorImpl.java原有代碼,Eclipse 3.2版本】
mouseListener = new MouseAdapter() {
public void mouseDown(MouseEvent e) {
// time wrap?
// check for expiration of doubleClickTime
if (e.time <= doubleClickExpirationTime) {
control.removeMouseListener(mouseListener);
cancelEditing();
handleDoubleClickEvent();
} else if (mouseListener != null) {
control.removeMouseListener(mouseListener);
}
}
};
【修改后代碼】
1、首先在TableEditorImpl中定義了一個雙擊事件的時間戳,緩存上次處理的雙擊事件的時間戳
2、在轉發雙擊事件之前,做時間戳檢查,避免同一雙擊事件被處理了多次
abstract class TableEditorImpl {
...其他代碼省略
//add by zhuxing:上次處理的雙擊事件的時間戳
private int lastDoubleclickEventTime;
private void activateCellEditor() {
..................................
mouseListener = new MouseAdapter() {
public void mouseDown(MouseEvent e) {
// time wrap?
// check for expiration of doubleClickTime
if (e.time <= doubleClickExpirationTime) {
control.removeMouseListener(mouseListener);
cancelEditing();
//檢查本次要處理的雙擊事件是否和上次的事件重復
if (e.time != lastDoubleclickEventTime) {
lastDoubleclickEventTime = e.time;
handleDoubleClickEvent();
}
} else if (mouseListener != null) {
control.removeMouseListener(mouseListener);
}
}
};
}
}
希望能幫到遇到類似問題的同學
本博客中的所有文章、隨筆除了標題中含有引用或者轉載字樣的,其他均為原創。轉載請注明出處,謝謝!