[英]Why won't JComboBox display Strings from custom model?
我为JCombobox创建了一个自定义模型,以使其与LinkedHashMap同步。 我想创建一个可以与任何LinkedHashMap重用的模型(因此可以使用泛型)。 当我用我的应用程序运行它时,字符串无法出现。 当我单击“字符串”应位于的位置时,动作侦听器将正确触发,并且所选项目会更改。 但是,下拉菜单不显示字符串? 我知道它是基于JList的,所以我需要对JLabel等进行任何处理吗? 这也是模型的一部分吗? 我还需要执行其他任何操作吗?
这是我的自定义模型
import java.io.Serializable;
import java.util.ArrayList;
import java.util.Collection;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map.Entry;
import javax.swing.AbstractListModel;
import javax.swing.MutableComboBoxModel;
/**
* A model design to automatically update based of a the key values of a Hashmap
* @author Skylion
*
* @param <K> The generic representing key of the Hashmap
* @param <V> The generic representing the value of the Hashmap
*/
public class HashMapComboBoxModel<K, V> extends AbstractListModel<K> implements MutableComboBoxModel<K>,
Serializable {
/**
* Auto-generated
*/
private static final long serialVersionUID = -1293826656458176439L;
/**
* The Selected Item
*/
private K selectedItem;
LinkedHashMap<K,V> data;
public HashMapComboBoxModel(LinkedHashMap<K,V> data){
this.data = data;
}
@Override
public K getElementAt(int index) {
List<Entry<K,V>> randAccess = new ArrayList<Entry<K,V>>((Collection<? extends Entry<K, V>>) data.entrySet());
return randAccess.get(index).getKey();
}
@Override
public int getSize() {
return data.size();
}
@Override
public K getSelectedItem() {
return selectedItem;
}
@SuppressWarnings("unchecked")//Equals() implements the check
@Override
public void setSelectedItem(Object anItem) {
for(K keys: data.keySet()){
if(keys.equals(anItem)){
this.selectedItem = (K) anItem;
return;
}
}
}
@Override
public void addElement(Object obj) {
addElement(downcastToEntry(obj));
}
@SuppressWarnings("unchecked")
private Entry<K,V> downcastToEntry(Object obj){
if(obj instanceof Entry && obj.getClass().isAssignableFrom(
data.entrySet().iterator().next().getClass())){
return (Entry<K,V>)obj;
}
return null;
}
/**
* Adds an Entry value to the hashmap
* @param obj The Entry value you want to add
* @return return true if added false otherwise
*/
public boolean addElement(Entry<K,V> obj){
if(obj == null){return false;}
return this.data.entrySet().add(obj);
}
@Override
public void insertElementAt(Object obj, int index) {
Entry<K,V> entry = downcastToEntry(obj);
addToMapAtIndex(index, entry.getKey(), entry.getValue());
}
private void addToMapAtIndex(int index, K key, V value) {
assert (data != null);
assert !data.containsKey(key);
assert (index >= 0) && (index < data.size());
int i = 0;
List<Entry<K, V>> rest = new ArrayList<Entry<K, V>>();
for (Entry<K, V> entry : data.entrySet()) {
if (i++ >= index) {
rest.add(entry);
}
}
data.put(key, value);
for (int j = 0; j < rest.size(); j++) {
Entry<K, V> entry = rest.get(j);
data.remove(entry.getKey());
data.put(entry.getKey(), entry.getValue());
}
}
@Override
public void removeElement(Object obj) {
data.remove(obj);
}
@Override
public void removeElementAt(int index) {
data.remove(getElementAt(index));
}
}
这是我构造JCombobox的方法。
LinkedHashMap<String, myCustomClass> map = new LinkedHashMap<String, myCustomClass();
//代码...代码...更多代码...
JCombobox box = new JCombobox<String>(new HashMapComboBoxModel<String,myCustomClass>(map));
//稍后在代码中
map.put("myString", myObject);
// JCombobox更新,但下拉菜单变白。
NVM,我知道了。 我需要调用Fire的DataListener方法。 查看JCombobox的源代码后,我需要通过模型使用add方法对其进行正确更新。 我决定不重写类,而只是简单地在getSize()方法上触发侦听器,作为一点小技巧。 我强制在调用该方法时更新整个列表,以便在组合框执行for循环时,它会自动更新模型。
它可能很hack,但是可以用!
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.