我想做以下事情,我需要一些建议来了解最佳方法。
我有一个JList,通过单击添加(+)按钮显示用户添加的文件,并在用户单击删除( - )按钮时删除文件。对于每个添加的文件,我想关联一个应该指示文件状态的图标。例如,如果用户只添加了文件而没有运行文件(我有另一个JButton用于运行带有所选文件的应用程序),那么这个图标应该是红色的,一旦用户运行它,该图标应该变为绿色。此外,如果用户通过单击( - )按钮删除文件,它也应删除与该特定文件关联的图标。下面是我想要的图形表示。
我正在考虑将ImageIcon与每个添加的文件相关联,但我不确定如何更改其外观以显示状态。我也不确定删除文件后如何删除ImageIcon。有没有其他方法(除了ImageIcon)呢?任何帮助/建议表示赞赏。
在编程中,数据是王道。如何表示数据的表示不应该考虑数据,即UI /视图层的域/责任。
这通常由model-view-controller pattern表示
在您的示例中,您有两条(基本)信息。文件和状态(不运行,运行,删除),您希望将此信息组合为“数据”。在Java中,这通常意味着普通的旧Java对象(或Pojo)
因为状态只有有限的可能性,我们可以使用enum
来表示它,从而限制有效值
public enum FileStatus {
NOT_RUN, RUN, DELETED;
}
然后我们可以创建自己的pojo ...
public class FileOperation {
private File file;
private FileStatus status;
public FileOperation(File file, FileStatus status) {
this.file = file;
this.status = status;
}
public FileOperation(File file) {
this(file, FileStatus.NOT_RUN);
}
public File getFile() {
return file;
}
public FileStatus getStatus() {
return status;
}
public void setStatus(FileStatus newStatus) {
if (status == newStatus) {
return;
}
this.status = newStatus;
}
}
现在,当我们想知道文件的状态时,我们知道从哪里获取它。
但是JList
怎么样?你问,好问题。我们真正想要的是某种方式,当任何JList
对象的状态发生变化时,可以通知FileOperation
。
现在,你可以迭代ListModel
,但这不是一个非常干净的解决方案,更好的解决方案是允许FileOperation
在它发生变化时生成事件并让ListModel
监听它们并采取它自己的行动。
这是observer patternƒ的基本概念
你可以通过多种方式实现这一目标,但我很懒,所以我只想使用可用的属性更改API
public class FileOperation {
private File file;
private FileStatus status;
private PropertyChangeSupport propertyChangeSupport = new PropertyChangeSupport(this);
public FileOperation(File file, FileStatus status) {
this.file = file;
this.status = status;
}
public FileOperation(File file) {
this(file, FileStatus.NOT_RUN);
}
public File getFile() {
return file;
}
public FileStatus getStatus() {
return status;
}
public void setStatus(FileStatus newStatus) {
if (status == newStatus) {
return;
}
FileStatus oldStatus = status;
status = newStatus;
propertyChangeSupport.firePropertyChange("status", oldStatus, status);
}
public void addPropertyChangeListener(PropertyChangeListener listener) {
propertyChangeSupport.addPropertyChangeListener(listener);
}
public void removePropertyChangeListener(PropertyChangeListener listener) {
propertyChangeSupport.removePropertyChangeListener(listener);
}
}
现在我们需要一个可以响应它的ListModel
......
public class FileOperationListModel extends AbstractListModel<FileOperation> {
private List<FileOperation> items = new ArrayList<FileOperation>(25);
private PropertyChangeListener handler = new PropertyChangeHandler();
public void add(FileOperation fo) {
fo.addPropertyChangeListener(handler);
int size = items.size();
items.add(fo);
fireIntervalAdded(this, size, size);
}
public void remove(FileOperation fo) {
int index = items.indexOf(fo);
if (index < 0) {
return;
}
fo.removePropertyChangeListener(handler);
items.remove(fo);
fireIntervalRemoved(this, index, index);
}
@Override
public int getSize() {
return items.size();
}
@Override
public FileOperation getElementAt(int index) {
return items.get(index);
}
public class PropertyChangeHandler implements PropertyChangeListener {
@Override
public void propertyChange(PropertyChangeEvent evt) {
if (!(evt.getSource() instanceof FileOperation)) {
return;
}
FileOperation fo = (FileOperation) evt.getSource();
int index = items.indexOf(fo);
fireContentsChanged(FileOperationListModel.this, index, index);
}
}
}
现在,最后一块拼图,你需要一个可以显示你想要的信息的自定义ListCellRenderer
。
为此,你将不得不首先阅读How to use lists和Writing a Custom Cell Renderer