我编写了一个程序,该程序通过文本框,按钮和文件对话框从用户那里获取多个输入,并将其格式化为SWT树,用户可以在其中通过复选框选择元素。
我面临的问题是我希望它为用户实时更新,并且我在这个论坛中找到了一个关于使用容器的layout()
方法的解决方案,它可以与我的树一起使用,但前提是在这棵树上调用dispose()
,然后重建它/之后重新绘制它。
在这棵树的后面,我有一个树数据结构,用于管理所有数据。
另外,我正在使用的类实现了SWT的Dialog
接口,因此当我按下按钮时,将显示一个带有树和文本输入区域的窗口。
在该类中,我已经声明了树将驻留在其中的容器。
final Composite container = new Composite(parent, SWT.NONE);
接下来,我调用此静态方法,其中包含树的实际构建过程
createTree(container);
对于用户输入的按钮和文本区域,我还有一些其他代码,但这不会影响程序的行为或引发的异常。
这是静态方法
createTree(container);
public void createTree(final Composite container){
try{
for(Object element : container.getChildren()){
if(element instanceof Tree){
((Tree) element).dispose();
}
}//here I am disposing of the previous tree and then I'm creating a new one to be drawn in the container when the layout() method will be called
final Tree variantTree = new Tree(container, SWT.CHECK | SWT.V_SCROLL | SWT.H_SCROLL);
variantTree.setBounds(10, 65, 400, 400);
//here is where I am populating the tree with the data I have stored in the Tree Data Structure that I've mentioned
if(TestControl.getTree().getChildren().size() > 0){
for(final Node variantElement : TestControl.getTree().getChildren()){
final TreeItem variantTreeItem = new TreeItem(variantTree, 0);
variantTreeItem.setText(variantElement.getName());
variantTreeItem.setChecked(variantElement.getState());
for(Node suiteElement : variantElement.getChildren()){
TreeItem suiteTreeItem = new TreeItem(variantTreeItem, 0);
suiteTreeItem.setText(suiteElement.getName());
suiteTreeItem.setChecked(suiteElement.getState());
for(Node testElement : suiteElement.getChildren()){
TreeItem testTreeItem = new TreeItem(suiteTreeItem, 0);
testTreeItem.setText(testElement.getName());
testTreeItem.setChecked(testElement.getState());
}
}
}
}
//here is the actual problem, the exception's stack trace points to the line where my next comment is. this listener is used to bring a file dialog window where I can select a file and use it later on
variantTree.addListener(SWT.MouseDoubleClick, new Listener(){
@Override
public void handleEvent(Event event) {
try{
// TODO Auto-generated method stub
Point point = new Point(event.x, event.y);
if(!point.equals(null)){
TreeItem item = variantTree.getItem(point);
for(Node element : TestControl.getTree().getChildren()){
if(element.getName().equals(item.getText())){//here is the problem, why is it trying to tell me
FileDialog fileDialog = new FileDialog(container.getParent().getShell(), SWT.OPEN);
//filtering for extensions
//filtering for path
String path;
if((path = fileDialog.open()) != null){
String fileName = fileDialog.getFileName();
Node suiteNode = new Node(element);
suiteNode.setName(fileName);
TestControl.addChild(suiteNode);
createTree(container);
//here I call the method in a recursive way. After I modified my Tree Data Structure with the data I got from the user, I want to redraw the tree in a real timed fashion
}
}
}
}
}catch(Exception exception){
exception.printStackTrace();
Status status = new Status(IStatus.ERROR, Activator.PLUGIN_ID, exception.getLocalizedMessage(), exception);
ErrorDialog.openError(null, "Error", "Error occured!", status);
}
}
});
variantTree.addListener(SWT.Selection, new Listener(){
@Override
public void handleEvent(Event event) {
// TODO Auto-generated method stub
if(event.detail == SWT.CHECK){
TreeItem item = (TreeItem) event.item;
for(Node element : TestControl.getTree().getChildren()){
if(element.getName().equals(item.getText())){
element.setState(item.getChecked());
}
}
for(Node element : TestControl.getTree().getChildren()){
for(Node nextElement : element.getChildren()){
if(nextElement.getName().equals(item.getText())){//here the error doesnt show up, even though I am using the SWT Tree element as above
nextElement.setState(item.getChecked());
}
}
}
for(Node element : TestControl.getTree().getChildren()){
for(Node nextElement : element.getChildren()){
for(Node lastElement : nextElement.getChildren()){
if(lastElement.getName().equals(item.getText())){
lastElement.setState(item.getChecked());
}
}
}
}
}
}
});
}catch(Exception exception){
exception.printStackTrace();
Status status = new Status(IStatus.ERROR, Activator.PLUGIN_ID, exception.getLocalizedMessage(), exception);
ErrorDialog.openError(null, "Error", "Error occured!", status);
}
}
我还读到可能会出现此错误,因为当我调用
dispose
时,我也应该摆脱监听器。是否有必要,这可能是异常的来源?
谢谢,对不起,拥抱代码部分。
最佳答案
我假设在显示createTree
后调用FileDialog
方法时发生错误。
调用createTree
时,您将处于以下循环中:
for(Node element : TestControl.getTree().getChildren()){
因此,在
createTree
调用之后,您立即回到循环的开始并运行if(element.getName().equals(item.getText()))
但是这里的
item
指的是您刚刚处置的TreeItem
中的Tree
,因此它不再有效,并且会出现“处置小部件”错误。调用
createTree
后,必须立即停止循环,并且不要在现有树上执行任何其他操作。一个break
来停止循环就足够了: createTree(container);
break;
注意:您不必丢弃
Tree
,只需删除TreeItem
并添加新的即可。